+++ /dev/null
-// -*- mode:C++; tab-width:8; c-basic-offset:2; indent-tabs-mode:t -*-
-// vim: ts=8 sw=2 smarttab
-/*
- * Ceph - scalable distributed file system
- *
- * Copyright (C) 2004-2012 Inktank
- *
- * This is free software; you can redistribute it and/or
- * modify it under the terms of the GNU Lesser General Public
- * License version 2.1, as published by the Free Software
- * Foundation. See file COPYING.
- *
- */
-
-#include "common/pick_address.h"
-#include "include/ipaddr.h"
-#include "include/str_list.h"
-#include "common/debug.h"
-#include "common/errno.h"
-
-#include <netdb.h>
-
-#define dout_subsys ceph_subsys_
-
-const struct sockaddr *find_ip_in_subnet_list(
- CephContext *cct,
- const struct ifaddrs *ifa,
- const std::string &networks,
- const std::string &interfaces)
-{
- std::list<string> nets;
- get_str_list(networks, nets);
- std::list<string> ifs;
- get_str_list(interfaces, ifs);
-
- // filter interfaces by name
- const struct ifaddrs *filtered = 0;
- if (ifs.empty()) {
- filtered = ifa;
- } else {
- if (nets.empty()) {
- lderr(cct) << "interface names specified but not network names" << dendl;
- exit(1);
- }
- const struct ifaddrs *t = ifa;
- struct ifaddrs *head = 0;
- while (t != NULL) {
- bool match = false;
- for (auto& i : ifs) {
- if (strcmp(i.c_str(), t->ifa_name) == 0) {
- match = true;
- break;
- }
- }
- if (match) {
- struct ifaddrs *n = new ifaddrs;
- memcpy(n, t, sizeof(*t));
- n->ifa_next = head;
- head = n;
- }
- t = t->ifa_next;
- }
- if (head == NULL) {
- lderr(cct) << "no interfaces matching " << ifs << dendl;
- exit(1);
- }
- filtered = head;
- }
-
- struct sockaddr *r = NULL;
- for (std::list<string>::iterator s = nets.begin(); s != nets.end(); ++s) {
- struct sockaddr_storage net;
- unsigned int prefix_len;
-
- if (!parse_network(s->c_str(), &net, &prefix_len)) {
- lderr(cct) << "unable to parse network: " << *s << dendl;
- exit(1);
- }
-
- const struct ifaddrs *found = find_ip_in_subnet(
- filtered,
- (struct sockaddr *) &net, prefix_len);
- if (found) {
- r = found->ifa_addr;
- break;
- }
- }
-
- if (filtered != ifa) {
- while (filtered) {
- struct ifaddrs *t = filtered->ifa_next;
- delete filtered;
- filtered = t;
- }
- }
-
- return r;
-}
-
-// observe this change
-struct Observer : public md_config_obs_t {
- const char *keys[2];
- explicit Observer(const char *c) {
- keys[0] = c;
- keys[1] = NULL;
- }
-
- const char** get_tracked_conf_keys() const override {
- return (const char **)keys;
- }
- void handle_conf_change(const struct md_config_t *conf,
- const std::set <std::string> &changed) override {
- // do nothing.
- }
-};
-
-static void fill_in_one_address(CephContext *cct,
- const struct ifaddrs *ifa,
- const string networks,
- const string interfaces,
- const char *conf_var)
-{
- const struct sockaddr *found = find_ip_in_subnet_list(cct, ifa, networks,
- interfaces);
- if (!found) {
- lderr(cct) << "unable to find any IP address in networks '" << networks
- << "' interfaces '" << interfaces << "'" << dendl;
- exit(1);
- }
-
- char buf[INET6_ADDRSTRLEN];
- int err;
-
- err = getnameinfo(found,
- (found->sa_family == AF_INET)
- ? sizeof(struct sockaddr_in)
- : sizeof(struct sockaddr_in6),
-
- buf, sizeof(buf),
- NULL, 0,
- NI_NUMERICHOST);
- if (err != 0) {
- lderr(cct) << "unable to convert chosen address to string: " << gai_strerror(err) << dendl;
- exit(1);
- }
-
- Observer obs(conf_var);
-
- cct->_conf->add_observer(&obs);
-
- cct->_conf->set_val_or_die(conf_var, buf);
- cct->_conf->apply_changes(NULL);
-
- cct->_conf->remove_observer(&obs);
-}
-
-void pick_addresses(CephContext *cct, int needs)
-{
- struct ifaddrs *ifa;
- int r = getifaddrs(&ifa);
- if (r<0) {
- string err = cpp_strerror(errno);
- lderr(cct) << "unable to fetch interfaces and addresses: " << err << dendl;
- exit(1);
- }
-
- if ((needs & CEPH_PICK_ADDRESS_PUBLIC)
- && cct->_conf->public_addr.is_blank_ip()
- && !cct->_conf->public_network.empty()) {
- fill_in_one_address(cct, ifa, cct->_conf->public_network,
- cct->_conf->get_val<string>("public_network_interface"),
- "public_addr");
- }
-
- if ((needs & CEPH_PICK_ADDRESS_CLUSTER)
- && cct->_conf->cluster_addr.is_blank_ip()) {
- if (!cct->_conf->cluster_network.empty()) {
- fill_in_one_address(
- cct, ifa, cct->_conf->cluster_network,
- cct->_conf->get_val<string>("cluster_network_interface"),
- "cluster_addr");
- } else {
- if (!cct->_conf->public_network.empty()) {
- lderr(cct) << "Public network was set, but cluster network was not set " << dendl;
- lderr(cct) << " Using public network also for cluster network" << dendl;
- fill_in_one_address(
- cct, ifa, cct->_conf->public_network,
- cct->_conf->get_val<string>("public_network_interface"),
- "cluster_addr");
- }
- }
- }
-
- freeifaddrs(ifa);
-}
-
-
-std::string pick_iface(CephContext *cct, const struct sockaddr_storage &network)
-{
- struct ifaddrs *ifa;
- int r = getifaddrs(&ifa);
- if (r < 0) {
- string err = cpp_strerror(errno);
- lderr(cct) << "unable to fetch interfaces and addresses: " << err << dendl;
- return {};
- }
-
- unsigned int prefix_len = 0;
- const struct ifaddrs *found = find_ip_in_subnet(ifa,
- (const struct sockaddr *) &network, prefix_len);
-
- std::string result;
- if (found) {
- result = found->ifa_name;
- }
-
- freeifaddrs(ifa);
-
- return result;
-}
-
-
-bool have_local_addr(CephContext *cct, const list<entity_addr_t>& ls, entity_addr_t *match)
-{
- struct ifaddrs *ifa;
- int r = getifaddrs(&ifa);
- if (r < 0) {
- lderr(cct) << "unable to fetch interfaces and addresses: " << cpp_strerror(errno) << dendl;
- exit(1);
- }
-
- bool found = false;
- for (struct ifaddrs *addrs = ifa; addrs != NULL; addrs = addrs->ifa_next) {
- if (addrs->ifa_addr) {
- entity_addr_t a;
- a.set_sockaddr(addrs->ifa_addr);
- for (list<entity_addr_t>::const_iterator p = ls.begin(); p != ls.end(); ++p) {
- if (a.is_same_host(*p)) {
- *match = *p;
- found = true;
- goto out;
- }
- }
- }
- }
-
- out:
- freeifaddrs(ifa);
- return found;
-}