mirror of
https://github.com/mozilla/gecko-dev.git
synced 2024-11-24 13:21:05 +00:00
171 lines
5.0 KiB
C++
171 lines
5.0 KiB
C++
/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
|
|
/* vim: set ts=2 et sw=2 tw=80: */
|
|
/* This Source Code Form is subject to the terms of the Mozilla Public
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this file,
|
|
* You can obtain one at http://mozilla.org/MPL/2.0/. */
|
|
extern "C" {
|
|
#include <arpa/inet.h>
|
|
#include "r_types.h"
|
|
#include "stun.h"
|
|
#include "addrs.h"
|
|
}
|
|
|
|
#include <vector>
|
|
#include <string>
|
|
#include "nsINetworkInterface.h"
|
|
#include "nsINetworkInterfaceListService.h"
|
|
#include "runnable_utils.h"
|
|
#include "nsCOMPtr.h"
|
|
#include "nsMemory.h"
|
|
#include "nsThreadUtils.h"
|
|
#include "nsServiceManagerUtils.h"
|
|
#include "mozilla/SyncRunnable.h"
|
|
|
|
namespace {
|
|
struct NetworkInterface {
|
|
struct sockaddr_in addr;
|
|
std::string name;
|
|
// See NR_INTERFACE_TYPE_* in nICEr/src/net/local_addrs.h
|
|
int type;
|
|
};
|
|
|
|
nsresult
|
|
GetInterfaces(std::vector<NetworkInterface>* aInterfaces)
|
|
{
|
|
MOZ_ASSERT(aInterfaces);
|
|
|
|
// Obtain network interfaces from network manager.
|
|
nsresult rv;
|
|
nsCOMPtr<nsINetworkInterfaceListService> listService =
|
|
do_GetService("@mozilla.org/network/interface-list-service;1", &rv);
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
|
|
int32_t flags =
|
|
nsINetworkInterfaceListService::LIST_NOT_INCLUDE_SUPL_INTERFACES |
|
|
nsINetworkInterfaceListService::LIST_NOT_INCLUDE_MMS_INTERFACES |
|
|
nsINetworkInterfaceListService::LIST_NOT_INCLUDE_IMS_INTERFACES |
|
|
nsINetworkInterfaceListService::LIST_NOT_INCLUDE_DUN_INTERFACES |
|
|
nsINetworkInterfaceListService::LIST_NOT_INCLUDE_FOTA_INTERFACES;
|
|
nsCOMPtr<nsINetworkInterfaceList> networkList;
|
|
NS_ENSURE_SUCCESS(listService->GetDataInterfaceList(flags,
|
|
getter_AddRefs(networkList)),
|
|
NS_ERROR_FAILURE);
|
|
|
|
// Translate nsINetworkInterfaceList to NetworkInterface.
|
|
int32_t listLength;
|
|
NS_ENSURE_SUCCESS(networkList->GetNumberOfInterface(&listLength),
|
|
NS_ERROR_FAILURE);
|
|
aInterfaces->clear();
|
|
|
|
for (int32_t i = 0; i < listLength; i++) {
|
|
nsCOMPtr<nsINetworkInfo> info;
|
|
if (NS_FAILED(networkList->GetInterfaceInfo(i, getter_AddRefs(info)))) {
|
|
continue;
|
|
}
|
|
|
|
char16_t **ips = nullptr;
|
|
uint32_t *prefixs = nullptr;
|
|
uint32_t count = 0;
|
|
bool isAddressGot = false;
|
|
NetworkInterface interface;
|
|
memset(&(interface.addr), 0, sizeof(interface.addr));
|
|
interface.addr.sin_family = AF_INET;
|
|
|
|
if (NS_FAILED(info->GetAddresses(&ips, &prefixs, &count))) {
|
|
continue;
|
|
}
|
|
|
|
for (uint32_t j = 0; j < count; j++) {
|
|
nsAutoString ip;
|
|
|
|
ip.Assign(ips[j]);
|
|
if (inet_pton(AF_INET, NS_ConvertUTF16toUTF8(ip).get(),
|
|
&(interface.addr.sin_addr.s_addr)) == 1) {
|
|
isAddressGot = true;
|
|
break;
|
|
}
|
|
}
|
|
|
|
free(prefixs);
|
|
NS_FREE_XPCOM_ALLOCATED_POINTER_ARRAY(count, ips);
|
|
|
|
if (!isAddressGot) {
|
|
continue;
|
|
}
|
|
|
|
nsAutoString ifaceName;
|
|
if (NS_FAILED(info->GetName(ifaceName))) {
|
|
continue;
|
|
}
|
|
interface.name = NS_ConvertUTF16toUTF8(ifaceName).get();
|
|
|
|
int32_t type;
|
|
if (NS_FAILED(info->GetType(&type))) {
|
|
continue;
|
|
}
|
|
switch (type) {
|
|
case nsINetworkInfo::NETWORK_TYPE_WIFI:
|
|
interface.type = NR_INTERFACE_TYPE_WIFI;
|
|
break;
|
|
case nsINetworkInfo::NETWORK_TYPE_MOBILE:
|
|
interface.type = NR_INTERFACE_TYPE_MOBILE;
|
|
break;
|
|
}
|
|
|
|
aInterfaces->push_back(interface);
|
|
}
|
|
return NS_OK;
|
|
}
|
|
} // anonymous namespace
|
|
|
|
int
|
|
nr_stun_get_addrs(nr_local_addr aAddrs[], int aMaxAddrs,
|
|
int aDropLoopback, int aDropLinkLocal, int* aCount)
|
|
{
|
|
nsresult rv;
|
|
int r;
|
|
|
|
// Get network interface list.
|
|
std::vector<NetworkInterface> interfaces;
|
|
nsCOMPtr<nsIThread> mainThread = do_GetMainThread();
|
|
mozilla::SyncRunnable::DispatchToThread(
|
|
mainThread.get(),
|
|
mozilla::WrapRunnableNMRet(&rv, &GetInterfaces, &interfaces),
|
|
false);
|
|
if (NS_FAILED(rv)) {
|
|
return R_FAILED;
|
|
}
|
|
|
|
// Translate to nr_transport_addr.
|
|
int32_t n = 0;
|
|
size_t num_interface = std::min(interfaces.size(), (size_t)aMaxAddrs);
|
|
for (size_t i = 0; i < num_interface; ++i) {
|
|
NetworkInterface &interface = interfaces[i];
|
|
if (nr_sockaddr_to_transport_addr((sockaddr*)&(interface.addr),
|
|
IPPROTO_UDP, 0, &(aAddrs[n].addr))) {
|
|
r_log(NR_LOG_STUN, LOG_WARNING, "Problem transforming address");
|
|
return R_FAILED;
|
|
}
|
|
strlcpy(aAddrs[n].addr.ifname, interface.name.c_str(),
|
|
sizeof(aAddrs[n].addr.ifname));
|
|
aAddrs[n].interface.type = interface.type;
|
|
aAddrs[n].interface.estimated_speed = 0;
|
|
n++;
|
|
}
|
|
|
|
*aCount = n;
|
|
r = nr_stun_remove_duplicate_addrs(aAddrs, aDropLoopback, aDropLinkLocal, aCount);
|
|
if (r != 0) {
|
|
return r;
|
|
}
|
|
|
|
for (int i = 0; i < *aCount; ++i) {
|
|
char typestr[100];
|
|
nr_local_addr_fmt_info_string(aAddrs + i, typestr, sizeof(typestr));
|
|
r_log(NR_LOG_STUN, LOG_DEBUG, "Address %d: %s on %s, type: %s\n",
|
|
i, aAddrs[i].addr.as_string, aAddrs[i].addr.ifname, typestr);
|
|
}
|
|
|
|
return 0;
|
|
}
|