228 lines
7.1 KiB
C
228 lines
7.1 KiB
C
|
/**
|
||
|
* @file tapwin32-funcs.c
|
||
|
* @author Ambroz Bizjak <ambrop7@gmail.com>
|
||
|
*
|
||
|
* @section LICENSE
|
||
|
*
|
||
|
* Redistribution and use in source and binary forms, with or without
|
||
|
* modification, are permitted provided that the following conditions are met:
|
||
|
* 1. Redistributions of source code must retain the above copyright
|
||
|
* notice, this list of conditions and the following disclaimer.
|
||
|
* 2. Redistributions in binary form must reproduce the above copyright
|
||
|
* notice, this list of conditions and the following disclaimer in the
|
||
|
* documentation and/or other materials provided with the distribution.
|
||
|
* 3. Neither the name of the author nor the
|
||
|
* names of its contributors may be used to endorse or promote products
|
||
|
* derived from this software without specific prior written permission.
|
||
|
*
|
||
|
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
|
||
|
* ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
||
|
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
||
|
* DISCLAIMED. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY
|
||
|
* DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
||
|
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
|
||
|
* LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
|
||
|
* ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||
|
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
||
|
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||
|
*/
|
||
|
|
||
|
#include <stddef.h>
|
||
|
#include <stdint.h>
|
||
|
#include <stdlib.h>
|
||
|
#include <string.h>
|
||
|
|
||
|
#include <misc/debug.h>
|
||
|
#include <misc/ipaddr.h>
|
||
|
#include <misc/maxalign.h>
|
||
|
#include <misc/strdup.h>
|
||
|
|
||
|
#include "wintap-common.h"
|
||
|
|
||
|
#include <tuntap/tapwin32-funcs.h>
|
||
|
|
||
|
static int split_spec (char *name, char *sep, char **out_fields[], int num_fields)
|
||
|
{
|
||
|
ASSERT(num_fields > 0)
|
||
|
ASSERT(strlen(sep) > 0)
|
||
|
|
||
|
size_t seplen = strlen(sep);
|
||
|
|
||
|
int i = 0;
|
||
|
while (i < num_fields - 1) {
|
||
|
char *s = strstr(name, sep);
|
||
|
if (!s) {
|
||
|
DEBUG("missing separator number %d", (i + 1));
|
||
|
goto fail;
|
||
|
}
|
||
|
|
||
|
if (!(*out_fields[i] = b_strdup_bin(name, s - name))) {
|
||
|
DEBUG("b_strdup_bin failed");
|
||
|
goto fail;
|
||
|
}
|
||
|
|
||
|
name = s + seplen;
|
||
|
i++;
|
||
|
}
|
||
|
|
||
|
if (!(*out_fields[i] = b_strdup(name))) {
|
||
|
DEBUG("b_strdup_bin failed");
|
||
|
goto fail;
|
||
|
}
|
||
|
|
||
|
return 1;
|
||
|
|
||
|
fail:
|
||
|
while (i-- > 0) {
|
||
|
free(*out_fields[i]);
|
||
|
}
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
int tapwin32_parse_tap_spec (char *name, char **out_component_id, char **out_human_name)
|
||
|
{
|
||
|
char **out_fields[2];
|
||
|
out_fields[0] = out_component_id;
|
||
|
out_fields[1] = out_human_name;
|
||
|
|
||
|
return split_spec(name, ":", out_fields, 2);
|
||
|
}
|
||
|
|
||
|
int tapwin32_parse_tun_spec (char *name, char **out_component_id, char **out_human_name, uint32_t out_addrs[3])
|
||
|
{
|
||
|
char *addr_strs[3];
|
||
|
|
||
|
char **out_fields[5];
|
||
|
out_fields[0] = out_component_id;
|
||
|
out_fields[1] = out_human_name;
|
||
|
out_fields[2] = &addr_strs[0];
|
||
|
out_fields[3] = &addr_strs[1];
|
||
|
out_fields[4] = &addr_strs[2];
|
||
|
|
||
|
if (!split_spec(name, ":", out_fields, 5)) {
|
||
|
goto fail0;
|
||
|
}
|
||
|
|
||
|
for (int i = 0; i < 3; i++) {
|
||
|
if (!ipaddr_parse_ipv4_addr(addr_strs[i], &out_addrs[i])) {
|
||
|
goto fail1;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
free(addr_strs[0]);
|
||
|
free(addr_strs[1]);
|
||
|
free(addr_strs[2]);
|
||
|
|
||
|
return 1;
|
||
|
|
||
|
fail1:
|
||
|
free(*out_component_id);
|
||
|
free(*out_human_name);
|
||
|
free(addr_strs[0]);
|
||
|
free(addr_strs[1]);
|
||
|
free(addr_strs[2]);
|
||
|
fail0:
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
int tapwin32_find_device (char *device_component_id, char *device_name, char (*device_path)[TAPWIN32_MAX_REG_SIZE])
|
||
|
{
|
||
|
// open adapter key
|
||
|
// used to find all devices with the given ComponentId
|
||
|
HKEY adapter_key;
|
||
|
if (RegOpenKeyEx(HKEY_LOCAL_MACHINE, ADAPTER_KEY, 0, KEY_READ, &adapter_key) != ERROR_SUCCESS) {
|
||
|
DEBUG("Error opening adapter key");
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
char net_cfg_instance_id[TAPWIN32_MAX_REG_SIZE];
|
||
|
int found = 0;
|
||
|
int pres;
|
||
|
|
||
|
DWORD i;
|
||
|
for (i = 0;; i++) {
|
||
|
DWORD len;
|
||
|
DWORD type;
|
||
|
|
||
|
char key_name[TAPWIN32_MAX_REG_SIZE];
|
||
|
len = sizeof(key_name);
|
||
|
if (RegEnumKeyEx(adapter_key, i, key_name, &len, NULL, NULL, NULL, NULL) != ERROR_SUCCESS) {
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
char unit_string[TAPWIN32_MAX_REG_SIZE];
|
||
|
pres = _snprintf(unit_string, sizeof(unit_string), "%s\\%s", ADAPTER_KEY, key_name);
|
||
|
if (pres < 0 || pres == sizeof(unit_string)) {
|
||
|
continue;
|
||
|
}
|
||
|
HKEY unit_key;
|
||
|
if (RegOpenKeyEx(HKEY_LOCAL_MACHINE, unit_string, 0, KEY_READ, &unit_key) != ERROR_SUCCESS) {
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
char component_id[TAPWIN32_MAX_REG_SIZE];
|
||
|
len = sizeof(component_id);
|
||
|
if (RegQueryValueEx(unit_key, "ComponentId", NULL, &type, (LPBYTE)component_id, &len) != ERROR_SUCCESS || type != REG_SZ) {
|
||
|
ASSERT_FORCE(RegCloseKey(unit_key) == ERROR_SUCCESS)
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
len = sizeof(net_cfg_instance_id);
|
||
|
if (RegQueryValueEx(unit_key, "NetCfgInstanceId", NULL, &type, (LPBYTE)net_cfg_instance_id, &len) != ERROR_SUCCESS || type != REG_SZ) {
|
||
|
ASSERT_FORCE(RegCloseKey(unit_key) == ERROR_SUCCESS)
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
RegCloseKey(unit_key);
|
||
|
|
||
|
// check if ComponentId matches
|
||
|
if (!strcmp(component_id, device_component_id)) {
|
||
|
// if no name was given, use the first device with the given ComponentId
|
||
|
if (!device_name) {
|
||
|
found = 1;
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
// open connection key
|
||
|
char conn_string[TAPWIN32_MAX_REG_SIZE];
|
||
|
pres = _snprintf(conn_string, sizeof(conn_string), "%s\\%s\\Connection", NETWORK_CONNECTIONS_KEY, net_cfg_instance_id);
|
||
|
if (pres < 0 || pres == sizeof(conn_string)) {
|
||
|
continue;
|
||
|
}
|
||
|
HKEY conn_key;
|
||
|
if (RegOpenKeyEx(HKEY_LOCAL_MACHINE, conn_string, 0, KEY_READ, &conn_key) != ERROR_SUCCESS) {
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
// read name
|
||
|
char name[TAPWIN32_MAX_REG_SIZE];
|
||
|
len = sizeof(name);
|
||
|
if (RegQueryValueEx(conn_key, "Name", NULL, &type, (LPBYTE)name, &len) != ERROR_SUCCESS || type != REG_SZ) {
|
||
|
ASSERT_FORCE(RegCloseKey(conn_key) == ERROR_SUCCESS)
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
ASSERT_FORCE(RegCloseKey(conn_key) == ERROR_SUCCESS)
|
||
|
|
||
|
// check name
|
||
|
if (!strcmp(name, device_name)) {
|
||
|
found = 1;
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
ASSERT_FORCE(RegCloseKey(adapter_key) == ERROR_SUCCESS)
|
||
|
|
||
|
if (!found) {
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
pres = _snprintf(*device_path, sizeof(*device_path), "\\\\.\\Global\\%s.tap", net_cfg_instance_id);
|
||
|
if (pres < 0 || pres == sizeof(*device_path)) {
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
return 1;
|
||
|
}
|