-
Notifications
You must be signed in to change notification settings - Fork 58
/
Copy pathsession.c
165 lines (145 loc) · 4.98 KB
/
session.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
/*******************************************************************************
*
* Copyright (c) 2011, 2012, 2013, 2014, 2015 Olaf Bergmann (TZI) and others.
* All rights reserved. This program and the accompanying materials
* are made available under the terms of the Eclipse Public License v1.0
* and Eclipse Distribution License v. 1.0 which accompanies this distribution.
*
* The Eclipse Public License is available at http://www.eclipse.org/legal/epl-v10.html
* and the Eclipse Distribution License is available at
* http://www.eclipse.org/org/documents/edl-v10.php.
*
* Contributors:
* Olaf Bergmann - initial API and implementation
*
*******************************************************************************/
#include <string.h>
#include "session.h"
#ifdef HAVE_ASSERT_H
#include <assert.h>
#else
#ifndef assert
#warning "assertions are disabled"
# define assert(x)
#endif
#endif
#if defined(WITH_CONTIKI)
#define _dtls_address_equals_impl(A,B) \
((A)->size == (B)->size \
&& (A)->port == (B)->port \
&& uip_ipaddr_cmp(&((A)->addr),&((B)->addr)) \
&& (A)->ifindex == (B)->ifindex)
#elif defined(WITH_RIOT_SOCK)
#include "net/af.h"
#ifdef SOCK_HAS_IPV4
#define _dtls_ipv4_address_equals_impl(A,B) \
((A)->size == (B)->size \
&& (A)->addr.port == (B)->addr.port \
&& (A)->ifindex == (B)->ifindex) \
&& (A)->addr.family == (B)->addr.family \
&& ipv4_addr_equal(&((A)->addr.ipv4),&((B)->addr.ipv4))
#endif
#ifdef SOCK_HAS_IPV6
#define _dtls_ipv6_address_equals_impl(A,B) \
((A)->size == (B)->size \
&& (A)->addr.port == (B)->addr.port \
&& (A)->ifindex == (B)->ifindex) \
&& (A)->addr.family == (B)->addr.family \
&& ipv6_addr_equal(&((A)->addr.ipv6),&((B)->addr.ipv6))
#endif
#elif defined(WITH_LWIP_NO_SOCKET)
#define _dtls_address_equals_impl(A,B) \
((A)->size == (B)->size \
&& (A)->port == (B)->port \
&& ip_addr_cmp(&((A)->addr),&((B)->addr)) \
&& (A)->ifindex == (B)->ifindex)
#else /* ! WITH_CONTIKI && !WITH_RIOT_SOCK && ! WITH_LWIP_NO_SOCKET */
static inline int
_dtls_address_equals_impl(const session_t *a,
const session_t *b) {
if (a->ifindex != b->ifindex ||
a->size != b->size || a->addr.sa.sa_family != b->addr.sa.sa_family)
return 0;
/* need to compare only relevant parts of sockaddr_in6 */
switch (a->addr.sa.sa_family) {
case AF_INET:
return
a->addr.sin.sin_port == b->addr.sin.sin_port &&
memcmp(&a->addr.sin.sin_addr, &b->addr.sin.sin_addr,
sizeof(struct in_addr)) == 0;
case AF_INET6:
return a->addr.sin6.sin6_port == b->addr.sin6.sin6_port &&
memcmp(&a->addr.sin6.sin6_addr, &b->addr.sin6.sin6_addr,
sizeof(struct in6_addr)) == 0;
default: /* fall through and signal error */
;
}
return 0;
}
#endif /* ! WITH_CONTIKI && !WITH_RIOT_SOCK && ! WITH_LWIP_NO_SOCKET */
void
dtls_session_init(session_t *sess) {
assert(sess);
memset(sess, 0, sizeof(session_t));
sess->size = sizeof(sess->addr);
}
/* These functions are primarly needed for the tinydtls ruby gem.
*
* They are not implemented on Contiki and RIOT because these operating
* system don't supply malloc(3). This could be fixed by fixed by using
* memory pools on these operating system as in `peer.c`. However, the
* downside of this approach is that the memory pools reserve memory
* even if the `dtls_new_session` isn't used and usually memory for the
* `session_t` type is already resevered in the `peer_t` struct.
* Therefore it would introduces quite some overhead on these
* constrained platforms.
*
* In the long run we probably want to create two seperate memory pools
* for sessions and peers and store a pointer to a session in the peer
* struct.
*/
#if !(defined (WITH_CONTIKI)) && !(defined (RIOT_VERSION)) && !(defined (WITH_LWIP_NO_SOCKET))
session_t*
dtls_new_session(struct sockaddr *addr, socklen_t addrlen) {
session_t *sess;
sess = malloc(sizeof(session_t));
if (!sess)
return NULL;
dtls_session_init(sess);
sess->size = addrlen;
memcpy(&sess->addr.sa, addr, sess->size);
return sess;
}
void
dtls_free_session(session_t *sess) {
free(sess);
}
struct sockaddr*
dtls_session_addr(session_t *sess, socklen_t *addrlen) {
if (!sess)
return NULL;
*addrlen = sess->size;
return &sess->addr.sa;
}
#endif /* ! WITH_CONTIKI && ! RIOT_VERSION && ! WITH_LWIP_NO_SOCKET */
int
dtls_session_equals(const session_t *a, const session_t *b) {
assert(a); assert(b);
#ifdef RIOT_VERSION
switch (a->addr.family) {
#ifdef SOCK_HAS_IPV4
case AF_INET:
return _dtls_ipv4_address_equals_impl(a, b);
#endif
#ifdef SOCK_HAS_IPV6
case AF_INET6:
return _dtls_ipv6_address_equals_impl(a, b);
#endif
default:
assert(0);
return false;
}
#else
return _dtls_address_equals_impl(a, b);
#endif /* RIOT_VERSION */
}