forked from sleepybishop/nanorq
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathencode.c
102 lines (85 loc) · 2.85 KB
/
encode.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
#include "wrapendian.h"
#include <math.h>
#include <stdio.h>
#include <stdlib.h>
#include <time.h>
#include <nanorq.h>
void dump_esi(nanorq *rq, struct ioctx *myio, FILE *oh, uint8_t sbn,
uint32_t esi) {
uint32_t fid = htobe32(nanorq_fid(sbn, esi));
uint16_t packet_size = nanorq_symbol_size(rq);
uint8_t data[packet_size];
uint64_t written = nanorq_encode(rq, (void *)data, esi, sbn, myio);
if (written != packet_size) {
fprintf(stderr, "failed to encode packet data for sbn %d esi %d.", sbn,
esi);
abort();
} else {
fwrite(&fid, 1, sizeof(fid), oh);
fwrite(data, 1, packet_size, oh);
}
}
void dump_block(nanorq *rq, struct ioctx *myio, FILE *oh, uint8_t sbn, float overhead_percent, float drop_percent) {
uint32_t num_esi = nanorq_block_symbols(rq, sbn);
int num_dropped = 0, num_rep = 0;
for (uint32_t esi = 0; esi < num_esi; esi++) {
float dropped = ((float)(rand()) / (float)RAND_MAX);
if (dropped < drop_percent) {
num_dropped++;
} else {
dump_esi(rq, myio, oh, sbn, esi);
}
}
int overhead = (int) (num_esi * overhead_percent);
for (uint32_t esi = num_esi; esi < num_esi + overhead; esi++) {
dump_esi(rq, myio, oh, sbn, esi);
num_rep++;
}
nanorq_encode_cleanup(rq, sbn);
fprintf(stderr, "block %d is %d packets, dropped %d, created %d repair\n",
sbn, num_esi, num_dropped, num_rep);
}
void usage(char *prog) {
fprintf(stderr, "usage:\n%s <filename> <packet_size> <overhead_percent> <drop_percent>", prog);
exit(1);
}
int main(int argc, char *argv[]) {
if (argc < 5)
usage(argv[0]);
char *infile = argv[1];
struct ioctx *myio = ioctx_from_file(infile, 1);
if (!myio) {
fprintf(stderr, "couldnt access file %s\n", infile);
return -1;
}
size_t filesize = myio->size(myio);
// determine chunks, symbol size, memory usage from size
uint16_t packet_size = strtol(argv[2], NULL, 10); // T
float overhead_percent = strtof(argv[3], NULL);
float drop_percent = strtof(argv[4], NULL);
uint8_t align = 4;
uint16_t ss = packet_size / 2;
uint32_t ws = packet_size * 100;
srand((unsigned int)time(0));
nanorq *rq = nanorq_encoder_new(filesize, packet_size, ss, align, ws);
if (rq == NULL) {
fprintf(stderr, "Coud not initialize encoder.\n");
return -1;
}
uint8_t num_sbn = nanorq_blocks(rq);
for (uint8_t b = 0; b < num_sbn; b++) {
nanorq_generate_symbols(rq, b, myio);
}
uint64_t oti_common = htobe64(nanorq_oti_common(rq));
uint32_t oti_scheme = htobe32(nanorq_oti_scheme_specific(rq));
FILE *oh = fopen("data.rq", "w+");
fwrite(&oti_common, 1, sizeof(oti_common), oh);
fwrite(&oti_scheme, 1, sizeof(oti_scheme), oh);
for (uint8_t sbn = 0; sbn < num_sbn; sbn++) {
dump_block(rq, myio, oh, sbn, overhead_percent, drop_percent);
}
fclose(oh);
nanorq_free(rq);
myio->destroy(myio);
return 0;
}