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
|
#include "core.h"
#include <string.h>
#define ACK_RATE_UNIT 10000
static ssize_t handle_ack(ECPConnection *conn, ecp_seq_t seq, unsigned char mtype, unsigned char *msg, ssize_t size) {
int idx = 0;
ecp_seq_t seq_ack = 0;
ecp_ack_t ack_map = 0;
ECPRBSend *buf = conn->rbuf.send;
ECPRBuffer *rbuf = &buf->rbuf;
idx = ecp_rbuf_msg_idx(rbuf, seq_ack);
if (idx < 0) return idx;
seq_ack++;
buf->in_transit -= seq_ack - rbuf->seq_start;
if (ack_map != ECP_RBUF_ACK_FULL) {
int i;
int nack = 0;
ecp_win_t nack_cnt = 0;
seq_ack -= ECP_RBUF_ACK_SIZE;
for (i=0; i<ECP_RBUF_ACK_SIZE; i++) {
if ((ack_map & ((ecp_ack_t)1 << (ECP_RBUF_ACK_SIZE - i - 1))) == 0) {
nack_cnt++;
if (buf->reliable) {
idx = ecp_rbuf_msg_idx(rbuf, seq_ack + i);
// resend packet
// ecp_pkt_send(conn->sock, &conn->node.addr, packet, rv);
if (!nack) {
nack = 1;
rbuf->seq_start = seq_ack + i;
rbuf->msg_start = idx;
}
}
}
}
buf->in_transit += nack_cnt;
buf->nack_rate = (buf->nack_rate * 7 + ((ECP_RBUF_ACK_SIZE - nack_cnt) * ACK_RATE_UNIT) / ECP_RBUF_ACK_SIZE) / 8;
if (!buf->reliable) {
rbuf->seq_start = seq_ack + ECP_RBUF_ACK_SIZE;
}
} else {
rbuf->seq_start = seq_ack;
buf->nack_rate = (buf->nack_rate * 7) / 8;
if (buf->reliable) {
rbuf->msg_start = ECP_RBUF_IDX_MASK(idx + 1, rbuf->msg_size);
}
}
return size;
}
int ecp_rbuf_send_create(ECPRBSend *buf, ECPRBMessage *msg, unsigned int msg_size) {
memset(buf, 0, sizeof(ECPRBRecv));
ecp_rbuf_init(&buf->rbuf, msg, msg_size);
return ECP_OK;
}
int ecp_rbuf_send_start(ECPRBSend *buf, ecp_seq_t seq) {
buf->rbuf.seq_start = seq + 1;
return ECP_OK;
}
ssize_t ecp_rbuf_send_store(ECPConnection *conn, ecp_seq_t seq, unsigned char *msg, size_t msg_size) {
ECPRBSend *buf = conn->rbuf.send;
buf->in_transit++;
return ecp_rbuf_msg_store(&buf->rbuf, seq, msg, msg_size, 0, 0);
}
|