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
|
/*
* Copyright (c) 2015, 2016 Nicira, Inc.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at:
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
#include <config.h>
#include <errno.h>
#include <sys/types.h>
#include <netinet/in.h>
#include <netinet/icmp6.h>
#include "conntrack-private.h"
#include "dp-packet.h"
enum icmp_state {
ICMPS_FIRST,
ICMPS_REPLY,
};
struct conn_icmp {
struct conn up;
enum icmp_state state;
};
static const enum ct_timeout icmp_timeouts[] = {
[ICMPS_FIRST] = CT_TM_ICMP_FIRST,
[ICMPS_REPLY] = CT_TM_ICMP_REPLY,
};
static struct conn_icmp *
conn_icmp_cast(const struct conn *conn)
{
return CONTAINER_OF(conn, struct conn_icmp, up);
}
static enum ct_update_res
icmp_conn_update(struct conn *conn_, struct conntrack_bucket *ctb,
struct dp_packet *pkt OVS_UNUSED, bool reply, long long now)
{
struct conn_icmp *conn = conn_icmp_cast(conn_);
if (reply && conn->state != ICMPS_REPLY) {
conn->state = ICMPS_REPLY;
}
conn_update_expiration(ctb, &conn->up, icmp_timeouts[conn->state], now);
return CT_UPDATE_VALID;
}
static bool
icmp4_valid_new(struct dp_packet *pkt)
{
struct icmp_header *icmp = dp_packet_l4(pkt);
return icmp->icmp_type == ICMP4_ECHO_REQUEST
|| icmp->icmp_type == ICMP4_INFOREQUEST
|| icmp->icmp_type == ICMP4_TIMESTAMP;
}
static bool
icmp6_valid_new(struct dp_packet *pkt)
{
struct icmp6_header *icmp6 = dp_packet_l4(pkt);
return icmp6->icmp6_type == ICMP6_ECHO_REQUEST;
}
static struct conn *
icmp_new_conn(struct conntrack_bucket *ctb, struct dp_packet *pkt OVS_UNUSED,
long long now)
{
struct conn_icmp *conn;
conn = xzalloc(sizeof *conn);
conn->state = ICMPS_FIRST;
conn_init_expiration(ctb, &conn->up, icmp_timeouts[conn->state], now);
return &conn->up;
}
struct ct_l4_proto ct_proto_icmp4 = {
.new_conn = icmp_new_conn,
.valid_new = icmp4_valid_new,
.conn_update = icmp_conn_update,
};
struct ct_l4_proto ct_proto_icmp6 = {
.new_conn = icmp_new_conn,
.valid_new = icmp6_valid_new,
.conn_update = icmp_conn_update,
};
|