#include "mpd_defs.h"
#include "mpd_tables.h"
struct phyint *phyints = NULL;
struct phyint_instance *phyint_instances = NULL;
struct phyint_group *phyint_groups = NULL;
struct phyint_group *phyint_anongroup;
static uint64_t phyint_grouplistsig;
static void phyint_inst_insert(struct phyint_instance *pii);
static void phyint_inst_print(struct phyint_instance *pii);
static void phyint_insert(struct phyint *pi, struct phyint_group *pg);
static void phyint_delete(struct phyint *pi);
static boolean_t phyint_is_usable(struct phyint *pi);
static void logint_print(struct logint *li);
static void logint_insert(struct phyint_instance *pii, struct logint *li);
static struct logint *logint_lookup(struct phyint_instance *pii, char *li_name);
static void target_print(struct target *tg);
static void target_insert(struct phyint_instance *pii, struct target *tg);
static struct target *target_first(struct phyint_instance *pii);
static struct target *target_select_best(struct phyint_instance *pii);
static void target_flush_hosts(struct phyint_group *pg);
static void reset_pii_probes(struct phyint_instance *pii, struct target *tg);
static boolean_t phyint_inst_v6_sockinit(struct phyint_instance *pii);
static boolean_t phyint_inst_v4_sockinit(struct phyint_instance *pii);
static int phyint_state_event(struct phyint_group *pg, struct phyint *pi);
static int phyint_group_state_event(struct phyint_group *pg);
static int phyint_group_change_event(struct phyint_group *pg, ipmp_group_op_t);
static int phyint_group_member_event(struct phyint_group *pg, struct phyint *pi,
ipmp_if_op_t op);
static int logint_upcount(struct phyint *pi);
static uint64_t gensig(void);
int
phyint_init(void)
{
phyint_grouplistsig = gensig();
if (track_all_phyints) {
phyint_anongroup = phyint_group_create("");
if (phyint_anongroup == NULL)
return (-1);
phyint_group_insert(phyint_anongroup);
}
return (0);
}
struct phyint *
phyint_lookup(const char *name)
{
struct phyint *pi;
if (debug & D_PHYINT)
logdebug("phyint_lookup(%s)\n", name);
for (pi = phyints; pi != NULL; pi = pi->pi_next) {
if (strncmp(pi->pi_name, name, sizeof (pi->pi_name)) == 0)
break;
}
return (pi);
}
static struct phyint *
phyint_lookup_hwaddr(struct phyint *pi, boolean_t online_only)
{
struct phyint *pi2;
if (pi->pi_group == phyint_anongroup)
return (NULL);
for (pi2 = pi->pi_group->pg_phyint; pi2 != NULL; pi2 = pi2->pi_pgnext) {
if (pi2 == pi)
continue;
if (pi2->pi_state == PI_OFFLINE &&
(online_only || !pi2->pi_hwaddrdup))
continue;
if (pi2->pi_hwaddrlen == pi->pi_hwaddrlen &&
bcmp(pi2->pi_hwaddr, pi->pi_hwaddr, pi->pi_hwaddrlen) == 0)
return (pi2);
}
return (NULL);
}
static void
phyint_link_notify(dlpi_handle_t dh, dlpi_notifyinfo_t *dnip, void *arg)
{
struct phyint *pi = arg;
struct phyint *oduppi = NULL, *duppi = NULL;
assert((dnip->dni_note & pi->pi_notes) != 0);
if (dnip->dni_note != DL_NOTE_PHYS_ADDR)
return;
assert(dnip->dni_physaddrlen <= DLPI_PHYSADDR_MAX);
if (pi->pi_hwaddrlen == dnip->dni_physaddrlen &&
bcmp(pi->pi_hwaddr, dnip->dni_physaddr, pi->pi_hwaddrlen) == 0)
return;
oduppi = phyint_lookup_hwaddr(pi, _B_FALSE);
pi->pi_hwaddrlen = dnip->dni_physaddrlen;
(void) memcpy(pi->pi_hwaddr, dnip->dni_physaddr, pi->pi_hwaddrlen);
duppi = phyint_lookup_hwaddr(pi, _B_FALSE);
if (oduppi != NULL || pi->pi_hwaddrdup) {
if (pi->pi_hwaddrdup) {
if (duppi == NULL)
(void) phyint_undo_offline(pi);
} else {
assert(oduppi->pi_hwaddrdup);
(void) phyint_undo_offline(oduppi);
}
}
if (duppi != NULL && !pi->pi_hwaddrdup) {
pi->pi_hwaddrdup = _B_TRUE;
(void) phyint_offline(pi, 0);
}
}
boolean_t
phyint_link_init(struct phyint *pi)
{
int retval;
uint_t notes;
const char *errmsg;
dlpi_notifyid_t id;
pi->pi_notes = 0;
retval = dlpi_open(pi->pi_name, &pi->pi_dh, 0);
if (retval != DLPI_SUCCESS) {
pi->pi_dh = NULL;
errmsg = "cannot open";
goto failed;
}
pi->pi_hwaddrlen = DLPI_PHYSADDR_MAX;
retval = dlpi_get_physaddr(pi->pi_dh, DL_CURR_PHYS_ADDR, pi->pi_hwaddr,
&pi->pi_hwaddrlen);
if (retval != DLPI_SUCCESS) {
errmsg = "cannot get hardware address";
goto failed;
}
notes = DL_NOTE_LINK_UP | DL_NOTE_LINK_DOWN;
retval = dlpi_enabnotify(pi->pi_dh, notes, phyint_link_notify, pi, &id);
if (retval == DLPI_SUCCESS) {
(void) dlpi_disabnotify(pi->pi_dh, id, NULL);
pi->pi_notes |= notes;
}
notes = DL_NOTE_PHYS_ADDR;
retval = dlpi_enabnotify(pi->pi_dh, notes, phyint_link_notify, pi, &id);
if (retval == DLPI_SUCCESS && poll_add(dlpi_fd(pi->pi_dh)) == 0)
pi->pi_notes |= notes;
return (_B_TRUE);
failed:
logerr("%s: %s: %s\n", pi->pi_name, errmsg, dlpi_strerror(retval));
if (pi->pi_dh != NULL) {
dlpi_close(pi->pi_dh);
pi->pi_dh = NULL;
}
return (_B_FALSE);
}
void
phyint_link_close(struct phyint *pi)
{
if (pi->pi_notes & DL_NOTE_PHYS_ADDR) {
(void) poll_remove(dlpi_fd(pi->pi_dh));
pi->pi_notes &= ~DL_NOTE_PHYS_ADDR;
}
dlpi_close(pi->pi_dh);
pi->pi_dh = NULL;
}
struct phyint_instance *
phyint_inst_lookup(int af, char *name)
{
struct phyint *pi;
if (debug & D_PHYINT)
logdebug("phyint_inst_lookup(%s %s)\n", AF_STR(af), name);
assert(af == AF_INET || af == AF_INET6);
pi = phyint_lookup(name);
if (pi == NULL)
return (NULL);
return (PHYINT_INSTANCE(pi, af));
}
struct phyint_group *
phyint_group_lookup(const char *pg_name)
{
struct phyint_group *pg;
if (debug & D_PHYINT)
logdebug("phyint_group_lookup(%s)\n", pg_name);
for (pg = phyint_groups; pg != NULL; pg = pg->pg_next) {
if (strncmp(pg->pg_name, pg_name, sizeof (pg->pg_name)) == 0)
break;
}
return (pg);
}
static void
phyint_insert(struct phyint *pi, struct phyint_group *pg)
{
if (debug & D_PHYINT)
logdebug("phyint_insert(%s '%s')\n", pi->pi_name, pg->pg_name);
pi->pi_next = phyints;
pi->pi_prev = NULL;
if (phyints != NULL)
phyints->pi_prev = pi;
phyints = pi;
pi->pi_pgnext = NULL;
pi->pi_pgprev = NULL;
pi->pi_group = pg;
pi->pi_pgnext = pg->pg_phyint;
if (pi->pi_pgnext != NULL)
pi->pi_pgnext->pi_pgprev = pi;
pg->pg_phyint = pi;
phyint_group_refresh_state(pg);
pg->pg_sig++;
(void) phyint_group_member_event(pg, pi, IPMP_IF_ADD);
}
static void
phyint_inst_insert(struct phyint_instance *pii)
{
if (debug & D_PHYINT) {
logdebug("phyint_inst_insert(%s %s)\n",
AF_STR(pii->pii_af), pii->pii_name);
}
pii->pii_next = phyint_instances;
pii->pii_prev = NULL;
if (phyint_instances != NULL)
phyint_instances->pii_prev = pii;
phyint_instances = pii;
}
static struct phyint *
phyint_create(char *pi_name, struct phyint_group *pg, uint_t ifindex,
uint64_t flags)
{
struct phyint *pi;
pi = calloc(1, sizeof (struct phyint));
if (pi == NULL) {
logperror("phyint_create: calloc");
return (NULL);
}
(void) strlcpy(pi->pi_name, pi_name, sizeof (pi->pi_name));
pi->pi_taddrthresh = getcurrentsec() + TESTADDR_CONF_TIME;
pi->pi_ifindex = ifindex;
pi->pi_icmpid = htons(((getpid() & 0xFF) << 8) | (ifindex & 0xFF));
pi->pi_state = PI_INIT;
pi->pi_flags = PHYINT_FLAGS(flags);
INIT_LINK_STATE(pi);
if (!phyint_link_init(pi)) {
free(pi);
return (NULL);
}
phyint_insert(pi, pg);
return (pi);
}
static struct phyint_instance *
phyint_inst_create(struct phyint *pi, int af)
{
struct phyint_instance *pii;
pii = calloc(1, sizeof (struct phyint_instance));
if (pii == NULL) {
logperror("phyint_inst_create: calloc");
return (NULL);
}
pii->pii_phyint = pi;
if (af == AF_INET)
pi->pi_v4 = pii;
else
pi->pi_v6 = pii;
pii->pii_in_use = 1;
pii->pii_probe_sock = -1;
pii->pii_snxt = 1;
pii->pii_af = af;
pii->pii_fd_hrtime = gethrtime() +
(FAILURE_DETECTION_QP * (hrtime_t)NANOSEC);
pii->pii_flags = pi->pi_flags;
phyint_inst_insert(pii);
return (pii);
}
void
phyint_chstate(struct phyint *pi, enum pi_state state)
{
if (pi->pi_state == state)
return;
pi->pi_state = state;
phyint_changed(pi);
}
void
phyint_changed(struct phyint *pi)
{
pi->pi_group->pg_sig++;
(void) phyint_state_event(pi->pi_group, pi);
}
void
phyint_group_insert(struct phyint_group *pg)
{
pg->pg_next = phyint_groups;
pg->pg_prev = NULL;
if (phyint_groups != NULL)
phyint_groups->pg_prev = pg;
phyint_groups = pg;
phyint_grouplistsig++;
(void) phyint_group_change_event(pg, IPMP_GROUP_ADD);
}
struct phyint_group *
phyint_group_create(const char *name)
{
struct phyint_group *pg;
if (debug & D_PHYINT)
logdebug("phyint_group_create(%s)\n", name);
pg = calloc(1, sizeof (struct phyint_group));
if (pg == NULL) {
logperror("phyint_group_create: calloc");
return (NULL);
}
(void) strlcpy(pg->pg_name, name, sizeof (pg->pg_name));
pg->pg_sig = gensig();
pg->pg_fdt = user_failure_detection_time;
pg->pg_probeint = user_probe_interval;
pg->pg_in_use = _B_TRUE;
pg->pg_state = (name[0] == '\0' ? PG_OK : PG_FAILED);
return (pg);
}
void
phyint_group_chstate(struct phyint_group *pg, enum pg_state state)
{
assert(pg != phyint_anongroup);
if (pg->pg_state == state)
return;
pg->pg_state = state;
switch (state) {
case PG_FAILED:
target_flush_hosts(pg);
break;
case PG_OK:
case PG_DEGRADED:
break;
default:
logerr("phyint_group_chstate: invalid group state %d; "
"aborting\n", state);
abort();
}
pg->pg_sig++;
(void) phyint_group_state_event(pg);
}
struct phyint_instance *
phyint_inst_init_from_k(int af, char *pi_name)
{
char pg_name[LIFNAMSIZ + 1];
int ifsock;
uint_t ifindex;
uint64_t flags;
struct lifreq lifr;
struct phyint *pi;
struct phyint_instance *pii;
boolean_t pi_created;
struct phyint_group *pg;
retry:
pii = NULL;
pi = NULL;
pg = NULL;
pi_created = _B_FALSE;
if (debug & D_PHYINT) {
logdebug("phyint_inst_init_from_k(%s %s)\n",
AF_STR(af), pi_name);
}
assert(af == AF_INET || af == AF_INET6);
ifsock = (af == AF_INET) ? ifsock_v4 : ifsock_v6;
(void) strlcpy(lifr.lifr_name, pi_name, sizeof (lifr.lifr_name));
if (ioctl(ifsock, SIOCGLIFFLAGS, (char *)&lifr) < 0) {
if (errno != ENXIO) {
logperror("phyint_inst_init_from_k:"
" ioctl (get flags)");
}
return (NULL);
}
flags = lifr.lifr_flags;
if (!(flags & IFF_MULTICAST) ||
(flags & (IFF_VIRTUAL|IFF_IPMP|IFF_POINTOPOINT)))
return (NULL);
if (ioctl(ifsock, SIOCGLIFINDEX, (char *)&lifr) < 0) {
if (errno != ENXIO) {
logperror("phyint_inst_init_from_k: "
" ioctl (get lifindex)");
}
return (NULL);
}
ifindex = lifr.lifr_index;
if (ioctl(ifsock, SIOCGLIFGROUPNAME, (char *)&lifr) < 0) {
if (errno != ENXIO) {
logperror("phyint_inst_init_from_k: "
"ioctl (get group name)");
}
return (NULL);
}
(void) strlcpy(pg_name, lifr.lifr_groupname, sizeof (pg_name));
if (pg_name[0] == '\0' && !track_all_phyints) {
if ((flags & (IFF_FAILED | IFF_INACTIVE | IFF_OFFLINE))) {
lifr.lifr_flags = flags &
~(IFF_FAILED | IFF_INACTIVE | IFF_OFFLINE);
if (ioctl(ifsock, SIOCSLIFFLAGS, (char *)&lifr) < 0) {
if (errno != ENXIO) {
logperror("phyint_inst_init_from_k:"
" ioctl (set flags)");
}
}
}
return (NULL);
}
if ((pg = phyint_group_lookup(pg_name)) == NULL) {
if ((pg = phyint_group_create(pg_name)) == NULL) {
logerr("phyint_inst_init_from_k: cannot create group "
"%s\n", pg_name);
return (NULL);
}
phyint_group_insert(pg);
}
pi = phyint_lookup(pi_name);
if (pi == NULL) {
pi = phyint_create(pi_name, pg, ifindex, flags);
if (pi == NULL) {
logerr("phyint_inst_init_from_k:"
" unable to create phyint %s\n", pi_name);
return (NULL);
}
pi_created = _B_TRUE;
} else {
assert(pi_created == _B_FALSE);
if (pi->pi_ifindex != ifindex) {
phyint_inst_delete(PHYINT_INSTANCE(pi, AF_OTHER(af)));
goto retry;
}
assert(PHYINT_INSTANCE(pi, af) == NULL);
if (strcmp(pi->pi_group->pg_name, pg_name) != 0) {
phyint_inst_delete(PHYINT_INSTANCE(pi,
AF_OTHER(af)));
goto retry;
}
}
pii = phyint_inst_create(pi, af);
if (pii == NULL) {
logerr("phyint_inst_init_from_k: unable to create"
"phyint inst %s\n", pi->pi_name);
if (pi_created)
phyint_delete(pi);
return (NULL);
}
if (pi_created) {
if (pi->pi_flags & IFF_OFFLINE) {
phyint_chstate(pi, PI_OFFLINE);
} else {
phyint_transition_to_running(pi);
}
if (pi->pi_flags & IFF_STANDBY)
phyint_standby_refresh_inactive(pi);
if (phyint_lookup_hwaddr(pi, _B_TRUE) != NULL) {
pi->pi_hwaddrdup = _B_TRUE;
(void) phyint_offline(pi, 0);
}
}
return (pii);
}
boolean_t
phyint_inst_sockinit(struct phyint_instance *pii)
{
boolean_t success;
struct phyint_group *pg;
if (debug & D_PHYINT) {
logdebug("phyint_inst_sockinit(%s %s)\n",
AF_STR(pii->pii_af), pii->pii_name);
}
assert(pii->pii_probe_logint != NULL);
assert(pii->pii_probe_logint->li_flags & IFF_UP);
assert(pii->pii_probe_logint->li_flags & IFF_NOFAILOVER);
assert(pii->pii_af == AF_INET || pii->pii_af == AF_INET6);
if (pii->pii_probe_sock != -1)
close_probe_socket(pii, _B_TRUE);
pg = pii->pii_phyint->pi_group;
if (pg == phyint_anongroup && !track_all_phyints) {
if (debug & D_PHYINT)
logdebug("phyint_inst_sockinit: no group\n");
return (_B_FALSE);
}
if (pii->pii_af == AF_INET6)
success = phyint_inst_v6_sockinit(pii);
else
success = phyint_inst_v4_sockinit(pii);
if (success && (poll_add(pii->pii_probe_sock) == 0))
return (_B_TRUE);
if (pii->pii_probe_sock != -1)
close_probe_socket(pii, _B_FALSE);
return (_B_FALSE);
}
static boolean_t
phyint_inst_v6_sockinit(struct phyint_instance *pii)
{
icmp6_filter_t filter;
int hopcount = 1;
int off = 0;
int on = 1;
struct sockaddr_in6 testaddr;
int flags;
pii->pii_probe_sock = socket(pii->pii_af, SOCK_RAW, IPPROTO_ICMPV6);
if (pii->pii_probe_sock < 0) {
logperror_pii(pii, "phyint_inst_v6_sockinit: socket");
return (_B_FALSE);
}
if ((flags = fcntl(pii->pii_probe_sock, F_GETFL, 0)) == -1) {
logperror_pii(pii, "phyint_inst_v6_sockinit: fcntl"
" F_GETFL");
return (_B_FALSE);
}
if (fcntl(pii->pii_probe_sock, F_SETFL,
flags | O_NONBLOCK) == -1) {
logperror_pii(pii, "phyint_inst_v6_sockinit: fcntl"
" F_SETFL O_NONBLOCK");
return (_B_FALSE);
}
bzero(&testaddr, sizeof (testaddr));
testaddr.sin6_family = AF_INET6;
testaddr.sin6_port = 0;
testaddr.sin6_addr = pii->pii_probe_logint->li_addr;
if (bind(pii->pii_probe_sock, (struct sockaddr *)&testaddr,
sizeof (testaddr)) < 0) {
logperror_pii(pii, "phyint_inst_v6_sockinit: IPv6 bind");
return (_B_FALSE);
}
if (setsockopt(pii->pii_probe_sock, IPPROTO_IPV6, IPV6_MULTICAST_IF,
(char *)&pii->pii_ifindex, sizeof (uint_t)) < 0) {
logperror_pii(pii, "phyint_inst_v6_sockinit: setsockopt"
" IPV6_MULTICAST_IF");
return (_B_FALSE);
}
if (setsockopt(pii->pii_probe_sock, IPPROTO_IPV6, IPV6_BOUND_IF,
&pii->pii_ifindex, sizeof (uint_t)) < 0) {
logperror_pii(pii, "phyint_inst_v6_sockinit: setsockopt"
" IPV6_BOUND_IF");
return (_B_FALSE);
}
if (setsockopt(pii->pii_probe_sock, IPPROTO_IPV6, IPV6_UNICAST_HOPS,
(char *)&hopcount, sizeof (hopcount)) < 0) {
logperror_pii(pii, "phyint_inst_v6_sockinit: setsockopt"
" IPV6_UNICAST_HOPS");
return (_B_FALSE);
}
if (setsockopt(pii->pii_probe_sock, IPPROTO_IPV6, IPV6_MULTICAST_HOPS,
(char *)&hopcount, sizeof (hopcount)) < 0) {
logperror_pii(pii, "phyint_inst_v6_sockinit: setsockopt"
" IPV6_MULTICAST_HOPS");
return (_B_FALSE);
}
if (setsockopt(pii->pii_probe_sock, IPPROTO_IPV6, IPV6_MULTICAST_LOOP,
(char *)&off, sizeof (off)) < 0) {
logperror_pii(pii, "phyint_inst_v6_sockinit: setsockopt"
" IPV6_MULTICAST_LOOP");
return (_B_FALSE);
}
ICMP6_FILTER_SETBLOCKALL(&filter);
ICMP6_FILTER_SETPASS(ICMP6_ECHO_REPLY, &filter);
if (setsockopt(pii->pii_probe_sock, IPPROTO_ICMPV6, ICMP6_FILTER,
(char *)&filter, sizeof (filter)) < 0) {
logperror_pii(pii, "phyint_inst_v6_sockinit: setsockopt"
" ICMP6_FILTER");
return (_B_FALSE);
}
if (setsockopt(pii->pii_probe_sock, IPPROTO_IPV6, IPV6_RECVHOPLIMIT,
&on, sizeof (on)) < 0) {
logperror_pii(pii, "phyint_inst_v6_sockinit: setsockopt"
" IPV6_RECVHOPLIMIT");
return (_B_FALSE);
}
if (setsockopt(pii->pii_probe_sock, SOL_SOCKET, SO_TIMESTAMP,
&on, sizeof (on)) < 0) {
logperror_pii(pii, "phyint_inst_v6_sockinit: setsockopt"
" SO_TIMESTAMP");
return (_B_FALSE);
}
return (_B_TRUE);
}
static boolean_t
phyint_inst_v4_sockinit(struct phyint_instance *pii)
{
struct sockaddr_in testaddr;
char char_off = 0;
int ttl = 1;
char char_ttl = 1;
int on = 1;
int flags;
pii->pii_probe_sock = socket(pii->pii_af, SOCK_RAW, IPPROTO_ICMP);
if (pii->pii_probe_sock < 0) {
logperror_pii(pii, "phyint_inst_v4_sockinit: socket");
return (_B_FALSE);
}
if ((flags = fcntl(pii->pii_probe_sock, F_GETFL, 0)) == -1) {
logperror_pii(pii, "phyint_inst_v4_sockinit: fcntl"
" F_GETFL");
return (_B_FALSE);
}
if (fcntl(pii->pii_probe_sock, F_SETFL,
flags | O_NONBLOCK) == -1) {
logperror_pii(pii, "phyint_inst_v4_sockinit: fcntl"
" F_SETFL O_NONBLOCK");
return (_B_FALSE);
}
bzero(&testaddr, sizeof (testaddr));
testaddr.sin_family = AF_INET;
testaddr.sin_port = 0;
IN6_V4MAPPED_TO_INADDR(&pii->pii_probe_logint->li_addr,
&testaddr.sin_addr);
if (bind(pii->pii_probe_sock, (struct sockaddr *)&testaddr,
sizeof (testaddr)) < 0) {
logperror_pii(pii, "phyint_inst_v4_sockinit: IPv4 bind");
return (_B_FALSE);
}
if (setsockopt(pii->pii_probe_sock, IPPROTO_IP, IP_BOUND_IF,
&pii->pii_ifindex, sizeof (uint_t)) < 0) {
logperror_pii(pii, "phyint_inst_v4_sockinit: setsockopt"
" IP_BOUND_IF");
return (_B_FALSE);
}
if (setsockopt(pii->pii_probe_sock, IPPROTO_IP, IP_MULTICAST_IF,
(char *)&testaddr.sin_addr, sizeof (struct in_addr)) < 0) {
logperror_pii(pii, "phyint_inst_v4_sockinit: setsockopt"
" IP_MULTICAST_IF");
return (_B_FALSE);
}
if (setsockopt(pii->pii_probe_sock, IPPROTO_IP, IP_TTL,
(char *)&ttl, sizeof (ttl)) < 0) {
logperror_pii(pii, "phyint_inst_v4_sockinit: setsockopt"
" IP_TTL");
return (_B_FALSE);
}
if (setsockopt(pii->pii_probe_sock, IPPROTO_IP, IP_MULTICAST_LOOP,
(char *)&char_off, sizeof (char_off)) == -1) {
logperror_pii(pii, "phyint_inst_v4_sockinit: setsockopt"
" IP_MULTICAST_LOOP");
return (_B_FALSE);
}
if (setsockopt(pii->pii_probe_sock, IPPROTO_IP, IP_MULTICAST_TTL,
(char *)&char_ttl, sizeof (char_ttl)) == -1) {
logperror_pii(pii, "phyint_inst_v4_sockinit: setsockopt"
" IP_MULTICAST_TTL");
return (_B_FALSE);
}
if (setsockopt(pii->pii_probe_sock, SOL_SOCKET, SO_TIMESTAMP, &on,
sizeof (on)) < 0) {
logperror_pii(pii, "phyint_inst_v4_sockinit: setsockopt"
" SO_TIMESTAMP");
return (_B_FALSE);
}
return (_B_TRUE);
}
void
phyint_group_delete(struct phyint_group *pg)
{
if (pg == phyint_anongroup)
return;
if (debug & D_PHYINT)
logdebug("phyint_group_delete('%s')\n", pg->pg_name);
assert(pg->pg_phyint == NULL);
assert(phyint_groups == pg || pg->pg_prev != NULL);
if (pg->pg_prev != NULL)
pg->pg_prev->pg_next = pg->pg_next;
else
phyint_groups = pg->pg_next;
if (pg->pg_next != NULL)
pg->pg_next->pg_prev = pg->pg_prev;
pg->pg_next = NULL;
pg->pg_prev = NULL;
phyint_grouplistsig++;
(void) phyint_group_change_event(pg, IPMP_GROUP_REMOVE);
addrlist_free(&pg->pg_addrs);
free(pg);
}
void
phyint_group_refresh_state(struct phyint_group *pg)
{
enum pg_state state;
enum pg_state origstate = pg->pg_state;
struct phyint *pi, *usablepi;
uint_t nif = 0, nusable = 0;
if (pg == phyint_anongroup)
return;
for (pi = pg->pg_phyint; pi != NULL; pi = pi->pi_pgnext) {
nif++;
if (phyint_is_usable(pi)) {
nusable++;
usablepi = pi;
}
}
if (nusable == 0)
state = PG_FAILED;
else if (nif == nusable)
state = PG_OK;
else
state = PG_DEGRADED;
phyint_group_chstate(pg, state);
if (cleanup_started)
return;
if (state != PG_FAILED && pg->pg_failmsg_printed) {
assert(origstate == PG_FAILED);
logerr("At least 1 IP interface (%s) in group %s is now "
"usable\n", usablepi->pi_name, pg->pg_name);
pg->pg_failmsg_printed = _B_FALSE;
} else if (origstate != PG_FAILED && state == PG_FAILED) {
logerr("All IP interfaces in group %s are now unusable\n",
pg->pg_name);
pg->pg_failmsg_printed = _B_TRUE;
}
}
int
phyint_inst_update_from_k(struct phyint_instance *pii)
{
struct lifreq lifr;
int ifsock;
struct phyint *pi;
pi = pii->pii_phyint;
if (debug & D_PHYINT) {
logdebug("phyint_inst_update_from_k(%s %s)\n",
AF_STR(pii->pii_af), pi->pi_name);
}
(void) strncpy(lifr.lifr_name, pi->pi_name, sizeof (lifr.lifr_name));
lifr.lifr_name[sizeof (lifr.lifr_name) - 1] = '\0';
ifsock = (pii->pii_af == AF_INET) ? ifsock_v4 : ifsock_v6;
if (ioctl(ifsock, SIOCGLIFINDEX, &lifr) < 0) {
if (errno == ENXIO) {
return (PI_DELETED);
} else {
logperror_pii(pii, "phyint_inst_update_from_k:"
" ioctl (get lifindex)");
return (PI_IOCTL_ERROR);
}
}
if (lifr.lifr_index != pi->pi_ifindex) {
if (debug & D_PHYINT) {
logdebug("phyint_inst_update_from_k:"
" old index %d new index %d\n",
pi->pi_ifindex, lifr.lifr_index);
}
return (PI_IFINDEX_CHANGED);
}
if (ioctl(ifsock, SIOCGLIFGROUPNAME, &lifr) < 0) {
if (errno == ENXIO) {
return (PI_DELETED);
} else {
logperror_pii(pii, "phyint_inst_update_from_k:"
" ioctl (get groupname)");
return (PI_IOCTL_ERROR);
}
}
if (strcmp(lifr.lifr_groupname, pi->pi_group->pg_name) != 0) {
if (debug & D_PHYINT) {
logdebug("phyint_inst_update_from_k:"
" groupname change\n");
}
return (PI_GROUP_CHANGED);
}
if (ioctl(ifsock, SIOCGLIFFLAGS, &lifr) < 0) {
if (errno == ENXIO) {
return (PI_DELETED);
} else {
logperror_pii(pii, "phyint_inst_update_from_k: "
" ioctl (get flags)");
return (PI_IOCTL_ERROR);
}
}
pi->pi_flags = PHYINT_FLAGS(lifr.lifr_flags);
if (pi->pi_v4 != NULL)
pi->pi_v4->pii_flags = pi->pi_flags;
if (pi->pi_v6 != NULL)
pi->pi_v6->pii_flags = pi->pi_flags;
if (pi->pi_flags & IFF_FAILED) {
if (pi->pi_state == PI_RUNNING)
(void) change_pif_flags(pi, 0, IFF_FAILED);
} else {
if (pi->pi_state == PI_FAILED)
(void) change_pif_flags(pi, IFF_FAILED, IFF_INACTIVE);
}
return (PI_OK);
}
static void
phyint_delete(struct phyint *pi)
{
boolean_t active;
struct phyint *pi2;
struct phyint_group *pg = pi->pi_group;
if (debug & D_PHYINT)
logdebug("phyint_delete(%s)\n", pi->pi_name);
assert(pi->pi_v4 == NULL && pi->pi_v6 == NULL);
assert(pg->pg_phyint == pi || pi->pi_pgprev != NULL);
assert(phyints == pi || pi->pi_prev != NULL);
pg->pg_sig++;
(void) phyint_group_member_event(pg, pi, IPMP_IF_REMOVE);
if (pi->pi_pgprev == NULL) {
pg->pg_phyint = pi->pi_pgnext;
} else {
pi->pi_pgprev->pi_pgnext = pi->pi_pgnext;
}
if (pi->pi_pgnext != NULL)
pi->pi_pgnext->pi_pgprev = pi->pi_pgprev;
pi->pi_pgnext = NULL;
pi->pi_pgprev = NULL;
phyint_group_refresh_state(pg);
if (pi->pi_prev == NULL) {
phyints = pi->pi_next;
} else {
pi->pi_prev->pi_next = pi->pi_next;
}
if (pi->pi_next != NULL)
pi->pi_next->pi_prev = pi->pi_prev;
pi->pi_next = NULL;
pi->pi_prev = NULL;
if (!pi->pi_hwaddrdup &&
(pi2 = phyint_lookup_hwaddr(pi, _B_FALSE)) != NULL) {
assert(pi2->pi_hwaddrdup);
(void) phyint_undo_offline(pi2);
}
if (pg != phyint_anongroup) {
active = _B_FALSE;
for (pi2 = pg->pg_phyint; pi2 != NULL; pi2 = pi2->pi_pgnext) {
if (phyint_is_functioning(pi2) &&
!(pi2->pi_flags & IFF_INACTIVE)) {
active = _B_TRUE;
break;
}
}
if (!active ||
(pi->pi_flags & (IFF_STANDBY|IFF_INACTIVE)) == IFF_STANDBY)
phyint_activate_another(pi);
}
phyint_link_close(pi);
free(pi);
}
int
phyint_offline(struct phyint *pi, uint_t minred)
{
boolean_t was_active;
unsigned int nusable = 0;
struct phyint *pi2;
struct phyint_group *pg = pi->pi_group;
if (pg != phyint_anongroup) {
for (pi2 = pg->pg_phyint; pi2 != NULL; pi2 = pi2->pi_pgnext) {
if (pi2 == pi)
continue;
if (phyint_is_usable(pi2) ||
(GROUP_FAILED(pg) && pi2->pi_state != PI_OFFLINE))
nusable++;
}
}
if (nusable < minred)
return (IPMP_EMINRED);
was_active = ((pi->pi_flags & IFF_INACTIVE) == 0);
if (!change_pif_flags(pi, IFF_OFFLINE, IFF_INACTIVE))
return (IPMP_FAILURE);
stop_probing(pi);
if (pi->pi_hwaddrdup) {
logerr("IP interface %s has a hardware address which is not "
"unique in group %s; offlining\n", pi->pi_name,
pg->pg_name);
} else {
phyint_link_close(pi);
}
if (!pi->pi_hwaddrdup &&
(pi2 = phyint_lookup_hwaddr(pi, _B_FALSE)) != NULL) {
assert(pi2->pi_hwaddrdup);
(void) phyint_undo_offline(pi2);
}
if (was_active)
phyint_activate_another(pi);
return (IPMP_SUCCESS);
}
int
phyint_undo_offline(struct phyint *pi)
{
if (pi->pi_state != PI_OFFLINE) {
errno = EINVAL;
return (IPMP_FAILURE);
}
if (pi->pi_dh == NULL && !phyint_link_init(pi)) {
errno = EIO;
return (IPMP_FAILURE);
}
if (phyint_lookup_hwaddr(pi, _B_TRUE) != NULL) {
pi->pi_hwaddrdup = _B_TRUE;
return (IPMP_EHWADDRDUP);
}
if (pi->pi_hwaddrdup) {
logerr("IP interface %s now has a unique hardware address in "
"group %s; onlining\n", pi->pi_name, pi->pi_group->pg_name);
pi->pi_hwaddrdup = _B_FALSE;
}
if (!change_pif_flags(pi, 0, IFF_OFFLINE))
return (IPMP_FAILURE);
if (pi->pi_flags & IFF_FAILED) {
phyint_chstate(pi, PI_FAILED);
} else {
phyint_transition_to_running(pi);
}
pi->pi_taddrthresh = getcurrentsec() + TESTADDR_CONF_TIME;
return (IPMP_SUCCESS);
}
void
phyint_inst_delete(struct phyint_instance *pii)
{
struct phyint *pi = pii->pii_phyint;
assert(pi != NULL);
if (debug & D_PHYINT) {
logdebug("phyint_inst_delete(%s %s)\n",
AF_STR(pii->pii_af), pi->pi_name);
}
while (pii->pii_targets != NULL)
target_delete(pii->pii_targets);
while (pii->pii_logint != NULL)
logint_delete(pii->pii_logint);
if (pii->pii_probe_sock != -1)
close_probe_socket(pii, _B_TRUE);
assert(phyint_instances == pii || pii->pii_prev != NULL);
if (pii->pii_prev == NULL) {
phyint_instances = pii->pii_next;
} else {
pii->pii_prev->pii_next = pii->pii_next;
}
if (pii->pii_next != NULL)
pii->pii_next->pii_prev = pii->pii_prev;
pii->pii_next = NULL;
pii->pii_prev = NULL;
if (pii->pii_af == AF_INET)
pi->pi_v4 = NULL;
else
pi->pi_v6 = NULL;
if (pi->pi_v4 == NULL && pi->pi_v6 == NULL)
phyint_delete(pi);
free(pii);
}
static void
phyint_inst_print(struct phyint_instance *pii)
{
struct logint *li;
struct target *tg;
char abuf[INET6_ADDRSTRLEN];
int most_recent;
int i;
if (pii->pii_phyint == NULL) {
logdebug("pii->pi_phyint NULL can't print\n");
return;
}
logdebug("\nPhyint instance: %s %s index %u state %x flags %llx "
"sock %x in_use %d\n",
AF_STR(pii->pii_af), pii->pii_name, pii->pii_ifindex,
pii->pii_state, pii->pii_phyint->pi_flags, pii->pii_probe_sock,
pii->pii_in_use);
for (li = pii->pii_logint; li != NULL; li = li->li_next)
logint_print(li);
logdebug("\n");
for (tg = pii->pii_targets; tg != NULL; tg = tg->tg_next)
target_print(tg);
if (pii->pii_targets == NULL)
logdebug("pi_targets NULL\n");
if (pii->pii_target_next != NULL) {
logdebug("pi_target_next %s %s\n", AF_STR(pii->pii_af),
pr_addr(pii->pii_af, pii->pii_target_next->tg_address,
abuf, sizeof (abuf)));
} else {
logdebug("pi_target_next NULL\n");
}
if (pii->pii_rtt_target_next != NULL) {
logdebug("pi_rtt_target_next %s %s\n", AF_STR(pii->pii_af),
pr_addr(pii->pii_af, pii->pii_rtt_target_next->tg_address,
abuf, sizeof (abuf)));
} else {
logdebug("pi_rtt_target_next NULL\n");
}
if (pii->pii_targets != NULL) {
most_recent = PROBE_INDEX_PREV(pii->pii_probe_next);
i = most_recent;
do {
if (pii->pii_probes[i].pr_target != NULL) {
logdebug("#%d target %s ", i,
pr_addr(pii->pii_af,
pii->pii_probes[i].pr_target->tg_address,
abuf, sizeof (abuf)));
} else {
logdebug("#%d target NULL ", i);
}
logdebug("time_start %lld status %d "
"time_ackproc %lld time_lost %u",
pii->pii_probes[i].pr_hrtime_start,
pii->pii_probes[i].pr_status,
pii->pii_probes[i].pr_hrtime_ackproc,
pii->pii_probes[i].pr_time_lost);
i = PROBE_INDEX_PREV(i);
} while (i != most_recent);
}
}
static struct logint *
logint_lookup(struct phyint_instance *pii, char *name)
{
struct logint *li;
if (debug & D_LOGINT) {
logdebug("logint_lookup(%s, %s)\n",
AF_STR(pii->pii_af), name);
}
for (li = pii->pii_logint; li != NULL; li = li->li_next) {
if (strncmp(name, li->li_name, sizeof (li->li_name)) == 0)
break;
}
return (li);
}
static void
logint_insert(struct phyint_instance *pii, struct logint *li)
{
li->li_next = pii->pii_logint;
li->li_prev = NULL;
if (pii->pii_logint != NULL)
pii->pii_logint->li_prev = li;
pii->pii_logint = li;
li->li_phyint_inst = pii;
}
static struct logint *
logint_create(struct phyint_instance *pii, char *name)
{
struct logint *li;
if (debug & D_LOGINT) {
logdebug("logint_create(%s %s %s)\n",
AF_STR(pii->pii_af), pii->pii_name, name);
}
li = calloc(1, sizeof (struct logint));
if (li == NULL) {
logperror("logint_create: calloc");
return (NULL);
}
(void) strncpy(li->li_name, name, sizeof (li->li_name));
li->li_name[sizeof (li->li_name) - 1] = '\0';
logint_insert(pii, li);
return (li);
}
void
logint_init_from_k(struct phyint_instance *pii, char *li_name)
{
int ifsock;
uint64_t flags;
uint64_t saved_flags;
struct logint *li;
struct lifreq lifr;
struct in6_addr test_subnet;
struct in6_addr testaddr;
int test_subnet_len;
struct sockaddr_in6 *sin6;
struct sockaddr_in *sin;
char abuf[INET6_ADDRSTRLEN];
boolean_t ptp = _B_FALSE;
struct in6_addr tgaddr;
if (debug & D_LOGINT) {
logdebug("logint_init_from_k(%s %s)\n",
AF_STR(pii->pii_af), li_name);
}
ifsock = (pii->pii_af == AF_INET) ? ifsock_v4 : ifsock_v6;
(void) strncpy(lifr.lifr_name, li_name, sizeof (lifr.lifr_name));
lifr.lifr_name[sizeof (lifr.lifr_name) - 1] = '\0';
if (ioctl(ifsock, SIOCGLIFFLAGS, (char *)&lifr) < 0) {
if (errno != ENXIO) {
logperror_pii(pii, "logint_init_from_k: "
"ioctl (get flags)");
}
return;
}
flags = lifr.lifr_flags;
li = logint_lookup(pii, li_name);
if (li == NULL) {
li = logint_create(pii, li_name);
if (li == NULL) {
return;
}
}
saved_flags = li->li_flags;
li->li_flags = flags;
if (ioctl(ifsock, SIOCGLIFADDR, (char *)&lifr) < 0) {
if (errno != ENXIO) {
logperror_li(li, "logint_init_from_k: (get addr)");
}
goto error;
}
if (pii->pii_af == AF_INET) {
sin = (struct sockaddr_in *)&lifr.lifr_addr;
IN6_INADDR_TO_V4MAPPED(&sin->sin_addr, &testaddr);
} else {
sin6 = (struct sockaddr_in6 *)&lifr.lifr_addr;
testaddr = sin6->sin6_addr;
}
if (ioctl(ifsock, SIOCGLIFSUBNET, (char *)&lifr) < 0) {
if (errno != ENXIO)
logperror_li(li, "logint_init_from_k: (get subnet)");
goto error;
}
if (lifr.lifr_subnet.ss_family == AF_INET6) {
sin6 = (struct sockaddr_in6 *)&lifr.lifr_subnet;
test_subnet = sin6->sin6_addr;
test_subnet_len = lifr.lifr_addrlen;
} else {
sin = (struct sockaddr_in *)&lifr.lifr_subnet;
IN6_INADDR_TO_V4MAPPED(&sin->sin_addr, &test_subnet);
test_subnet_len = lifr.lifr_addrlen + (IPV6_ABITS - IP_ABITS);
}
if (pii->pii_probe_logint == li) {
if (((li->li_flags ^ saved_flags) &
~(IFF_FAILED | IFF_RUNNING)) != 0 ||
!IN6_ARE_ADDR_EQUAL(&testaddr, &li->li_addr) ||
(!ptp && !IN6_ARE_ADDR_EQUAL(&test_subnet,
&li->li_subnet)) ||
(!ptp && test_subnet_len != li->li_subnet_len) ||
(ptp && !IN6_ARE_ADDR_EQUAL(&tgaddr, &li->li_dstaddr))) {
if (pii->pii_probe_sock != -1)
close_probe_socket(pii, _B_TRUE);
pii->pii_probe_logint = NULL;
}
}
li->li_addr = testaddr;
li->li_in_use = 1;
if (ptp) {
li->li_dstaddr = tgaddr;
li->li_subnet_len = (pii->pii_af == AF_INET) ?
IP_ABITS : IPV6_ABITS;
} else {
li->li_subnet = test_subnet;
li->li_subnet_len = test_subnet_len;
}
if (debug & D_LOGINT)
logint_print(li);
return;
error:
logerr("logint_init_from_k: IGNORED %s %s %s addr %s\n",
AF_STR(pii->pii_af), pii->pii_name, li->li_name,
pr_addr(pii->pii_af, testaddr, abuf, sizeof (abuf)));
logint_delete(li);
}
void
logint_delete(struct logint *li)
{
struct phyint_instance *pii;
pii = li->li_phyint_inst;
assert(pii != NULL);
if (debug & D_LOGINT) {
int af;
char abuf[INET6_ADDRSTRLEN];
af = pii->pii_af;
logdebug("logint_delete(%s %s %s/%u)\n",
AF_STR(af), li->li_name,
pr_addr(af, li->li_addr, abuf, sizeof (abuf)),
li->li_subnet_len);
}
assert(pii->pii_logint == li || li->li_prev != NULL);
if (li->li_prev == NULL) {
pii->pii_logint = li->li_next;
} else {
li->li_prev->li_next = li->li_next;
}
if (li->li_next != NULL)
li->li_next->li_prev = li->li_prev;
li->li_next = NULL;
li->li_prev = NULL;
if (pii->pii_probe_logint == li) {
if (pii->pii_probe_sock != -1)
close_probe_socket(pii, _B_TRUE);
pii->pii_probe_logint = NULL;
}
free(li);
}
static void
logint_print(struct logint *li)
{
char abuf[INET6_ADDRSTRLEN];
int af = li->li_phyint_inst->pii_af;
logdebug("logint: %s %s addr %s/%u", AF_STR(af), li->li_name,
pr_addr(af, li->li_addr, abuf, sizeof (abuf)), li->li_subnet_len);
logdebug("\tFlags: %llx in_use %d\n", li->li_flags, li->li_in_use);
}
char *
pr_addr(int af, struct in6_addr addr, char *abuf, int len)
{
struct in_addr addr_v4;
if (af == AF_INET) {
IN6_V4MAPPED_TO_INADDR(&addr, &addr_v4);
(void) inet_ntop(AF_INET, (void *)&addr_v4, abuf, len);
} else {
(void) inet_ntop(AF_INET6, (void *)&addr, abuf, len);
}
return (abuf);
}
void
addr2storage(int af, const struct in6_addr *addr, struct sockaddr_storage *ssp)
{
struct sockaddr_in *sinp = (struct sockaddr_in *)ssp;
struct sockaddr_in6 *sin6p = (struct sockaddr_in6 *)ssp;
assert(af == AF_INET || af == AF_INET6);
switch (af) {
case AF_INET:
(void) memset(sinp, 0, sizeof (*sinp));
sinp->sin_family = AF_INET;
IN6_V4MAPPED_TO_INADDR(addr, &sinp->sin_addr);
break;
case AF_INET6:
(void) memset(sin6p, 0, sizeof (*sin6p));
sin6p->sin6_family = AF_INET6;
sin6p->sin6_addr = *addr;
break;
}
}
struct target *
target_lookup(struct phyint_instance *pii, struct in6_addr addr)
{
struct target *tg;
if (debug & D_TARGET) {
char abuf[INET6_ADDRSTRLEN];
logdebug("target_lookup(%s %s): addr %s\n",
AF_STR(pii->pii_af), pii->pii_name,
pr_addr(pii->pii_af, addr, abuf, sizeof (abuf)));
}
for (tg = pii->pii_targets; tg != NULL; tg = tg->tg_next) {
if (IN6_ARE_ADDR_EQUAL(&tg->tg_address, &addr))
break;
}
return (tg);
}
struct target *
target_next(struct target *tg)
{
struct phyint_instance *pii = tg->tg_phyint_inst;
struct target *marker = tg;
hrtime_t now;
now = gethrtime();
assert(pii->pii_targets == tg || tg->tg_prev != NULL);
assert(pii->pii_targets != NULL);
do {
tg = tg->tg_next;
if (tg == NULL)
tg = pii->pii_targets;
assert(TG_STATUS_VALID(tg->tg_status));
switch (tg->tg_status) {
case TG_ACTIVE:
return (tg);
case TG_UNUSED:
assert(pii->pii_targets_are_routers);
if (pii->pii_ntargets < MAX_PROBE_TARGETS) {
tg->tg_status = TG_ACTIVE;
pii->pii_ntargets++;
return (tg);
}
break;
case TG_SLOW:
assert(pii->pii_targets_are_routers);
if (tg->tg_latime + MIN_RECOVERY_TIME < now) {
tg->tg_status = TG_UNUSED;
}
break;
case TG_DEAD:
assert(pii->pii_targets_are_routers);
if (tg->tg_latime + MIN_RECOVERY_TIME < now) {
tg->tg_status = TG_SLOW;
tg->tg_latime = now;
}
break;
}
} while (tg != marker);
return (NULL);
}
static struct target *
target_select_best(struct phyint_instance *pii)
{
struct target *tg;
struct target *slow = NULL;
struct target *dead = NULL;
struct target *slow_recovered = NULL;
struct target *dead_recovered = NULL;
hrtime_t now;
now = gethrtime();
for (tg = pii->pii_targets; tg != NULL; tg = tg->tg_next) {
assert(TG_STATUS_VALID(tg->tg_status));
switch (tg->tg_status) {
case TG_UNUSED:
return (tg);
case TG_SLOW:
if (tg->tg_latime + MIN_RECOVERY_TIME < now) {
slow_recovered = tg;
tg->tg_status = TG_UNUSED;
} else {
slow = tg;
}
break;
case TG_DEAD:
if (tg->tg_latime + MIN_RECOVERY_TIME < now) {
dead_recovered = tg;
tg->tg_status = TG_SLOW;
tg->tg_latime = now;
} else {
dead = tg;
}
break;
default:
break;
}
}
if (slow_recovered != NULL)
return (slow_recovered);
else if (slow != NULL)
return (slow);
else if (dead_recovered != NULL)
return (dead_recovered);
else
return (dead);
}
static void
target_activate_all(struct phyint_instance *pii)
{
struct target *tg;
assert(pii->pii_ntargets == 0);
assert(pii->pii_target_next == NULL);
assert(pii->pii_rtt_target_next == NULL);
assert(pii->pii_targets_are_routers);
while (pii->pii_ntargets < MIN_PROBE_TARGETS) {
tg = target_select_best(pii);
if (tg == NULL) {
return;
}
assert(TG_STATUS_VALID(tg->tg_status));
assert(tg->tg_status != TG_ACTIVE);
tg->tg_status = TG_ACTIVE;
pii->pii_ntargets++;
if (pii->pii_target_next == NULL) {
pii->pii_target_next = tg;
pii->pii_rtt_target_next = tg;
}
}
}
static struct target *
target_first(struct phyint_instance *pii)
{
struct target *tg;
for (tg = pii->pii_targets; tg != NULL; tg = tg->tg_next) {
assert(TG_STATUS_VALID(tg->tg_status));
if (tg->tg_status == TG_ACTIVE)
break;
}
return (tg);
}
void
target_create(struct phyint_instance *pii, struct in6_addr addr,
boolean_t is_router)
{
struct target *tg;
struct phyint *pi;
struct logint *li;
if (debug & D_TARGET) {
char abuf[INET6_ADDRSTRLEN];
logdebug("target_create(%s %s, %s)\n",
AF_STR(pii->pii_af), pii->pii_name,
pr_addr(pii->pii_af, addr, abuf, sizeof (abuf)));
}
li = pii->pii_probe_logint;
if (li == NULL)
return;
if (!prefix_equal(li->li_subnet, addr, li->li_subnet_len))
return;
if (pii->pii_targets != NULL) {
assert(pii->pii_ntargets <= MAX_PROBE_TARGETS);
if (is_router) {
if (!pii->pii_targets_are_routers) {
while (pii->pii_targets != NULL)
target_delete(pii->pii_targets);
}
} else {
if (pii->pii_targets_are_routers ||
pii->pii_ntargets == MAX_PROBE_TARGETS)
return;
}
}
tg = calloc(1, sizeof (struct target));
if (tg == NULL) {
logperror("target_create: calloc");
return;
}
tg->tg_phyint_inst = pii;
tg->tg_address = addr;
tg->tg_in_use = 1;
tg->tg_rtt_sa = -1;
tg->tg_num_deferred = 0;
if (pii->pii_targets == NULL) {
assert(pii->pii_ntargets == 0);
assert(pii->pii_target_next == NULL);
assert(pii->pii_rtt_target_next == NULL);
pii->pii_targets_are_routers = is_router ? 1 : 0;
}
if (pii->pii_ntargets == MAX_PROBE_TARGETS) {
assert(pii->pii_targets_are_routers);
assert(pii->pii_target_next != NULL);
assert(pii->pii_rtt_target_next != NULL);
tg->tg_status = TG_UNUSED;
} else {
if (pii->pii_ntargets == 0) {
assert(pii->pii_target_next == NULL);
pii->pii_target_next = tg;
pii->pii_rtt_target_next = tg;
}
pii->pii_ntargets++;
tg->tg_status = TG_ACTIVE;
}
target_insert(pii, tg);
pi = pii->pii_phyint;
if (pi->pi_state == PI_NOTARGETS && PROBE_CAPABLE(pii)) {
if (pi->pi_flags & IFF_FAILED)
phyint_chstate(pi, PI_FAILED);
else
phyint_chstate(pi, PI_RUNNING);
}
}
void
target_add(struct phyint_instance *pii, struct in6_addr addr,
boolean_t is_router)
{
struct target *tg;
if (pii == NULL)
return;
tg = target_lookup(pii, addr);
if (tg == NULL || (is_router && !pii->pii_targets_are_routers))
target_create(pii, addr, is_router);
else if (is_router)
tg->tg_in_use = 1;
}
static void
target_insert(struct phyint_instance *pii, struct target *tg)
{
tg->tg_next = pii->pii_targets;
tg->tg_prev = NULL;
if (tg->tg_next != NULL)
tg->tg_next->tg_prev = tg;
pii->pii_targets = tg;
}
void
target_delete(struct target *tg)
{
int af;
struct phyint_instance *pii;
struct phyint_instance *pii_other;
pii = tg->tg_phyint_inst;
af = pii->pii_af;
if (debug & D_TARGET) {
char abuf[INET6_ADDRSTRLEN];
logdebug("target_delete(%s %s, %s)\n",
AF_STR(af), pii->pii_name,
pr_addr(af, tg->tg_address, abuf, sizeof (abuf)));
}
assert(pii->pii_targets == tg || tg->tg_prev != NULL);
reset_pii_probes(pii, tg);
if (tg->tg_prev == NULL) {
pii->pii_targets = tg->tg_next;
} else {
tg->tg_prev->tg_next = tg->tg_next;
}
if (tg->tg_next != NULL)
tg->tg_next->tg_prev = tg->tg_prev;
tg->tg_next = NULL;
tg->tg_prev = NULL;
if (tg->tg_status == TG_ACTIVE)
pii->pii_ntargets--;
if (pii->pii_target_next == tg)
pii->pii_target_next = target_first(pii);
if (pii->pii_rtt_target_next == tg)
pii->pii_rtt_target_next = target_first(pii);
free(tg);
if (pii->pii_ntargets != 0) {
assert(pii->pii_target_next != NULL);
assert(pii->pii_rtt_target_next != NULL);
assert(pii->pii_target_next->tg_status == TG_ACTIVE);
assert(pii->pii_rtt_target_next->tg_status == TG_ACTIVE);
return;
}
assert(pii->pii_target_next == NULL);
assert(pii->pii_rtt_target_next == NULL);
if (pii->pii_targets_are_routers) {
target_activate_all(pii);
if (pii->pii_ntargets != 0) {
assert(pii->pii_target_next != NULL);
assert(pii->pii_rtt_target_next != NULL);
assert(pii->pii_target_next->tg_status == TG_ACTIVE);
assert(pii->pii_rtt_target_next->tg_status ==
TG_ACTIVE);
return;
}
}
assert(pii->pii_targets == NULL);
pii->pii_targets_are_routers = _B_FALSE;
clear_pii_probe_stats(pii);
pii_other = phyint_inst_other(pii);
if (!PROBE_CAPABLE(pii_other) && LINK_UP(pii->pii_phyint) &&
pii->pii_phyint->pi_state != PI_OFFLINE)
phyint_chstate(pii->pii_phyint, PI_NOTARGETS);
}
static void
target_flush_hosts(struct phyint_group *pg)
{
struct phyint *pi;
struct phyint_instance *pii;
if (debug & D_TARGET)
logdebug("target_flush_hosts(%s)\n", pg->pg_name);
for (pi = pg->pg_phyint; pi != NULL; pi = pi->pi_pgnext) {
pii = pi->pi_v4;
if (pii != NULL && !pii->pii_targets_are_routers) {
while (pii->pii_targets != NULL)
target_delete(pii->pii_targets);
}
pii = pi->pi_v6;
if (pii != NULL && !pii->pii_targets_are_routers) {
while (pii->pii_targets != NULL)
target_delete(pii->pii_targets);
}
}
}
static void
reset_pii_probes(struct phyint_instance *pii, struct target *tg)
{
int i;
for (i = 0; i < PROBE_STATS_COUNT; i++) {
if (pii->pii_probes[i].pr_target == tg) {
if (pii->pii_probes[i].pr_status == PR_UNACKED) {
probe_chstate(&pii->pii_probes[i], pii,
PR_LOST);
}
pii->pii_probes[i].pr_target = NULL;
}
}
}
void
clear_pii_probe_stats(struct phyint_instance *pii)
{
bzero(pii->pii_probes, sizeof (struct probe_stats) * PROBE_STATS_COUNT);
pii->pii_probe_next = 0;
}
static void
target_print(struct target *tg)
{
char abuf[INET6_ADDRSTRLEN];
char buf[128];
char buf2[128];
int af;
int i;
af = tg->tg_phyint_inst->pii_af;
logdebug("Target on %s %s addr %s\n"
"status %d rtt_sa %lld rtt_sd %lld crtt %d tg_in_use %d\n",
AF_STR(af), tg->tg_phyint_inst->pii_name,
pr_addr(af, tg->tg_address, abuf, sizeof (abuf)),
tg->tg_status, tg->tg_rtt_sa, tg->tg_rtt_sd,
tg->tg_crtt, tg->tg_in_use);
buf[0] = '\0';
for (i = 0; i < tg->tg_num_deferred; i++) {
(void) snprintf(buf2, sizeof (buf2), " %dms",
tg->tg_deferred[i]);
(void) strlcat(buf, buf2, sizeof (buf));
}
logdebug("deferred rtts:%s\n", buf);
}
void
phyint_inst_print_all(void)
{
struct phyint_instance *pii;
for (pii = phyint_instances; pii != NULL; pii = pii->pii_next) {
phyint_inst_print(pii);
}
}
boolean_t
prefix_equal(struct in6_addr p1, struct in6_addr p2, uint_t prefix_len)
{
uchar_t mask;
int j;
if (prefix_len > IPV6_ABITS)
return (_B_FALSE);
for (j = 0; prefix_len > 8; prefix_len -= 8, j++)
if (p1.s6_addr[j] != p2.s6_addr[j])
return (_B_FALSE);
mask = 0xff << (8 - prefix_len);
if ((p1.s6_addr[j] & mask) != (p2.s6_addr[j] & mask))
return (_B_FALSE);
return (_B_TRUE);
}
static int
logint_upcount(struct phyint *pi)
{
struct logint *li;
int count = 0;
if (pi->pi_v4 != NULL) {
for (li = pi->pi_v4->pii_logint; li != NULL; li = li->li_next) {
if (li->li_flags & IFF_UP)
count++;
}
}
if (pi->pi_v6 != NULL) {
for (li = pi->pi_v6->pii_logint; li != NULL; li = li->li_next) {
if (li->li_flags & IFF_UP)
count++;
}
}
return (count);
}
struct phyint_instance *
phyint_inst_other(struct phyint_instance *pii)
{
if (pii->pii_af == AF_INET)
return (pii->pii_phyint->pi_v6);
else
return (pii->pii_phyint->pi_v4);
}
boolean_t
phyint_is_functioning(struct phyint *pi)
{
if (pi->pi_state == PI_RUNNING)
return (_B_TRUE);
return (pi->pi_state == PI_NOTARGETS && !(pi->pi_flags & IFF_FAILED));
}
boolean_t
phyint_is_usable(struct phyint *pi)
{
if (logint_upcount(pi) == 0)
return (_B_FALSE);
return (phyint_is_functioning(pi));
}
static int
post_event(const char *subclass, nvlist_t *nvl)
{
static evchan_t *evchp = NULL;
if (evchp == NULL) {
errno = sysevent_evc_bind(IPMP_EVENT_CHAN, &evchp, EVCH_CREAT);
if (errno != 0) {
logerr("cannot create event channel `%s': %s\n",
IPMP_EVENT_CHAN, strerror(errno));
goto failed;
}
}
errno = nvlist_add_uint32(nvl, IPMP_EVENT_VERSION,
IPMP_EVENT_CUR_VERSION);
if (errno != 0) {
logerr("cannot create `%s' event: %s", subclass,
strerror(errno));
goto failed;
}
errno = sysevent_evc_publish(evchp, EC_IPMP, subclass, "com.sun",
"in.mpathd", nvl, EVCH_NOSLEEP);
if (errno != 0) {
logerr("cannot send `%s' event: %s\n", subclass,
strerror(errno));
goto failed;
}
nvlist_free(nvl);
return (0);
failed:
nvlist_free(nvl);
return (-1);
}
static ipmp_if_state_t
ifstate(struct phyint *pi)
{
switch (pi->pi_state) {
case PI_INIT:
return (IPMP_IF_UNKNOWN);
case PI_NOTARGETS:
if (pi->pi_flags & IFF_FAILED)
return (IPMP_IF_FAILED);
return (IPMP_IF_UNKNOWN);
case PI_OFFLINE:
return (IPMP_IF_OFFLINE);
case PI_FAILED:
return (IPMP_IF_FAILED);
case PI_RUNNING:
return (IPMP_IF_OK);
}
logerr("ifstate: unknown state %d; aborting\n", pi->pi_state);
abort();
}
static ipmp_if_type_t
iftype(struct phyint *pi)
{
if (pi->pi_flags & IFF_STANDBY)
return (IPMP_IF_STANDBY);
else
return (IPMP_IF_NORMAL);
}
static ipmp_if_linkstate_t
iflinkstate(struct phyint *pi)
{
if (!(pi->pi_notes & (DL_NOTE_LINK_UP|DL_NOTE_LINK_DOWN)))
return (IPMP_LINK_UNKNOWN);
return (LINK_DOWN(pi) ? IPMP_LINK_DOWN : IPMP_LINK_UP);
}
static ipmp_if_probestate_t
ifprobestate(struct phyint *pi)
{
if (!PROBE_ENABLED(pi->pi_v4) && !PROBE_ENABLED(pi->pi_v6))
return (IPMP_PROBE_DISABLED);
if (pi->pi_state == PI_FAILED)
return (IPMP_PROBE_FAILED);
if (!PROBE_CAPABLE(pi->pi_v4) && !PROBE_CAPABLE(pi->pi_v6))
return (IPMP_PROBE_UNKNOWN);
return (IPMP_PROBE_OK);
}
static ipmp_if_targmode_t
iftargmode(struct phyint_instance *pii)
{
if (!PROBE_ENABLED(pii))
return (IPMP_TARG_DISABLED);
else if (pii->pii_targets_are_routers)
return (IPMP_TARG_ROUTES);
else
return (IPMP_TARG_MULTICAST);
}
static ipmp_if_flags_t
ifflags(struct phyint *pi)
{
ipmp_if_flags_t flags = 0;
if (logint_upcount(pi) == 0)
flags |= IPMP_IFFLAG_DOWN;
if (pi->pi_flags & IFF_INACTIVE)
flags |= IPMP_IFFLAG_INACTIVE;
if (pi->pi_hwaddrdup)
flags |= IPMP_IFFLAG_HWADDRDUP;
if (phyint_is_functioning(pi) && flags == 0)
flags |= IPMP_IFFLAG_ACTIVE;
return (flags);
}
static struct sockaddr_storage *
iftestaddr(struct phyint_instance *pii, struct sockaddr_storage *ssp)
{
if (PROBE_ENABLED(pii))
addr2storage(pii->pii_af, &pii->pii_probe_logint->li_addr, ssp);
else
addr2storage(AF_INET6, &in6addr_any, ssp);
return (ssp);
}
static ipmp_group_state_t
groupstate(struct phyint_group *pg)
{
switch (pg->pg_state) {
case PG_FAILED:
return (IPMP_GROUP_FAILED);
case PG_DEGRADED:
return (IPMP_GROUP_DEGRADED);
case PG_OK:
return (IPMP_GROUP_OK);
}
logerr("groupstate: unknown state %d; aborting\n", pg->pg_state);
abort();
}
static ipmp_probe_state_t
probestate(struct probe_stats *ps)
{
switch (ps->pr_status) {
case PR_UNUSED:
case PR_LOST:
return (IPMP_PROBE_LOST);
case PR_UNACKED:
return (IPMP_PROBE_SENT);
case PR_ACKED:
return (IPMP_PROBE_ACKED);
}
logerr("probestate: unknown state %d; aborting\n", ps->pr_status);
abort();
}
int
probe_state_event(struct probe_stats *pr, struct phyint_instance *pii)
{
nvlist_t *nvl;
hrtime_t proc_time = 0, recv_time = 0;
struct sockaddr_storage ss;
struct target *tg = pr->pr_target;
int64_t rttavg, rttdev;
errno = nvlist_alloc(&nvl, NV_UNIQUE_NAME, 0);
if (errno != 0) {
logperror("cannot create `interface change' event");
return (-1);
}
errno = nvlist_add_uint32(nvl, IPMP_PROBE_ID, pr->pr_id);
if (errno != 0)
goto failed;
errno = nvlist_add_string(nvl, IPMP_IF_NAME, pii->pii_phyint->pi_name);
if (errno != 0)
goto failed;
errno = nvlist_add_uint32(nvl, IPMP_PROBE_STATE, probestate(pr));
if (errno != 0)
goto failed;
errno = nvlist_add_hrtime(nvl, IPMP_PROBE_START_TIME,
pr->pr_hrtime_start);
if (errno != 0)
goto failed;
errno = nvlist_add_hrtime(nvl, IPMP_PROBE_SENT_TIME,
pr->pr_hrtime_sent);
if (errno != 0)
goto failed;
if (pr->pr_status == PR_ACKED) {
recv_time = pr->pr_hrtime_ackrecv;
proc_time = pr->pr_hrtime_ackproc;
}
errno = nvlist_add_hrtime(nvl, IPMP_PROBE_ACKRECV_TIME, recv_time);
if (errno != 0)
goto failed;
errno = nvlist_add_hrtime(nvl, IPMP_PROBE_ACKPROC_TIME, proc_time);
if (errno != 0)
goto failed;
if (tg != NULL)
addr2storage(pii->pii_af, &tg->tg_address, &ss);
else
addr2storage(pii->pii_af, &in6addr_any, &ss);
errno = nvlist_add_byte_array(nvl, IPMP_PROBE_TARGET, (uchar_t *)&ss,
sizeof (ss));
if (errno != 0)
goto failed;
rttavg = (tg != NULL) ? (tg->tg_rtt_sa / 8) : 0;
errno = nvlist_add_int64(nvl, IPMP_PROBE_TARGET_RTTAVG, rttavg);
if (errno != 0)
goto failed;
rttdev = (tg != NULL) ? (tg->tg_rtt_sd / 4) : 0;
errno = nvlist_add_int64(nvl, IPMP_PROBE_TARGET_RTTDEV, rttdev);
if (errno != 0)
goto failed;
return (post_event(ESC_IPMP_PROBE_STATE, nvl));
failed:
logperror("cannot create `probe state' event");
nvlist_free(nvl);
return (-1);
}
static int
phyint_group_state_event(struct phyint_group *pg)
{
nvlist_t *nvl;
errno = nvlist_alloc(&nvl, NV_UNIQUE_NAME, 0);
if (errno != 0) {
logperror("cannot create `group state change' event");
return (-1);
}
errno = nvlist_add_string(nvl, IPMP_GROUP_NAME, pg->pg_name);
if (errno != 0)
goto failed;
errno = nvlist_add_uint64(nvl, IPMP_GROUP_SIGNATURE, pg->pg_sig);
if (errno != 0)
goto failed;
errno = nvlist_add_uint32(nvl, IPMP_GROUP_STATE, groupstate(pg));
if (errno != 0)
goto failed;
return (post_event(ESC_IPMP_GROUP_STATE, nvl));
failed:
logperror("cannot create `group state change' event");
nvlist_free(nvl);
return (-1);
}
static int
phyint_group_change_event(struct phyint_group *pg, ipmp_group_op_t op)
{
nvlist_t *nvl;
errno = nvlist_alloc(&nvl, NV_UNIQUE_NAME, 0);
if (errno != 0) {
logperror("cannot create `group change' event");
return (-1);
}
errno = nvlist_add_string(nvl, IPMP_GROUP_NAME, pg->pg_name);
if (errno != 0)
goto failed;
errno = nvlist_add_uint64(nvl, IPMP_GROUP_SIGNATURE, pg->pg_sig);
if (errno != 0)
goto failed;
errno = nvlist_add_uint64(nvl, IPMP_GROUPLIST_SIGNATURE,
phyint_grouplistsig);
if (errno != 0)
goto failed;
errno = nvlist_add_uint32(nvl, IPMP_GROUP_OPERATION, op);
if (errno != 0)
goto failed;
return (post_event(ESC_IPMP_GROUP_CHANGE, nvl));
failed:
logperror("cannot create `group change' event");
nvlist_free(nvl);
return (-1);
}
static int
phyint_group_member_event(struct phyint_group *pg, struct phyint *pi,
ipmp_if_op_t op)
{
nvlist_t *nvl;
errno = nvlist_alloc(&nvl, NV_UNIQUE_NAME, 0);
if (errno != 0) {
logperror("cannot create `group member change' event");
return (-1);
}
errno = nvlist_add_string(nvl, IPMP_GROUP_NAME, pg->pg_name);
if (errno != 0)
goto failed;
errno = nvlist_add_uint64(nvl, IPMP_GROUP_SIGNATURE, pg->pg_sig);
if (errno != 0)
goto failed;
errno = nvlist_add_uint32(nvl, IPMP_IF_OPERATION, op);
if (errno != 0)
goto failed;
errno = nvlist_add_string(nvl, IPMP_IF_NAME, pi->pi_name);
if (errno != 0)
goto failed;
errno = nvlist_add_uint32(nvl, IPMP_IF_TYPE, iftype(pi));
if (errno != 0)
goto failed;
errno = nvlist_add_uint32(nvl, IPMP_IF_STATE, ifstate(pi));
if (errno != 0)
goto failed;
return (post_event(ESC_IPMP_GROUP_MEMBER_CHANGE, nvl));
failed:
logperror("cannot create `group member change' event");
nvlist_free(nvl);
return (-1);
}
static int
phyint_state_event(struct phyint_group *pg, struct phyint *pi)
{
nvlist_t *nvl;
errno = nvlist_alloc(&nvl, NV_UNIQUE_NAME, 0);
if (errno != 0) {
logperror("cannot create `interface change' event");
return (-1);
}
errno = nvlist_add_string(nvl, IPMP_GROUP_NAME, pg->pg_name);
if (errno != 0)
goto failed;
errno = nvlist_add_uint64(nvl, IPMP_GROUP_SIGNATURE, pg->pg_sig);
if (errno != 0)
goto failed;
errno = nvlist_add_string(nvl, IPMP_IF_NAME, pi->pi_name);
if (errno != 0)
goto failed;
errno = nvlist_add_uint32(nvl, IPMP_IF_TYPE, iftype(pi));
if (errno != 0)
goto failed;
errno = nvlist_add_uint32(nvl, IPMP_IF_STATE, ifstate(pi));
if (errno != 0)
goto failed;
return (post_event(ESC_IPMP_IF_CHANGE, nvl));
failed:
logperror("cannot create `interface change' event");
nvlist_free(nvl);
return (-1);
}
static uint64_t
gensig(void)
{
static int seeded = 0;
if (seeded == 0) {
srand48((long)gethrtime());
seeded++;
}
return ((uint64_t)lrand48() << 48 | 1);
}
unsigned int
getgroupinfo(const char *grname, ipmp_groupinfo_t **grinfopp)
{
struct phyint *pi;
struct phyint_group *pg;
char (*ifs)[LIFNAMSIZ];
unsigned int i, j;
unsigned int nif = 0, naddr = 0;
lifgroupinfo_t lifgr;
addrlist_t *addrp;
struct sockaddr_storage *addrs;
int fdt = 0;
pg = phyint_group_lookup(grname);
if (pg == NULL)
return (IPMP_EUNKGROUP);
for (pi = pg->pg_phyint; pi != NULL; pi = pi->pi_pgnext)
nif++;
ifs = alloca(nif * sizeof (*ifs));
for (i = 0, pi = pg->pg_phyint; pi != NULL; pi = pi->pi_pgnext, i++) {
assert(i < nif);
(void) strlcpy(ifs[i], pi->pi_name, LIFNAMSIZ);
if (PROBE_ENABLED(pi->pi_v4) || PROBE_ENABLED(pi->pi_v6))
fdt = pg->pg_fdt;
}
assert(i == nif);
if (pg == phyint_anongroup) {
*grinfopp = ipmp_groupinfo_create(pg->pg_name, pg->pg_sig, fdt,
groupstate(pg), nif, ifs, "", "", "", "", 0, NULL);
return (*grinfopp == NULL ? IPMP_ENOMEM : IPMP_SUCCESS);
}
(void) strlcpy(lifgr.gi_grname, grname, LIFGRNAMSIZ);
if (ioctl(ifsock_v4, SIOCGLIFGROUPINFO, &lifgr) == -1) {
if (errno == ENOENT)
return (IPMP_EUNKGROUP);
logperror("getgroupinfo: SIOCGLIFGROUPINFO");
return (IPMP_FAILURE);
}
for (addrp = pg->pg_addrs; addrp != NULL; addrp = addrp->al_next)
naddr++;
addrs = alloca(naddr * sizeof (*addrs));
i = 0;
for (addrp = pg->pg_addrs; addrp != NULL; addrp = addrp->al_next) {
for (j = 0; j < i; j++) {
if (sockaddrcmp(&addrs[j], &addrp->al_addr))
break;
}
if (j == i) {
assert(i < naddr);
addrs[i++] = addrp->al_addr;
}
}
naddr = i;
*grinfopp = ipmp_groupinfo_create(pg->pg_name, pg->pg_sig, fdt,
groupstate(pg), nif, ifs, lifgr.gi_grifname, lifgr.gi_m4ifname,
lifgr.gi_m6ifname, lifgr.gi_bcifname, naddr, addrs);
return (*grinfopp == NULL ? IPMP_ENOMEM : IPMP_SUCCESS);
}
unsigned int
gettarginfo(struct phyint_instance *pii, const char *name,
ipmp_targinfo_t **targinfopp)
{
uint_t ntarg = 0;
struct target *tg;
struct sockaddr_storage ss;
struct sockaddr_storage *targs = NULL;
if (PROBE_CAPABLE(pii)) {
targs = alloca(pii->pii_ntargets * sizeof (*targs));
tg = pii->pii_target_next;
do {
if (tg->tg_status == TG_ACTIVE) {
assert(ntarg < pii->pii_ntargets);
addr2storage(pii->pii_af, &tg->tg_address,
&targs[ntarg++]);
}
if ((tg = tg->tg_next) == NULL)
tg = pii->pii_targets;
} while (tg != pii->pii_target_next);
assert(ntarg == pii->pii_ntargets);
}
*targinfopp = ipmp_targinfo_create(name, iftestaddr(pii, &ss),
iftargmode(pii), ntarg, targs);
return (*targinfopp == NULL ? IPMP_ENOMEM : IPMP_SUCCESS);
}
unsigned int
getifinfo(const char *ifname, ipmp_ifinfo_t **ifinfopp)
{
int retval;
struct phyint *pi;
ipmp_targinfo_t *targinfo4;
ipmp_targinfo_t *targinfo6;
pi = phyint_lookup(ifname);
if (pi == NULL)
return (IPMP_EUNKIF);
if ((retval = gettarginfo(pi->pi_v4, pi->pi_name, &targinfo4)) != 0 ||
(retval = gettarginfo(pi->pi_v6, pi->pi_name, &targinfo6)) != 0)
goto out;
*ifinfopp = ipmp_ifinfo_create(pi->pi_name, pi->pi_group->pg_name,
ifstate(pi), iftype(pi), iflinkstate(pi), ifprobestate(pi),
ifflags(pi), targinfo4, targinfo6);
retval = (*ifinfopp == NULL ? IPMP_ENOMEM : IPMP_SUCCESS);
out:
if (targinfo4 != NULL)
ipmp_freetarginfo(targinfo4);
if (targinfo6 != NULL)
ipmp_freetarginfo(targinfo6);
return (retval);
}
unsigned int
getgrouplist(ipmp_grouplist_t **grlistpp)
{
struct phyint_group *pg;
char (*groups)[LIFGRNAMSIZ];
unsigned int i, ngroup;
for (ngroup = 0, pg = phyint_groups; pg != NULL; pg = pg->pg_next)
ngroup++;
groups = alloca(ngroup * sizeof (*groups));
for (i = 0, pg = phyint_groups; pg != NULL; pg = pg->pg_next, i++) {
assert(i < ngroup);
(void) strlcpy(groups[i], pg->pg_name, LIFGRNAMSIZ);
}
assert(i == ngroup);
*grlistpp = ipmp_grouplist_create(phyint_grouplistsig, ngroup, groups);
return (*grlistpp == NULL ? IPMP_ENOMEM : IPMP_SUCCESS);
}
unsigned int
getgraddrinfo(const char *grname, struct sockaddr_storage *ssp,
ipmp_addrinfo_t **adinfopp)
{
int ifsock;
addrlist_t *addrp, *addrmatchp = NULL;
ipmp_addr_state_t state;
const char *binding = "";
struct lifreq lifr;
struct phyint_group *pg;
if ((pg = phyint_group_lookup(grname)) == NULL)
return (IPMP_EUNKADDR);
for (addrp = pg->pg_addrs; addrp != NULL; addrp = addrp->al_next) {
if (sockaddrcmp(ssp, &addrp->al_addr)) {
addrmatchp = addrp;
if (addrmatchp->al_flags & IFF_UP)
break;
}
}
if (addrmatchp == NULL)
return (IPMP_EUNKADDR);
state = (addrmatchp->al_flags & IFF_UP) ? IPMP_ADDR_UP : IPMP_ADDR_DOWN;
if (state == IPMP_ADDR_UP) {
ifsock = (ssp->ss_family == AF_INET) ? ifsock_v4 : ifsock_v6;
(void) strlcpy(lifr.lifr_name, addrmatchp->al_name, LIFNAMSIZ);
if (ioctl(ifsock, SIOCGLIFBINDING, &lifr) >= 0)
binding = lifr.lifr_binding;
}
*adinfopp = ipmp_addrinfo_create(ssp, state, pg->pg_name, binding);
return (*adinfopp == NULL ? IPMP_ENOMEM : IPMP_SUCCESS);
}
unsigned int
getsnap(ipmp_snap_t **snapp)
{
ipmp_grouplist_t *grlistp;
ipmp_groupinfo_t *grinfop;
ipmp_addrinfo_t *adinfop;
ipmp_addrlist_t *adlistp;
ipmp_ifinfo_t *ifinfop;
ipmp_snap_t *snap;
struct phyint *pi;
unsigned int i, j;
int retval;
snap = ipmp_snap_create();
if (snap == NULL)
return (IPMP_ENOMEM);
retval = getgrouplist(&snap->sn_grlistp);
if (retval != IPMP_SUCCESS)
goto failed;
grlistp = snap->sn_grlistp;
for (i = 0; i < grlistp->gl_ngroup; i++) {
retval = getgroupinfo(grlistp->gl_groups[i], &grinfop);
if (retval != IPMP_SUCCESS)
goto failed;
retval = ipmp_snap_addgroupinfo(snap, grinfop);
if (retval != IPMP_SUCCESS) {
ipmp_freegroupinfo(grinfop);
goto failed;
}
adlistp = grinfop->gr_adlistp;
for (j = 0; j < adlistp->al_naddr; j++) {
retval = getgraddrinfo(grinfop->gr_name,
&adlistp->al_addrs[j], &adinfop);
if (retval != IPMP_SUCCESS)
goto failed;
retval = ipmp_snap_addaddrinfo(snap, adinfop);
if (retval != IPMP_SUCCESS) {
ipmp_freeaddrinfo(adinfop);
goto failed;
}
}
}
for (pi = phyints; pi != NULL; pi = pi->pi_next) {
retval = getifinfo(pi->pi_name, &ifinfop);
if (retval != IPMP_SUCCESS)
goto failed;
retval = ipmp_snap_addifinfo(snap, ifinfop);
if (retval != IPMP_SUCCESS) {
ipmp_freeifinfo(ifinfop);
goto failed;
}
}
*snapp = snap;
return (IPMP_SUCCESS);
failed:
ipmp_snap_free(snap);
return (retval);
}