OpenVPN
main.c
Go to the documentation of this file.
1 /*
2  * tapctl -- Utility to manipulate TUN/TAP adapters on Windows
3  * https://community.openvpn.net/openvpn/wiki/Tapctl
4  *
5  * Copyright (C) 2002-2024 OpenVPN Inc <sales@openvpn.net>
6  * Copyright (C) 2018-2024 Simon Rozman <simon@rozman.si>
7  *
8  * This program is free software; you can redistribute it and/or modify
9  * it under the terms of the GNU General Public License version 2
10  * as published by the Free Software Foundation.
11  *
12  * This program is distributed in the hope that it will be useful,
13  * but WITHOUT ANY WARRANTY; without even the implied warranty of
14  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15  * GNU General Public License for more details.
16  *
17  * You should have received a copy of the GNU General Public License along
18  * with this program; if not, write to the Free Software Foundation, Inc.,
19  * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
20  */
21 
22 #ifdef HAVE_CONFIG_H
23 #include <config.h>
24 #endif
25 
26 #include "tap.h"
27 #include "error.h"
28 
29 #include <objbase.h>
30 #include <setupapi.h>
31 #include <stdio.h>
32 #include <wchar.h>
33 
34 #ifdef _MSC_VER
35 #pragma comment(lib, "ole32.lib")
36 #pragma comment(lib, "setupapi.lib")
37 #endif
38 
39 
40 const WCHAR title_string[] =
42 ;
43 
44 static const WCHAR usage_message[] =
45  L"%ls\n"
46  L"\n"
47  L"Usage:\n"
48  L"\n"
49  L"tapctl <command> [<command specific options>]\n"
50  L"\n"
51  L"Commands:\n"
52  L"\n"
53  L"create Create a new TUN/TAP adapter\n"
54  L"list List TUN/TAP adapters\n"
55  L"delete Delete specified network adapter\n"
56  L"help Display this text\n"
57  L"\n"
58  L"Hint: Use \"tapctl help <command>\" to display help for particular command.\n"
59 ;
60 
61 static const WCHAR usage_message_create[] =
62  L"%ls\n"
63  L"\n"
64  L"Creates a new TUN/TAP adapter\n"
65  L"\n"
66  L"Usage:\n"
67  L"\n"
68  L"tapctl create [<options>]\n"
69  L"\n"
70  L"Options:\n"
71  L"\n"
72  L"--name <name> Set TUN/TAP adapter name. Should the adapter with given name \n"
73  L" already exist, an error is returned. If this option is not \n"
74  L" specified, a default adapter name is chosen by Windows. \n"
75  L" Note: This name can also be specified as OpenVPN's --dev-node \n"
76  L" option. \n"
77  L"--hwid <hwid> Adapter hardware ID. Default value is root\\tap0901, which \n"
78  L" describes tap-windows6 driver. To work with wintun or ovpn-dco \n"
79  L" driver, specify 'wintun' or 'ovpn-dco'. \n"
80  L"\n"
81  L"Output:\n"
82  L"\n"
83  L"This command prints newly created TUN/TAP adapter's GUID to stdout. \n"
84 ;
85 
86 static const WCHAR usage_message_list[] =
87  L"%ls\n"
88  L"\n"
89  L"Lists TUN/TAP adapters\n"
90  L"\n"
91  L"Usage:\n"
92  L"\n"
93  L"tapctl list\n"
94  L"\n"
95  L"Options:\n"
96  L"\n"
97  L"--hwid <hwid> Adapter hardware ID. By default, root\\tap0901, tap0901, wintun and \n"
98  L" ovpn-dco adapters are listed. Use this switch to limit the list.\n"
99  L"\n"
100  L"Output:\n"
101  L"\n"
102  L"This command prints all TUN/TAP adapters to stdout. \n"
103 ;
104 
105 static const WCHAR usage_message_delete[] =
106  L"%ls\n"
107  L"\n"
108  L"Deletes the specified network adapter\n"
109  L"\n"
110  L"Usage:\n"
111  L"\n"
112  L"tapctl delete <adapter GUID | adapter name>\n"
113 ;
114 
115 
119 static void
120 usage(void)
121 {
122  fwprintf(stderr,
124  title_string);
125 }
126 
130 static BOOL
131 is_adapter_name_available(LPCWSTR name, struct tap_adapter_node *adapter_list, BOOL log)
132 {
133  for (struct tap_adapter_node *a = adapter_list; a; a = a->pNext)
134  {
135  if (wcsicmp(name, a->szName) == 0)
136  {
137  if (log)
138  {
139  LPOLESTR adapter_id = NULL;
140  StringFromIID((REFIID)&a->guid, &adapter_id);
141  fwprintf(stderr, L"Adapter \"%ls\" already exists (GUID %"
142  L"ls).\n", a->szName, adapter_id);
143  CoTaskMemFree(adapter_id);
144  }
145 
146  return FALSE;
147  }
148  }
149 
150  return TRUE;
151 }
152 
157 static LPWSTR
158 get_unique_adapter_name(LPCWSTR hwid, struct tap_adapter_node *adapter_list)
159 {
160  if (hwid == NULL)
161  {
162  return NULL;
163  }
164 
165  LPCWSTR base_name;
166  if (wcsicmp(hwid, L"ovpn-dco") == 0)
167  {
168  base_name = L"OpenVPN Data Channel Offload";
169  }
170  else if (wcsicmp(hwid, L"wintun") == 0)
171  {
172  base_name = L"OpenVPN Wintun";
173  }
174  else if (wcsicmp(hwid, L"root\\" _L(TAP_WIN_COMPONENT_ID)) == 0)
175  {
176  base_name = L"OpenVPN TAP-Windows6";
177  }
178  else
179  {
180  return NULL;
181  }
182 
183  if (is_adapter_name_available(base_name, adapter_list, FALSE))
184  {
185  return wcsdup(base_name);
186  }
187 
188  size_t name_len = wcslen(base_name) + 10;
189  LPWSTR name = malloc(name_len * sizeof(WCHAR));
190  if (name == NULL)
191  {
192  return NULL;
193  }
194  for (int i = 1; i < 100; ++i)
195  {
196  swprintf_s(name, name_len, L"%ls #%d", base_name, i);
197 
198  if (is_adapter_name_available(name, adapter_list, FALSE))
199  {
200  return name;
201  }
202  }
203 
204  return NULL;
205 }
206 
210 int __cdecl
211 wmain(int argc, LPCWSTR argv[])
212 {
213  int iResult;
214  BOOL bRebootRequired = FALSE;
215 
216  /* Ask SetupAPI to keep quiet. */
217  SetupSetNonInteractiveMode(TRUE);
218 
219  if (argc < 2)
220  {
221  usage();
222  return 1;
223  }
224  else if (wcsicmp(argv[1], L"help") == 0)
225  {
226  /* Output help. */
227  if (argc < 3)
228  {
229  usage();
230  }
231  else if (wcsicmp(argv[2], L"create") == 0)
232  {
233  fwprintf(stderr, usage_message_create, title_string);
234  }
235  else if (wcsicmp(argv[2], L"list") == 0)
236  {
237  fwprintf(stderr, usage_message_list, title_string);
238  }
239  else if (wcsicmp(argv[2], L"delete") == 0)
240  {
241  fwprintf(stderr, usage_message_delete, title_string);
242  }
243  else
244  {
245  fwprintf(stderr, L"Unknown command \"%ls"
246  L"\". Please, use \"tapctl help\" to list supported commands.\n", argv[2]);
247  }
248 
249  return 1;
250  }
251  else if (wcsicmp(argv[1], L"create") == 0)
252  {
253  LPCWSTR szName = NULL;
254  LPCWSTR szHwId = L"root\\" _L(TAP_WIN_COMPONENT_ID);
255 
256  /* Parse options. */
257  for (int i = 2; i < argc; i++)
258  {
259  if (wcsicmp(argv[i], L"--name") == 0)
260  {
261  szName = argv[++i];
262  }
263  else if (wcsicmp(argv[i], L"--hwid") == 0)
264  {
265  szHwId = argv[++i];
266  }
267  else
268  {
269  fwprintf(stderr, L"Unknown option \"%ls"
270  L"\". Please, use \"tapctl help create\" to list supported options. Ignored.\n",
271  argv[i]);
272  }
273  }
274 
275  /* Create TUN/TAP adapter. */
276  GUID guidAdapter;
277  LPOLESTR szAdapterId = NULL;
278  DWORD dwResult = tap_create_adapter(
279  NULL,
280  L"Virtual Ethernet",
281  szHwId,
282  &bRebootRequired,
283  &guidAdapter);
284  if (dwResult != ERROR_SUCCESS)
285  {
286  fwprintf(stderr, L"Creating TUN/TAP adapter failed (error 0x%x).\n", dwResult);
287  iResult = 1; goto quit;
288  }
289 
290  /* Get existing network adapters. */
291  struct tap_adapter_node *pAdapterList = NULL;
292  dwResult = tap_list_adapters(NULL, NULL, &pAdapterList);
293  if (dwResult != ERROR_SUCCESS)
294  {
295  fwprintf(stderr, L"Enumerating adapters failed (error 0x%x).\n", dwResult);
296  iResult = 1;
297  goto create_delete_adapter;
298  }
299 
300  LPWSTR adapter_name = szName ? wcsdup(szName) : get_unique_adapter_name(szHwId, pAdapterList);
301  if (adapter_name)
302  {
303  /* Check for duplicates when name was specified,
304  * otherwise get_adapter_default_name() takes care of it */
305  if (szName && !is_adapter_name_available(adapter_name, pAdapterList, TRUE))
306  {
307  iResult = 1;
308  goto create_cleanup_pAdapterList;
309  }
310 
311  /* Rename the adapter. */
312  dwResult = tap_set_adapter_name(&guidAdapter, adapter_name, FALSE);
313  if (dwResult != ERROR_SUCCESS)
314  {
315  StringFromIID((REFIID)&guidAdapter, &szAdapterId);
316  fwprintf(stderr, L"Renaming TUN/TAP adapter %ls"
317  L" to \"%ls\" failed (error 0x%x).\n",
318  szAdapterId, adapter_name, dwResult);
319  CoTaskMemFree(szAdapterId);
320  iResult = 1; goto quit;
321  }
322  }
323 
324  iResult = 0;
325 
326 create_cleanup_pAdapterList:
327  free(adapter_name);
328 
329  tap_free_adapter_list(pAdapterList);
330  if (iResult)
331  {
332  goto create_delete_adapter;
333  }
334 
335  /* Output adapter GUID. */
336  StringFromIID((REFIID)&guidAdapter, &szAdapterId);
337  fwprintf(stdout, L"%ls\n", szAdapterId);
338  CoTaskMemFree(szAdapterId);
339 
340  iResult = 0; goto quit;
341 
342 create_delete_adapter:
344  NULL,
345  &guidAdapter,
346  &bRebootRequired);
347  iResult = 1; goto quit;
348  }
349  else if (wcsicmp(argv[1], L"list") == 0)
350  {
351  WCHAR szzHwId[0x100] =
352  L"root\\" _L(TAP_WIN_COMPONENT_ID) L"\0"
353  _L(TAP_WIN_COMPONENT_ID) L"\0"
354  L"Wintun\0"
355  L"ovpn-dco\0";
356 
357  /* Parse options. */
358  for (int i = 2; i < argc; i++)
359  {
360  if (wcsicmp(argv[i], L"--hwid") == 0)
361  {
362  memset(szzHwId, 0, sizeof(szzHwId));
363  ++i;
364  memcpy_s(szzHwId, sizeof(szzHwId) - 2*sizeof(WCHAR) /*requires double zero termination*/, argv[i], wcslen(argv[i])*sizeof(WCHAR));
365  }
366  else
367  {
368  fwprintf(stderr, L"Unknown option \"%ls"
369  L"\". Please, use \"tapctl help list\" to list supported options. Ignored.\n",
370  argv[i]);
371  }
372  }
373 
374  /* Output list of adapters with given hardware ID. */
375  struct tap_adapter_node *pAdapterList = NULL;
376  DWORD dwResult = tap_list_adapters(NULL, szzHwId, &pAdapterList);
377  if (dwResult != ERROR_SUCCESS)
378  {
379  fwprintf(stderr, L"Enumerating TUN/TAP adapters failed (error 0x%x).\n", dwResult);
380  iResult = 1; goto quit;
381  }
382 
383  for (struct tap_adapter_node *pAdapter = pAdapterList; pAdapter; pAdapter = pAdapter->pNext)
384  {
385  LPOLESTR szAdapterId = NULL;
386  StringFromIID((REFIID)&pAdapter->guid, &szAdapterId);
387  fwprintf(stdout, L"%ls\t%"
388  L"ls\n", szAdapterId, pAdapter->szName);
389  CoTaskMemFree(szAdapterId);
390  }
391 
392  iResult = 0;
393  tap_free_adapter_list(pAdapterList);
394  }
395  else if (wcsicmp(argv[1], L"delete") == 0)
396  {
397  if (argc < 3)
398  {
399  fwprintf(stderr, L"Missing adapter GUID or name. Please, use \"tapctl help delete\" for usage info.\n");
400  return 1;
401  }
402 
403  GUID guidAdapter;
404  if (FAILED(IIDFromString(argv[2], (LPIID)&guidAdapter)))
405  {
406  /* The argument failed to covert to GUID. Treat it as the adapter name. */
407  struct tap_adapter_node *pAdapterList = NULL;
408  DWORD dwResult = tap_list_adapters(NULL, NULL, &pAdapterList);
409  if (dwResult != ERROR_SUCCESS)
410  {
411  fwprintf(stderr, L"Enumerating TUN/TAP adapters failed (error 0x%x).\n", dwResult);
412  iResult = 1; goto quit;
413  }
414 
415  for (struct tap_adapter_node *pAdapter = pAdapterList;; pAdapter = pAdapter->pNext)
416  {
417  if (pAdapter == NULL)
418  {
419  fwprintf(stderr, L"\"%ls\" adapter not found.\n", argv[2]);
420  iResult = 1; goto delete_cleanup_pAdapterList;
421  }
422  else if (wcsicmp(argv[2], pAdapter->szName) == 0)
423  {
424  memcpy(&guidAdapter, &pAdapter->guid, sizeof(GUID));
425  break;
426  }
427  }
428 
429  iResult = 0;
430 
431 delete_cleanup_pAdapterList:
432  tap_free_adapter_list(pAdapterList);
433  if (iResult)
434  {
435  goto quit;
436  }
437  }
438 
439  /* Delete the network adapter. */
440  DWORD dwResult = tap_delete_adapter(
441  NULL,
442  &guidAdapter,
443  &bRebootRequired);
444  if (dwResult != ERROR_SUCCESS)
445  {
446  fwprintf(stderr, L"Deleting adapter \"%ls"
447  L"\" failed (error 0x%x).\n", argv[2], dwResult);
448  iResult = 1; goto quit;
449  }
450 
451  iResult = 0; goto quit;
452  }
453  else
454  {
455  fwprintf(stderr, L"Unknown command \"%ls"
456  L"\". Please, use \"tapctl help\" to list supported commands.\n", argv[1]);
457  return 1;
458  }
459 
460 quit:
461  if (bRebootRequired)
462  {
463  fwprintf(stderr, L"A system reboot is required.\n");
464  }
465 
466  return iResult;
467 }
468 
469 
470 bool
471 dont_mute(unsigned int flags)
472 {
473  UNREFERENCED_PARAMETER(flags);
474 
475  return true;
476 }
477 
478 
479 void
480 x_msg_va(const unsigned int flags, const char *format, va_list arglist)
481 {
482  /* Output message string. Note: Message strings don't contain line terminators. */
483  vfprintf(stderr, format, arglist);
484  fwprintf(stderr, L"\n");
485 
486  if ((flags & M_ERRNO) != 0)
487  {
488  /* Output system error message (if possible). */
489  DWORD dwResult = GetLastError();
490  LPWSTR szErrMessage = NULL;
491  if (FormatMessage(
492  FORMAT_MESSAGE_FROM_SYSTEM | FORMAT_MESSAGE_ALLOCATE_BUFFER | FORMAT_MESSAGE_IGNORE_INSERTS,
493  0,
494  dwResult,
495  0,
496  (LPWSTR)&szErrMessage,
497  0,
498  NULL) && szErrMessage)
499  {
500  /* Trim trailing whitespace. Set terminator after the last non-whitespace character. This prevents excessive trailing line breaks. */
501  for (size_t i = 0, i_last = 0;; i++)
502  {
503  if (szErrMessage[i])
504  {
505  if (!iswspace(szErrMessage[i]))
506  {
507  i_last = i + 1;
508  }
509  }
510  else
511  {
512  szErrMessage[i_last] = 0;
513  break;
514  }
515  }
516 
517  /* Output error message. */
518  fwprintf(stderr, L"Error 0x%x: %ls\n", dwResult, szErrMessage);
519 
520  LocalFree(szErrMessage);
521  }
522  else
523  {
524  fwprintf(stderr, L"Error 0x%x\n", dwResult);
525  }
526  }
527 }
tap_adapter_node::pNext
struct tap_adapter_node * pNext
Pointer to next adapter.
Definition: tap.h:142
_L
#define _L(q)
Definition: basic.h:37
M_ERRNO
#define M_ERRNO
Definition: error.h:94
tap_list_adapters
DWORD tap_list_adapters(_In_opt_ HWND hwndParent, _In_opt_ LPCWSTR szzHwIDs, _Out_ struct tap_adapter_node **ppAdapter)
Creates a list of existing network adapters.
Definition: tap.c:1146
PACKAGE_NAME
#define PACKAGE_NAME
Definition: config.h:492
usage_message_delete
static const WCHAR usage_message_delete[]
Definition: main.c:105
argv
Definition: argv.h:35
usage_message_list
static const WCHAR usage_message_list[]
Definition: main.c:86
tap_set_adapter_name
DWORD tap_set_adapter_name(_In_ LPCGUID pguidAdapter, _In_ LPCWSTR szName, _In_ BOOL bSilent)
Sets adapter name.
Definition: tap.c:1063
PACKAGE_VERSION
#define PACKAGE_VERSION
Definition: config.h:504
title_string
const WCHAR title_string[]
Definition: main.c:40
tap_delete_adapter
DWORD tap_delete_adapter(_In_opt_ HWND hwndParent, _In_ LPCGUID pguidAdapter, _Inout_ LPBOOL pbRebootRequired)
Deletes an adapter.
Definition: tap.c:1005
tap_adapter_node::szName
LPWSTR szName
Adapter name.
Definition: tap.h:140
tap_free_adapter_list
void tap_free_adapter_list(_In_ struct tap_adapter_node *pAdapterList)
Frees a list of network adapters.
Definition: tap.c:1355
dont_mute
bool dont_mute(unsigned int flags)
Check muting filter.
Definition: main.c:471
usage
static void usage(void)
Print the help message.
Definition: main.c:120
tap.h
is_adapter_name_available
static BOOL is_adapter_name_available(LPCWSTR name, struct tap_adapter_node *adapter_list, BOOL log)
Checks if adapter with given name doesn't already exist.
Definition: main.c:131
wmain
int __cdecl wmain(int argc, LPCWSTR argv[])
Program entry point.
Definition: main.c:211
usage_message_create
static const WCHAR usage_message_create[]
Definition: main.c:61
get_unique_adapter_name
static LPWSTR get_unique_adapter_name(LPCWSTR hwid, struct tap_adapter_node *adapter_list)
Returns unique adapter name based on hwid or NULL if name cannot be generated.
Definition: main.c:158
tap_adapter_node
Network adapter list node.
Definition: tap.h:136
tap_create_adapter
DWORD tap_create_adapter(_In_opt_ HWND hwndParent, _In_opt_ LPCWSTR szDeviceDescription, _In_ LPCWSTR szHwId, _Inout_ LPBOOL pbRebootRequired, _Out_ LPGUID pguidAdapter)
Creates a TUN/TAP adapter.
Definition: tap.c:723
error.h
x_msg_va
void x_msg_va(const unsigned int flags, const char *format, va_list arglist)
Definition: main.c:480
config.h
usage_message
static const WCHAR usage_message[]
Definition: main.c:44
TAP_WIN_COMPONENT_ID
#define TAP_WIN_COMPONENT_ID
Definition: config.h:546