OpenVPN
mtcp.c
Go to the documentation of this file.
1/*
2 * OpenVPN -- An application to securely tunnel IP networks
3 * over a single TCP/UDP port, with support for SSL/TLS-based
4 * session authentication and key exchange,
5 * packet encryption, packet authentication, and
6 * packet compression.
7 *
8 * Copyright (C) 2002-2026 OpenVPN Inc <sales@openvpn.net>
9 *
10 * This program is free software; you can redistribute it and/or modify
11 * it under the terms of the GNU General Public License version 2
12 * as published by the Free Software Foundation.
13 *
14 * This program is distributed in the hope that it will be useful,
15 * but WITHOUT ANY WARRANTY; without even the implied warranty of
16 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
17 * GNU General Public License for more details.
18 *
19 * You should have received a copy of the GNU General Public License along
20 * with this program; if not, see <https://www.gnu.org/licenses/>.
21 */
22
23#ifdef HAVE_CONFIG_H
24#include "config.h"
25#endif
26
27#include "syshead.h"
28
29#include "multi.h"
30#include "forward.h"
31#include "mtcp.h"
32#include "multi_io.h"
33
34#include "memdbg.h"
35
36#ifdef HAVE_SYS_INOTIFY_H
37#include <sys/inotify.h>
38#endif
39
40struct multi_instance *
42{
43 struct gc_arena gc = gc_new();
44 struct multi_instance *mi = NULL;
45 struct hash *hash = m->hash;
46
47 mi = multi_create_instance(m, NULL, sock);
48 if (mi)
49 {
50 mi->real.proto = sock->info.proto;
51 struct hash_element *he;
52 const uint32_t hv = hash_value(hash, &mi->real);
53 struct hash_bucket *bucket = hash_bucket(hash, hv);
54
56
57 he = hash_lookup_fast(hash, bucket, &mi->real, hv);
58
59 if (he)
60 {
61 struct multi_instance *oldmi = (struct multi_instance *)he->value;
63 "MULTI TCP: new incoming client address matches existing client address -- new client takes precedence");
64 oldmi->did_real_hash = false;
65 multi_close_instance(m, oldmi, false);
66 he->key = &mi->real;
67 he->value = mi;
68 }
69 else
70 {
71 hash_add_fast(hash, bucket, &mi->real, hv, mi);
72 }
73
74 mi->did_real_hash = true;
75 }
76
77#ifdef ENABLE_DEBUG
78 if (mi)
79 {
80 dmsg(D_MULTI_DEBUG, "MULTI TCP: instance added: %s", mroute_addr_print(&mi->real, &gc));
81 }
82 else
83 {
84 dmsg(D_MULTI_DEBUG, "MULTI TCP: new client instance failed");
85 }
86#endif
87
88 gc_free(&gc);
89 ASSERT(!(mi && mi->halt));
90 return mi;
91}
92
93bool
95{
96 /* buffer for queued TCP socket output packets */
98
103 ASSERT(mi->context.c2.link_sockets[0]->info.lsa->actual.dest.addr.sa.sa_family == AF_INET
104 || mi->context.c2.link_sockets[0]->info.lsa->actual.dest.addr.sa.sa_family == AF_INET6);
105 mi->real.proto = mi->context.c2.link_sockets[0]->info.proto;
107 &mi->real, &mi->context.c2.link_sockets[0]->info.lsa->actual.dest, true))
108 {
109 msg(D_MULTI_ERRORS, "MULTI TCP: TCP client address is undefined");
110 return false;
111 }
112 return true;
113}
114
115void
120
121void
123{
124 if (multi_io && multi_io->es)
125 {
126 event_del(multi_io->es, event);
127 }
128}
129
130void
132{
133 struct link_socket *sock = mi->context.c2.link_sockets[0];
134 if (sock && mi->socket_set_called)
135 {
137 mi->socket_set_called = false;
138 }
139 multi_io->n_esr = 0;
140}
141
142bool
144 const unsigned int mpp_flags)
145{
146 struct mbuf_item item;
147 bool ret = true;
148 ASSERT(mi);
149
150 /* extract from queue */
151 if (mbuf_extract_item(mi->tcp_link_out_deferred, &item)) /* ciphertext IP packet */
152 {
153 dmsg(D_MULTI_TCP, "MULTI TCP: transmitting previously deferred packet");
154
155 ASSERT(mi == item.instance);
156 mi->context.c2.to_link = item.buffer->buf;
157 ret = multi_process_outgoing_link_dowork(m, mi, mpp_flags);
158 mbuf_free_buf(item.buffer);
159 }
160 return ret;
161}
162
163bool
164multi_tcp_process_outgoing_link(struct multi_context *m, bool defer, const unsigned int mpp_flags)
165{
167 bool ret = true;
168
169 if (mi)
170 {
171 if ((defer && !proto_is_dgram(mi->context.c2.link_sockets[0]->info.proto))
173 {
174 /* save to queue */
175 struct buffer *buf = &mi->context.c2.to_link;
176 if (BLEN(buf) > 0)
177 {
178 struct mbuf_buffer *mb = mbuf_alloc_buf(buf);
179 struct mbuf_item item;
180
181 set_prefix(mi);
182 dmsg(D_MULTI_TCP, "MULTI TCP: queuing deferred packet");
183 item.buffer = mb;
184 item.instance = mi;
186 mbuf_free_buf(mb);
187 buf_reset(buf);
188 ret = multi_process_post(m, mi, mpp_flags);
189 if (!ret)
190 {
191 mi = NULL;
192 }
193 clear_prefix();
194 }
195 }
196 else
197 {
198 ret = multi_process_outgoing_link_dowork(m, mi, mpp_flags);
199 if (!ret)
200 {
201 mi = NULL;
202 }
203 }
204 }
205 return ret;
206}
static void buf_reset(struct buffer *buf)
Definition buffer.h:304
#define BLEN(buf)
Definition buffer.h:126
static void gc_free(struct gc_arena *a)
Definition buffer.h:1049
static struct gc_arena gc_new(void)
Definition buffer.h:1041
#define D_MULTI_ERRORS
Definition errlevel.h:64
#define D_MULTI_DEBUG
Definition errlevel.h:126
#define D_MULTI_TCP
Definition errlevel.h:162
#define D_MULTI_LOW
Definition errlevel.h:85
static void event_del(struct event_set *es, event_t event)
Definition event.h:174
Interface functions to the internal and external multiplexers.
struct hash_element * hash_lookup_fast(struct hash *hash, struct hash_bucket *bucket, const void *key, uint32_t hv)
Definition list.c:79
static uint32_t hash_value(const struct hash *hash, const void *key)
Definition list.h:104
static void hash_add_fast(struct hash *hash, struct hash_bucket *bucket, const void *key, uint32_t hv, void *value)
Definition list.h:146
void mbuf_add_item(struct mbuf_set *ms, const struct mbuf_item *item)
Definition mbuf.c:99
struct mbuf_buffer * mbuf_alloc_buf(const struct buffer *buf)
Definition mbuf.c:75
void mbuf_free_buf(struct mbuf_buffer *mb)
Definition mbuf.c:86
bool mbuf_extract_item(struct mbuf_set *ms, struct mbuf_item *item)
Definition mbuf.c:121
void mbuf_free(struct mbuf_set *ms)
Definition mbuf.c:59
struct mbuf_set * mbuf_init(unsigned int size)
Definition mbuf.c:43
static bool mbuf_defined(const struct mbuf_set *ms)
Definition mbuf.h:81
bool mroute_extract_openvpn_sockaddr(struct mroute_addr *addr, const struct openvpn_sockaddr *osaddr, bool use_port)
Definition mroute.c:254
const char * mroute_addr_print(const struct mroute_addr *ma, struct gc_arena *gc)
Definition mroute.c:371
void multi_tcp_instance_specific_free(struct multi_instance *mi)
Definition mtcp.c:116
struct multi_instance * multi_create_instance_tcp(struct multi_context *m, struct link_socket *sock)
Definition mtcp.c:41
void multi_tcp_delete_event(struct multi_io *multi_io, event_t event)
Definition mtcp.c:122
bool multi_tcp_instance_specific_init(struct multi_context *m, struct multi_instance *mi)
Definition mtcp.c:94
bool multi_tcp_process_outgoing_link(struct multi_context *m, bool defer, const unsigned int mpp_flags)
Definition mtcp.c:164
bool multi_tcp_process_outgoing_link_ready(struct multi_context *m, struct multi_instance *mi, const unsigned int mpp_flags)
Definition mtcp.c:143
void multi_tcp_dereference_instance(struct multi_io *multi_io, struct multi_instance *mi)
Definition mtcp.c:131
struct multi_instance * multi_create_instance(struct multi_context *m, const struct mroute_addr *real, struct link_socket *sock)
Definition multi.c:702
bool multi_process_post(struct multi_context *m, struct multi_instance *mi, const unsigned int flags)
Perform postprocessing of a VPN tunnel instance.
Definition multi.c:2981
void multi_close_instance(struct multi_context *m, struct multi_instance *mi, bool shutdown)
Definition multi.c:558
void multi_assign_peer_id(struct multi_context *m, struct multi_instance *mi)
Assigns a peer-id to a a client and adds the instance to the the instances array of the multi_context...
Definition multi.c:4080
Header file for server-mode related structures and functions.
static void set_prefix(struct multi_instance *mi)
Definition multi.h:523
static void clear_prefix(void)
Definition multi.h:535
static struct multi_instance * multi_process_outgoing_link_pre(struct multi_context *m)
Definition multi.h:413
static bool multi_process_outgoing_link_dowork(struct multi_context *m, struct multi_instance *mi, const unsigned int mpp_flags)
Definition multi.h:661
#define dmsg(flags,...)
Definition error.h:172
#define msg(flags,...)
Definition error.h:152
#define ASSERT(x)
Definition error.h:219
static event_t socket_event_handle(const struct link_socket *sock)
Definition socket.h:796
#define LS_MODE_TCP_ACCEPT_FROM
Definition socket.h:199
static bool proto_is_dgram(int proto)
Return if the protocol is datagram (UDP)
Wrapper structure for dynamically allocated memory.
Definition buffer.h:60
struct buffer to_link
Definition openvpn.h:377
struct link_socket ** link_sockets
Definition openvpn.h:237
struct context_2 c2
Level 2 context.
Definition openvpn.h:514
struct options options
Options loaded from command line or configuration file.
Definition openvpn.h:472
Garbage collection arena used to keep track of dynamically allocated memory.
Definition buffer.h:116
void * value
Definition list.h:41
const void * key
Definition list.h:42
Definition list.h:53
struct buffer buf
Definition mbuf.h:44
struct mbuf_buffer * buffer
Definition mbuf.h:53
struct multi_instance * instance
Definition mbuf.h:54
uint8_t proto
Definition mroute.h:84
Main OpenVPN server state structure.
Definition multi.h:162
struct hash * hash
VPN tunnel instances indexed by real address of the remote peer.
Definition multi.h:169
struct context top
Storage structure for process-wide configuration.
Definition multi.h:201
Server-mode state structure for one single VPN tunnel.
Definition multi.h:102
struct mbuf_set * tcp_link_out_deferred
Definition multi.h:128
struct mroute_addr real
External network address of the remote peer.
Definition multi.h:121
bool socket_set_called
Definition multi.h:129
bool did_real_hash
Definition multi.h:134
struct context context
The context structure storing state for this VPN tunnel.
Definition multi.h:142
struct event_set * es
Definition multi_io.h:53
int n_esr
Definition multi_io.h:55
union openvpn_sockaddr::@27 addr
struct sockaddr sa
Definition socket_util.h:42
int n_bcast_buf
Definition options.h:507
struct gc_arena gc
Definition test_ssl.c:133