OpenVPN
openvpn.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-2024 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, write to the Free Software Foundation, Inc.,
21  * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
22  */
23 
24 #ifdef HAVE_CONFIG_H
25 #include "config.h"
26 #endif
27 
28 #include "syshead.h"
29 
30 #include "init.h"
31 #include "forward.h"
32 #include "multi.h"
33 #include "win32.h"
34 #include "platform.h"
35 #include "string.h"
36 
37 #include "memdbg.h"
38 
39 #define P2P_CHECK_SIG() EVENT_LOOP_CHECK_SIGNAL(c, process_signal_p2p, c);
40 
41 static bool
43 {
44  remap_signal(c);
45  return process_signal(c);
46 }
47 
48 
49 /**************************************************************************/
57 static void
59 {
60  context_clear_2(c);
61 
62  /* set point-to-point mode */
63  c->mode = CM_P2P;
64  /* initialize tunnel instance, avoid SIGHUP when config is stdin since
65  * re-reading the config from stdin will not work */
66  bool stdin_config = c->options.config && (strcmp(c->options.config, "stdin") == 0);
67  init_instance_handle_signals(c, c->es, stdin_config ? 0 : CC_HARD_USR1_TO_HUP);
68  if (IS_SIG(c))
69  {
70  return;
71  }
72 
73  /* main event loop */
74  while (true)
75  {
77 
78  /* process timers, TLS, etc. */
79  pre_select(c);
80  P2P_CHECK_SIG();
81 
82  /* set up and do the I/O wait */
83  io_wait(c, p2p_iow_flags(c));
84  P2P_CHECK_SIG();
85 
86  /* timeout? */
87  if (c->c2.event_set_status == ES_TIMEOUT)
88  {
89  perf_pop();
90  continue;
91  }
92 
93  /* process the I/O which triggered select */
94  process_io(c, c->c2.link_sockets[0]);
95  P2P_CHECK_SIG();
96 
97  perf_pop();
98  }
99 
101 
103 
104  /* tear down tunnel instance (unless --persist-tun) */
105  close_instance(c);
106 }
107 
108 #undef PROCESS_SIGNAL_P2P
109 
110 void
111 init_early(struct context *c)
112 {
113  net_ctx_init(c, &c->net_ctx);
114 
115  /* init verbosity and mute levels */
117 
118  /* Initialise OpenSSL provider, this needs to be initialised this
119  * early since option post-processing and also openssl info
120  * printing depends on it */
121  for (int j = 1; j < MAX_PARMS && c->options.providers.names[j]; j++)
122  {
123  c->options.providers.providers[j] =
125  }
126 }
127 
128 static void
130 {
131  for (int j = 1; j < MAX_PARMS && c->options.providers.providers[j]; j++)
132  {
134  c->options.providers.providers[j]);
135  }
136  net_ctx_free(&c->net_ctx);
137 }
138 
139 
140 /**************************************************************************/
160 static
161 int
162 openvpn_main(int argc, char *argv[])
163 {
164  struct context c;
165 
166 #if PEDANTIC
167  fprintf(stderr, "Sorry, I was built with --enable-pedantic and I am incapable of doing any real work!\n");
168  return 1;
169 #endif
170 
171 #ifdef _WIN32
172  SetConsoleOutputCP(CP_UTF8);
173 #endif
174 
175  CLEAR(c);
176 
177  /* signify first time for components which can
178  * only be initialized once per program instantiation. */
179  c.first_time = true;
180 
181  /* initialize program-wide statics */
182  if (init_static())
183  {
184  /*
185  * This loop is initially executed on startup and then
186  * once per SIGHUP.
187  */
188  do
189  {
190  /* enter pre-initialization mode with regard to signal handling */
192 
193  /* zero context struct but leave first_time member alone */
195 
196  /* static signal info object */
197  c.sig = &siginfo_static;
198 
199  /* initialize garbage collector scoped to context object */
200  gc_init(&c.gc);
201 
202  /* initialize environmental variable store */
203  c.es = env_set_create(NULL);
204 #ifdef _WIN32
206 #endif
207 
208 #ifdef ENABLE_MANAGEMENT
209  /* initialize management subsystem */
210  init_management();
211 #endif
212 
213  /* initialize options to default state */
214  init_options(&c.options, true);
215 
216  /* parse command line options, and read configuration file */
217  parse_argv(&c.options, argc, argv, M_USAGE, OPT_P_DEFAULT, NULL, c.es);
218 
219 #ifdef ENABLE_PLUGIN
220  /* plugins may contribute options configuration */
222  init_plugins(&c);
224 #endif
225 
226  /* Early initialisation that need to happen before option
227  * post processing and other early startup but after parsing */
228  init_early(&c);
229 
230  /* set dev options */
232 
233  /* openssl print info? */
234  if (print_openssl_info(&c.options))
235  {
236  break;
237  }
238 
239  /* --genkey mode? */
240  if (do_genkey(&c.options))
241  {
242  break;
243  }
244 
245  /* tun/tap persist command? */
246  if (do_persist_tuntap(&c.options, &c.net_ctx))
247  {
248  break;
249  }
250 
251  /* sanity check on options */
253 
254  /* show all option settings */
256 
257  /* print version number */
258  msg(M_INFO, "%s", title_string);
259 #ifdef _WIN32
261 #endif
263 
265 
266  /* misc stuff */
267  pre_setup(&c.options);
268 
269  /* test crypto? */
270  if (do_test_crypto(&c.options))
271  {
272  break;
273  }
274 
275  /* Query passwords before becoming a daemon if we don't use the
276  * management interface to get them. */
278  {
280  }
281 
282  /* become a daemon if --daemon */
283  if (c.first_time)
284  {
287  }
288 
289 #ifdef ENABLE_MANAGEMENT
290  /* open management subsystem */
291  if (!open_management(&c))
292  {
293  break;
294  }
295  /* query for passwords through management interface, if needed */
297  {
299  }
300 #endif
301 
302  /* set certain options as environmental variables */
303  setenv_settings(c.es, &c.options);
304 
305  /* finish context init */
306  context_init_1(&c);
307 
308  do
309  {
310  /* run tunnel depending on mode */
311  switch (c.options.mode)
312  {
313  case MODE_POINT_TO_POINT:
315  break;
316 
317  case MODE_SERVER:
318  tunnel_server(&c);
319  break;
320 
321  default:
322  ASSERT(0);
323  }
324 
325  /* indicates first iteration -- has program-wide scope */
326  c.first_time = false;
327 
328  /* any signals received? */
329  if (IS_SIG(&c))
330  {
331  print_signal(c.sig, NULL, M_INFO);
332  }
333 
334  /* pass restart status to management subsystem */
336  }
337  while (signal_reset(c.sig, SIGUSR1) == SIGUSR1);
338 
339  env_set_destroy(c.es);
341  gc_reset(&c.gc);
342  uninit_early(&c);
343  }
344  while (signal_reset(c.sig, SIGHUP) == SIGHUP);
345  }
346 
347  context_gc_free(&c);
348 
349 #ifdef ENABLE_MANAGEMENT
350  /* close management interface */
352 #endif
353 
354  /* uninitialize program-wide statics */
355  uninit_static();
356 
357  openvpn_exit(OPENVPN_EXIT_STATUS_GOOD); /* exit point */
358  return 0; /* NOTREACHED */
359 }
360 
361 #ifdef _WIN32
362 int
363 wmain(int argc, wchar_t *wargv[])
364 {
365  char **argv;
366  int ret;
367  int i;
368 
369  if ((argv = calloc(argc+1, sizeof(char *))) == NULL)
370  {
371  return 1;
372  }
373 
374  for (i = 0; i < argc; i++)
375  {
376  int n = WideCharToMultiByte(CP_UTF8, 0, wargv[i], -1, NULL, 0, NULL, NULL);
377  argv[i] = malloc(n);
378  WideCharToMultiByte(CP_UTF8, 0, wargv[i], -1, argv[i], n, NULL, NULL);
379  }
380 
381  ret = openvpn_main(argc, argv);
382 
383  for (i = 0; i < argc; i++)
384  {
385  free(argv[i]);
386  }
387  free(argv);
388 
389  return ret;
390 }
391 #else /* ifdef _WIN32 */
392 int
393 main(int argc, char *argv[])
394 {
395  return openvpn_main(argc, argv);
396 }
397 #endif /* ifdef _WIN32 */
signal_restart_status
void signal_restart_status(const struct signal_info *si)
Definition: sig.c:348
uninit_early
static void uninit_early(struct context *c)
Definition: openvpn.c:129
do_test_crypto
bool do_test_crypto(const struct options *o)
Definition: init.c:5187
context_2::event_set_status
unsigned int event_set_status
Definition: openvpn.h:235
OPENVPN_PLUGIN_INIT_PRE_CONFIG_PARSE
#define OPENVPN_PLUGIN_INIT_PRE_CONFIG_PARSE
Definition: openvpn-plugin.h:766
M_INFO
#define M_INFO
Definition: errlevel.h:55
openvpn_main
static int openvpn_main(int argc, char *argv[])
OpenVPN's main init-run-cleanup loop.
Definition: openvpn.c:162
init_early
void init_early(struct context *c)
Definition: openvpn.c:111
print_signal
void print_signal(const struct signal_info *si, const char *title, int msglevel)
Definition: sig.c:294
forward.h
open_management
bool open_management(struct context *c)
Definition: init.c:4469
OPT_P_DEFAULT
#define OPT_P_DEFAULT
Definition: options.h:763
init_query_passwords
void init_query_passwords(const struct context *c)
Query for private key and auth-user-pass username/passwords.
Definition: init.c:651
win32.h
env_set_destroy
void env_set_destroy(struct env_set *es)
Definition: env_set.c:166
argv
Definition: argv.h:35
context
Contains all state information for one tunnel.
Definition: openvpn.h:473
OPENVPN_EXIT_STATUS_GOOD
#define OPENVPN_EXIT_STATUS_GOOD
Definition: error.h:53
print_openssl_info
bool print_openssl_info(const struct options *options)
Definition: init.c:992
options::mode
int mode
Definition: options.h:260
MAX_PARMS
#define MAX_PARMS
Definition: options.h:52
pre_select
void pre_select(struct context *c)
Definition: forward.c:1997
provider_list::providers
provider_t * providers[MAX_PARMS]
Definition: options.h:215
context_gc_free
void context_gc_free(struct context *c)
Definition: init.c:798
gc_reset
static void gc_reset(struct gc_arena *a)
Definition: buffer.h:1046
context::mode
int mode
Role of this context within the OpenVPN process.
Definition: openvpn.h:487
MODE_SERVER
#define MODE_SERVER
Definition: options.h:259
provider_list::names
const char * names[MAX_PARMS]
Definition: options.h:213
net_ctx_init
static int net_ctx_init(struct context *c, openvpn_net_ctx_t *ctx)
Definition: networking.h:48
CLEAR
#define CLEAR(x)
Definition: basic.h:33
wmain
int wmain(int argc, wchar_t *wargv[])
Definition: openvpn.c:363
context::c2
struct context_2 c2
Level 2 context.
Definition: openvpn.h:514
process_signal_p2p
static bool process_signal_p2p(struct context *c)
Definition: openvpn.c:42
options::writepid
const char * writepid
Definition: options.h:382
ASSERT
#define ASSERT(x)
Definition: error.h:195
tunnel_point_to_point
static void tunnel_point_to_point(struct context *c)
Main event loop for OpenVPN in client mode, where only one VPN tunnel is active.
Definition: openvpn.c:58
context::gc
struct gc_arena gc
Garbage collection arena for allocations done in the scope of this context structure.
Definition: openvpn.h:492
open_plugins
void open_plugins(struct context *c, const bool import_options, int init_point)
Definition: init.c:4303
io_wait
static void io_wait(struct context *c, const unsigned int flags)
Definition: forward.h:377
tunnel_server
void tunnel_server(struct context *top)
Main event loop for OpenVPN in server mode.
Definition: multi.c:4283
MF_QUERY_PASSWORDS
#define MF_QUERY_PASSWORDS
Definition: manage.h:29
init.h
context::first_time
bool first_time
True on the first iteration of OpenVPN's main loop.
Definition: openvpn.h:478
close_instance
void close_instance(struct context *c)
Definition: init.c:4872
perf_pop
static void perf_pop(void)
Definition: perf.h:82
init_verb_mute
void init_verb_mute(struct context *c, unsigned int flags)
Definition: init.c:959
context::options
struct options options
Options loaded from command line or configuration file.
Definition: openvpn.h:475
options::providers
struct provider_list providers
Definition: options.h:581
multi.h
set_win_sys_path_via_env
void set_win_sys_path_via_env(struct env_set *es)
Definition: win32.c:1128
write_pid_file
void write_pid_file(const char *filename, const char *chroot_dir)
Definition: init.c:5116
IVM_LEVEL_1
#define IVM_LEVEL_1
Definition: init.h:49
options_postprocess
void options_postprocess(struct options *options, struct env_set *es)
Definition: options.c:4168
git-version.main
def main()
Definition: git-version.py:56
context::did_we_daemonize
bool did_we_daemonize
Whether demonization has already taken place.
Definition: openvpn.h:507
crypto_unload_provider
void crypto_unload_provider(const char *provname, provider_t *provider)
Unloads the given (OpenSSL) provider.
Definition: crypto_openssl.c:178
pre_setup
void pre_setup(const struct options *options)
Definition: init.c:1314
openvpn_exit
void openvpn_exit(const int status)
Definition: error.c:735
show_windows_version
void show_windows_version(const unsigned int flags)
Definition: options.c:4846
init_management
void init_management(void)
Definition: init.c:4460
pre_init_signal_catch
void pre_init_signal_catch(void)
Definition: sig.c:398
init_static
bool init_static(void)
Definition: init.c:837
process_signal
bool process_signal(struct context *c)
Definition: sig.c:640
env_set_create
struct env_set * env_set_create(struct gc_arena *gc)
Definition: env_set.c:156
signal_reset
int signal_reset(struct signal_info *si, int signum)
Clear the signal if its current value equals signum.
Definition: sig.c:266
net_ctx_free
static void net_ctx_free(openvpn_net_ctx_t *ctx)
Definition: networking.h:63
syshead.h
options::management_flags
unsigned int management_flags
Definition: options.h:459
platform.h
context_2::link_sockets
struct link_socket ** link_sockets
Definition: openvpn.h:237
uninit_static
void uninit_static(void)
Definition: init.c:941
context::sig
struct signal_info * sig
Internal error signaling object.
Definition: openvpn.h:500
context_clear_2
void context_clear_2(struct context *c)
Definition: init.c:89
MODE_POINT_TO_POINT
#define MODE_POINT_TO_POINT
Definition: options.h:258
do_genkey
bool do_genkey(const struct options *options)
Definition: init.c:1031
possibly_become_daemon
bool possibly_become_daemon(const struct options *options)
Definition: init.c:1177
show_library_versions
void show_library_versions(const unsigned int flags)
Definition: options.c:4865
uninit_management_callback
void uninit_management_callback(void)
Definition: init.c:4532
options::config
const char * config
Definition: options.h:255
uninit_options
void uninit_options(struct options *o)
Definition: options.c:927
context::es
struct env_set * es
Set of environment variables.
Definition: openvpn.h:496
show_dco_version
void show_dco_version(const unsigned int flags)
Definition: options.c:4855
CC_HARD_USR1_TO_HUP
#define CC_HARD_USR1_TO_HUP
Definition: init.h:106
IS_SIG
#define IS_SIG(c)
Definition: sig.h:48
init_instance_handle_signals
void init_instance_handle_signals(struct context *c, const struct env_set *env, const unsigned int flags)
Definition: init.c:4558
ES_TIMEOUT
#define ES_TIMEOUT
Definition: event.h:69
init_options
void init_options(struct options *o, const bool init_gc)
Definition: options.c:805
init_options_dev
void init_options_dev(struct options *options)
Definition: init.c:982
show_settings
void show_settings(const struct options *o)
Definition: options.c:1838
process_io
void process_io(struct context *c, struct link_socket *sock)
Definition: forward.c:2387
P2P_CHECK_SIG
#define P2P_CHECK_SIG()
Definition: openvpn.c:39
options::chroot_dir
const char * chroot_dir
Definition: options.h:377
crypto_load_provider
provider_t * crypto_load_provider(const char *provider)
Load the given (OpenSSL) providers.
Definition: crypto_openssl.c:161
do_persist_tuntap
bool do_persist_tuntap(struct options *options, openvpn_net_ctx_t *ctx)
Definition: init.c:1114
config.h
setenv_settings
void setenv_settings(struct env_set *es, const struct options *o)
Definition: options.c:1030
init_plugins
void init_plugins(struct context *c)
Definition: init.c:4293
p2p_iow_flags
static unsigned int p2p_iow_flags(const struct context *c)
Definition: forward.h:352
gc_init
static void gc_init(struct gc_arena *a)
Definition: buffer.h:1012
PERF_EVENT_LOOP
#define PERF_EVENT_LOOP
Definition: perf.h:44
parse_argv
void parse_argv(struct options *options, const int argc, char *argv[], const int msglevel, const unsigned int permission_mask, unsigned int *option_types_found, struct env_set *es)
Definition: options.c:5372
remap_signal
void remap_signal(struct context *c)
Definition: sig.c:591
memdbg.h
context_clear_all_except_first_time
void context_clear_all_except_first_time(struct context *c)
Definition: init.c:95
title_string
const char title_string[]
Definition: options.c:69
M_USAGE
#define M_USAGE
Definition: error.h:106
msg
#define msg(flags,...)
Definition: error.h:144
context_init_1
void context_init_1(struct context *c)
Definition: init.c:747
siginfo_static
struct signal_info siginfo_static
Definition: sig.c:45
persist_client_stats
void persist_client_stats(struct context *c)
Definition: init.c:4543
perf_push
static void perf_push(int type)
Definition: perf.h:78
CM_P2P
#define CM_P2P
Definition: openvpn.h:482
close_management
void close_management(void)
Definition: init.c:4519
context::net_ctx
openvpn_net_ctx_t net_ctx
Networking API opaque context.
Definition: openvpn.h:498