Viewing: nidstrings.c
// SPDX-License-Identifier: GPL-2.0
/* Copyright (c) 2008, 2010, Oracle and/or its affiliates. All rights reserved.
* Use is subject to license terms.
*
* Copyright (c) 2011, 2015, Intel Corporation.
*/
/* This file is part of Lustre, http://www.lustre.org/
*
* Author: Phil Schwan <phil@clusterfs.com>
*/
#define DEBUG_SUBSYSTEM S_LNET
#include <linux/sunrpc/addr.h>
#include <linux/libcfs/libcfs.h>
#include <uapi/linux/lnet/nidstr.h>
#include <linux/lnet/lib-types.h>
#include <linux/ctype.h>
#include <linux/inet.h>
#include <linux/inetdevice.h>
/* max value for numeric network address */
#define MAX_NUMERIC_VALUE 0xffffffff
#define IPSTRING_LENGTH 16
/* CAVEAT VENDITOR! Keep the canonical string representation of nets/nids
* consistent in all conversion functions. Some code fragments are copied
* around for the sake of clarity...
*/
/* CAVEAT EMPTOR! Racey temporary buffer allocation!
* Choose the number of nidstrings to support the MAXIMUM expected number of
* concurrent users. If there are more, the returned string will be volatile.
* NB this number must allow for a process to be descheduled for a timeslice
* between getting its string and using it.
*/
static char libcfs_nidstrings[LNET_NIDSTR_COUNT][LNET_NIDSTR_SIZE];
static int libcfs_nidstring_idx;
static DEFINE_SPINLOCK(libcfs_nidstring_lock);
static struct netstrfns *libcfs_namenum2netstrfns(const char *name);
char *
libcfs_next_nidstring(void)
{
char *str;
unsigned long flags;
spin_lock_irqsave(&libcfs_nidstring_lock, flags);
str = libcfs_nidstrings[libcfs_nidstring_idx++];
if (libcfs_nidstring_idx == ARRAY_SIZE(libcfs_nidstrings))
libcfs_nidstring_idx = 0;
spin_unlock_irqrestore(&libcfs_nidstring_lock, flags);
return str;
}
EXPORT_SYMBOL(libcfs_next_nidstring);
/* NID range list syntax.
* \verbatim
*
* <nidlist> :== <nidrange> [ ' ' <nidrange> ]
* <nidrange> :== <addrrange> '@' <net>
* <addrrange> :== '*' |
* <netmask> |
* <ipv6_addr> |
* <ipv4_addr_range> |
* <numaddr_range>
* <netmask> :== An IPv4 or IPv6 network mask in CIDR notation.
* e.g. 192.168.1.0/24 or 2001:0db8::/32
* <ipv6_addr> :== A single IPv6 address
* <ipv4_addr_range> :==
* <numaddr_range>.<numaddr_range>.<numaddr_range>.<numaddr_range>
* <numaddr_range> :== <number> |
* <expr_list>
* <expr_list> :== '[' <range_expr> [ ',' <range_expr>] ']'
* <range_expr> :== <number> |
* <number> '-' <number> |
* <number> '-' <number> '/' <number>
* <net> :== <netname> | <netname><number>
* <netname> :== "lo" | "tcp" | "o2ib" | "gni" | "gip" | "ptlf" | "kfi"
* \endverbatim
*/
/*
* Structure to represent \<nidrange\> token of the syntax.
*
* One of this is created for each \<net\> parsed.
*/
struct nidrange {
/**
* Link to list of this structures which is built on nid range
* list parsing.
*/
struct list_head nr_link;
/* List head for addrrange::ar_link. */
struct list_head nr_addrranges;
/* List head for nidmask::nm_link. */
struct list_head nr_nidmasks;
/* Flag indicating that *@<net> is found. */
int nr_all;
/* Pointer to corresponding element of libcfs_netstrfns. */
struct netstrfns *nr_netstrfns;
/* Number of network. E.g. 5 if \<net\> is "elan5". */
int nr_netnum;
};
struct nidmask {
/* Link to nidrange::nr_nidmasks */
struct list_head nm_link;
/* This is the base address that was parsed */
union {
struct in_addr ipv4;
struct in6_addr ipv6;
} nm_addr;
/* Netmask derived from the prefix length */
union {
struct in_addr ipv4;
struct in6_addr ipv6;
} nm_netmask;
/* Network address derived from the base address and the netmask */
union {
struct in_addr ipv4;
struct in6_addr ipv6;
} nm_netaddr;
/* Address family */
sa_family_t nm_family;
/* Prefix length */
u8 nm_prefix_len;
};
/*
* Structure to represent \<addrrange\> token of the syntax.
*/
struct addrrange {
/**
* Link to nidrange::nr_addrranges.
*/
struct list_head ar_link;
/**
* List head for cfs_expr_list::el_list.
*/
struct list_head ar_numaddr_ranges;
};
/**
* cfs_range_expr_parse() - Parses \<range_expr\> token of the syntax.
* @src: contains range expression to be parsed
* @min: Min value
* @max: Max value
* @bracketed: If %False @src must have only single number/range
* @expr: After successful parsing this will be populated [out]
*
* Parses \<range_expr\> token of the syntax. If @bracketed is false,
* @src should only have a single token which can be \<number\> or \*
*
* \retval pointer to allocated range_expr and initialized
* range_expr::re_lo, range_expr::re_hi & range_expr:re_stride if @src parses to
* \<number\> |
* \<number\> '-' \<number\> |
* \<number\> '-' \<number\> '/' \<number\>
*
* Returns %0 will be returned if it can be parsed, otherwise %-EINVAL or
* %-ENOMEM will be returned.
*/
static int
cfs_range_expr_parse(char *src, unsigned int min, unsigned int max,
int bracketed, struct cfs_range_expr **expr)
{
struct cfs_range_expr *re;
char *tok;
unsigned int num;
LIBCFS_ALLOC(re, sizeof(*re));
if (!re)
return -ENOMEM;
src = strim(src);
if (strcmp(src, "*") == 0) {
re->re_lo = min;
re->re_hi = max;
re->re_stride = 1;
goto out;
}
if (kstrtouint(src, 0, &num) == 0) {
if (num < min || num > max)
goto failed;
/* <number> is parsed */
re->re_lo = num;
re->re_hi = re->re_lo;
re->re_stride = 1;
goto out;
}
if (!bracketed)
goto failed;
tok = strim(strsep(&src, "-"));
if (!src)
goto failed;
if (kstrtouint(tok, 0, &num) != 0 ||
num < min || num > max)
goto failed;
re->re_lo = num;
/* <number> - */
if (kstrtouint(strim(src), 0, &num) == 0) {
if (num < min || num > max)
goto failed;
re->re_hi = num;
/* <number> - <number> is parsed */
re->re_stride = 1;
goto out;
}
/* go to check <number> '-' <number> '/' <number> */
tok = strim(strsep(&src, "/"));
if (!src)
goto failed;
if (kstrtouint(tok, 0, &num) != 0 ||
num < min || num > max)
goto failed;
re->re_hi = num;
if (kstrtouint(strim(src), 0, &num) != 0 ||
num < min || num > max)
goto failed;
re->re_stride = num;
out:
if (re->re_lo > re->re_hi)
goto failed;
*expr = re;
return 0;
failed:
LIBCFS_FREE(re, sizeof(*re));
return -EINVAL;
}
/**
* cfs_expr_list_match() - Matches value (@value) against ranges expression
* list @expr_list.
* @value: Value to be checked against the range
* @expr_list: list of (struct cfs_range_expr) it defines range
*
* Return:
* * %1 if @value matches
* * %0 otherwise
*/
int
cfs_expr_list_match(u32 value, struct cfs_expr_list *expr_list)
{
struct cfs_range_expr *expr;
list_for_each_entry(expr, &expr_list->el_exprs, re_link) {
if (value >= expr->re_lo && value <= expr->re_hi &&
((value - expr->re_lo) % expr->re_stride) == 0)
return 1;
}
return 0;
}
/**
* cfs_expr_list_values() - Convert express list (@expr_list) to an array of
* all matched values
* @expr_list: list of (struct cfs_range_expr) it defines range
* @max: Max array size
* @valpp: Array of all matched values [out]
*
* Return:
* * %N N is total number of all matched values
* * %0 if expression list is empty
* * %negative for failure
*/
int
cfs_expr_list_values(struct cfs_expr_list *expr_list, int max, u32 **valpp)
{
struct cfs_range_expr *expr;
int count = 0;
u32 *val;
int i;
list_for_each_entry(expr, &expr_list->el_exprs, re_link) {
for (i = expr->re_lo; i <= expr->re_hi; i++) {
if (((i - expr->re_lo) % expr->re_stride) == 0)
count++;
}
}
if (count == 0) /* empty expression list */
return 0;
if (count > max) {
CERROR("Number of values %d exceeds max allowed %d\n",
max, count);
return -EINVAL;
}
CFS_ALLOC_PTR_ARRAY(val, count);
if (!val)
return -ENOMEM;
count = 0;
list_for_each_entry(expr, &expr_list->el_exprs, re_link) {
for (i = expr->re_lo; i <= expr->re_hi; i++) {
if (((i - expr->re_lo) % expr->re_stride) == 0)
val[count++] = i;
}
}
*valpp = val;
return count;
}
EXPORT_SYMBOL(cfs_expr_list_values);
/**
* cfs_expr_list_free() - Frees cfs_range_expr structures of @expr_list.
* @expr_list: Struct cfs_range_expr to free
*/
void
cfs_expr_list_free(struct cfs_expr_list *expr_list)
{
while (!list_empty(&expr_list->el_exprs)) {
struct cfs_range_expr *expr;
expr = list_entry(expr_list->el_exprs.next,
struct cfs_range_expr, re_link);
list_del(&expr->re_link);
LIBCFS_FREE(expr, sizeof(*expr));
}
LIBCFS_FREE(expr_list, sizeof(*expr_list));
}
EXPORT_SYMBOL(cfs_expr_list_free);
/**
* cfs_expr_list_parse() - Parses \<cfs_expr_list\> token of the syntax.
* @str: Expression to be parsed
* @len: length of @str
* @min: Min value of parsed range
* @max: Max value of parsed range
* @elpp: Parsed range expression [out]
*
* Return:
* * %0 if @str parses to \<number\> | \<expr_list\>
* * %-errno otherwise
*/
int
cfs_expr_list_parse(char *str, int len, unsigned int min, unsigned int max,
struct cfs_expr_list **elpp)
{
struct cfs_expr_list *expr_list;
struct cfs_range_expr *expr;
char *src;
int rc;
CFS_ALLOC_PTR(expr_list);
if (!expr_list)
return -ENOMEM;
str = kstrndup(str, len, GFP_KERNEL);
if (!str) {
CFS_FREE_PTR(expr_list);
return -ENOMEM;
}
src = str;
INIT_LIST_HEAD(&expr_list->el_exprs);
if (src[0] == '[' &&
src[strlen(src) - 1] == ']') {
src++;
src[strlen(src)-1] = '\0';
rc = -EINVAL;
while (src) {
char *tok = strsep(&src, ",");
if (!*tok) {
rc = -EINVAL;
break;
}
tok = strim(tok);
rc = cfs_range_expr_parse(tok, min, max, 1, &expr);
if (rc != 0)
break;
list_add_tail(&expr->re_link, &expr_list->el_exprs);
}
} else {
rc = cfs_range_expr_parse(src, min, max, 0, &expr);
if (rc == 0)
list_add_tail(&expr->re_link, &expr_list->el_exprs);
}
kfree(str);
if (rc != 0)
cfs_expr_list_free(expr_list);
else
*elpp = expr_list;
return rc;
}
EXPORT_SYMBOL(cfs_expr_list_parse);
/**
* cfs_expr_list_free_list() - Frees cfs_expr_list structures of @list.
* @list: List to free (struct cfs_expr_list)
*
* For each struct cfs_expr_list structure found on @list it frees
* range_expr list attached to it and frees the cfs_expr_list itself.
*/
void
cfs_expr_list_free_list(struct list_head *list)
{
struct cfs_expr_list *el;
while (!list_empty(list)) {
el = list_first_entry(list,
struct cfs_expr_list, el_link);
list_del(&el->el_link);
cfs_expr_list_free(el);
}
}
/**
* parse_addrange() - Parses \<addrrange\> token on the syntax.
* @str: address range
* @nidrange: pointer to the struct nidrange
*
* Allocates struct addrrange and links to @nidrange via
* (nidrange::nr_addrranges)
*
* Return:
* * %0 if @src parses to '*' | \<ipaddr_range\> | \<cfs_expr_list\>
* * %-errno otherwise
*/
static int
parse_addrange(char *str, struct nidrange *nidrange)
{
struct addrrange *addrrange;
if (strcmp(str, "*") == 0) {
nidrange->nr_all = 1;
return 0;
}
CFS_ALLOC_PTR(addrrange);
if (addrrange == NULL)
return -ENOMEM;
list_add_tail(&addrrange->ar_link, &nidrange->nr_addrranges);
INIT_LIST_HEAD(&addrrange->ar_numaddr_ranges);
return nidrange->nr_netstrfns->nf_parse_addrlist(str, strlen(str),
&addrrange->ar_numaddr_ranges);
}
static void
init_ipv4_nidmask(struct sockaddr_in *sa, struct nidmask *nm)
{
memcpy(&nm->nm_addr.ipv4, &sa->sin_addr.s_addr, sizeof(struct in_addr));
nm->nm_netmask.ipv4.s_addr = inet_make_mask(nm->nm_prefix_len);
nm->nm_netaddr.ipv4.s_addr = sa->sin_addr.s_addr &
nm->nm_netmask.ipv4.s_addr;
}
/* Note: The memory of struct nidmask is allocated via CFS_ALLOC_PTR, so it has
* been set to zero as required by this function.
*/
static void
init_ipv6_nidmask(struct sockaddr_in6 *sa, struct nidmask *nm)
{
int i, j;
memcpy(&nm->nm_addr.ipv6, &sa->sin6_addr.s6_addr,
sizeof(struct in6_addr));
for (i = nm->nm_prefix_len, j = 0; i > 0; i -= 8, j++) {
if (i >= 8)
nm->nm_netmask.ipv6.s6_addr[j] = 0xff;
else
nm->nm_netmask.ipv6.s6_addr[j] =
(unsigned long)(0xffU << (8 - i));
}
for (i = 0; i < sizeof(struct in6_addr); i++)
nm->nm_netaddr.ipv6.s6_addr[i] = sa->sin6_addr.s6_addr[i] &
nm->nm_netmask.ipv6.s6_addr[i];
}
static u8
parse_prefix_len(char *str)
{
unsigned int max;
unsigned int prefix_len;
char *slash = strchr(str, '/');
/* IPv4 netmask must include an explicit prefix length */
if (!(slash || strchr(str, ':')))
return 0;
/* We treat an IPv6 address without a prefix length as having /128 */
if (!slash)
return 128;
if (strchr(str, ':'))
max = 128;
else
max = 32;
if (kstrtouint(slash + 1, 10, &prefix_len))
return 0;
if (prefix_len < 1 || prefix_len > max)
return 0;
return (u8)prefix_len;
}
static int
parse_nidmask(char *str, struct nidrange *nr)
{
struct nidmask *nm;
struct sockaddr_storage sa;
char *addrstr;
CFS_ALLOC_PTR(nm);
if (!nm)
return -ENOMEM;
/* Add to nr_nidmasks so that our caller can free us on error */
list_add_tail(&nm->nm_link, &nr->nr_nidmasks);
nm->nm_prefix_len = parse_prefix_len(str);
if (!nm->nm_prefix_len)
return -EINVAL;
addrstr = strsep(&str, "/");
if (rpc_pton(&init_net, addrstr, strlen(addrstr),
(struct sockaddr *)&sa, sizeof(sa)) == 0)
return -EINVAL;
nm->nm_family = sa.ss_family;
if (nm->nm_family == AF_INET6)
init_ipv6_nidmask((struct sockaddr_in6 *)&sa, nm);
else if (nm->nm_family == AF_INET)
init_ipv4_nidmask((struct sockaddr_in *)&sa, nm);
else
return -EINVAL;
return 0;
}
/**
* add_nidrange() - Finds or creates struct nidrange.
* @str: Input string
* @nidlist: list of nidranges
*
* Checks if @str is a valid network name, looks for corresponding
* nidrange on the list of nidranges (@nidlist), creates new struct
* nidrange if it is not found.
*
* Return:
* * %pointer to struct nidrange matching network specified via @str
* * %NULL if @str does not match any network
*/
static struct nidrange *
add_nidrange(char *str, struct list_head *nidlist)
{
struct netstrfns *nf;
struct nidrange *nr;
char *end;
unsigned int netnum;
nf = libcfs_namenum2netstrfns(str);
if (nf == NULL)
return NULL;
end = str + strlen(nf->nf_name);
if (!*end) {
/* network name only, e.g. "elan" or "tcp" */
netnum = 0;
} else {
/* e.g. "elan25" or "tcp23", refuse to parse if
* network name is not appended with decimal or
* hexadecimal number
*/
if (kstrtouint(end, 0, &netnum) != 0)
return NULL;
}
list_for_each_entry(nr, nidlist, nr_link) {
if (nr->nr_netstrfns != nf)
continue;
if (nr->nr_netnum != netnum)
continue;
return nr;
}
CFS_ALLOC_PTR(nr);
if (nr == NULL)
return NULL;
list_add_tail(&nr->nr_link, nidlist);
INIT_LIST_HEAD(&nr->nr_addrranges);
INIT_LIST_HEAD(&nr->nr_nidmasks);
nr->nr_netstrfns = nf;
nr->nr_all = 0;
nr->nr_netnum = netnum;
return nr;
}
/**
* parse_nidrange() - Parses \<nidrange\> token of the syntax.
* @str: Input string
* @nidlist: list of nidranges
*
* Return:
* * %0 if @src parses to \<addrrange\> '@' \<net\>
* * %-EINVAL otherwise
*/
static int
parse_nidrange(char *str, struct list_head *nidlist)
{
char *addrrange;
char *net;
char *slash;
struct nidrange *nr;
int rc;
addrrange = strim(strsep(&str, "@"));
if (!str)
return -EINVAL;
net = strim(str);
if (strchr(net, '@') || !*net)
return -EINVAL;
nr = add_nidrange(net, nidlist);
if (!nr)
return -EINVAL;
/* Check for a '/' that does not appear inside '[]' */
slash = strchr(addrrange, '/');
if (strchr(addrrange, ':') || (slash && !strchr(slash, ']')))
rc = parse_nidmask(addrrange, nr);
else
rc = parse_addrange(addrrange, nr);
return rc;
}
/**
* free_addrranges() - Frees addrrange structures of @list.
* @list: List to free (struct addrrange)
*
* For each struct addrrange structure found on @list it frees
* cfs_expr_list list attached to it and frees the addrrange itself.
*/
static void
free_addrranges(struct list_head *list)
{
struct addrrange *ar;
while ((ar = list_first_entry_or_null(list,
struct addrrange,
ar_link)) != NULL) {
cfs_expr_list_free_list(&ar->ar_numaddr_ranges);
list_del(&ar->ar_link);
CFS_FREE_PTR(ar);
}
}
static void
free_nidmasks(struct list_head *list)
{
struct nidmask *nm;
while ((nm = list_first_entry_or_null(list,
struct nidmask,
nm_link)) != NULL) {
list_del(&nm->nm_link);
CFS_FREE_PTR(nm);
}
}
/**
* cfs_free_nidlist() - Frees nidrange strutures of @list.
* @list: List fo free (struct nidrange)
*
* For each struct nidrange structure found on @list it frees
* addrrange list attached to it and frees the nidrange itself.
*/
void
cfs_free_nidlist(struct list_head *list)
{
struct list_head *pos, *next;
struct nidrange *nr;
list_for_each_safe(pos, next, list) {
nr = list_entry(pos, struct nidrange, nr_link);
free_addrranges(&nr->nr_addrranges);
free_nidmasks(&nr->nr_nidmasks);
list_del(pos);
CFS_FREE_PTR(nr);
}
}
EXPORT_SYMBOL(cfs_free_nidlist);
/**
* cfs_parse_nidlist() - Parses nid range list.
* @orig: Original string (NID)
* @len: length of @orig
* @nidlist: Populate after success parse (list of struct nidrange) [out]
*
* Parses with rigorous syntax and overflow checking @orig into
* \<nidrange\> [ ' ' \<nidrange\> ], compiles @orig into set of
* structures and links that structure to @nidlist. The resulting
* list can be used to match a NID againts set of NIDS defined by @orig.
* \see cfs_match_nid
*
* Return:
* * %0 on success
* * %-errno otherwise (-ENOMEM or -EINVAL)
*/
int
cfs_parse_nidlist(char *orig, int len, struct list_head *nidlist)
{
int rc = 0;
char *str;
orig = kstrndup(orig, len, GFP_KERNEL);
if (!orig)
return -ENOMEM;
INIT_LIST_HEAD(nidlist);
str = orig;
while (rc == 0 && str) {
char *tok = strsep(&str, " ");
if (*tok)
rc = parse_nidrange(tok, nidlist);
}
kfree(orig);
if (rc)
cfs_free_nidlist(nidlist);
else if (list_empty(nidlist))
rc = -EINVAL;
return rc;
}
EXPORT_SYMBOL(cfs_parse_nidlist);
static int
match_nidmask(const struct lnet_nid *nid, struct nidmask *nm,
struct netstrfns *nf)
{
__be32 addr[4] = { nid->nid_addr[0], nid->nid_addr[1],
nid->nid_addr[2], nid->nid_addr[3] };
__be32 *netmask, *netaddr;
if (!nf->nf_match_netmask)
return 0;
if (nid_is_nid4(nid) && nm->nm_family == AF_INET) {
netmask = (__be32 *)&nm->nm_netmask.ipv4.s_addr;
netaddr = (__be32 *)&nm->nm_netaddr.ipv4.s_addr;
} else if (!nid_is_nid4(nid) && nm->nm_family == AF_INET6) {
netmask = (__be32 *)&nm->nm_netmask.ipv6.s6_addr;
netaddr = (__be32 *)&nm->nm_netaddr.ipv6.s6_addr;
} else {
return 0;
}
return nf->nf_match_netmask(addr, NID_ADDR_BYTES(nid), netmask,
netaddr);
}
/**
* cfs_match_nid() - Matches a nid (@nid) against the compiled list of
* nidranges (@nidlist).
* @nid: NID to match
* @nidlist: Compiled list of nidranges
*
* see cfs_parse_nidlist()
*
* Return:
* * %1 on match
* * %0 otherwises
*/
int cfs_match_nid(const struct lnet_nid *nid, struct list_head *nidlist)
{
struct nidrange *nr;
struct nidmask *nm;
struct addrrange *ar;
list_for_each_entry(nr, nidlist, nr_link) {
if (nr->nr_netstrfns->nf_type != nid->nid_type)
continue;
if (nr->nr_netnum != be16_to_cpu(nid->nid_num))
continue;
if (nr->nr_all)
return 1;
list_for_each_entry(nm, &nr->nr_nidmasks, nm_link)
if (match_nidmask(nid, nm, nr->nr_netstrfns))
return 1;
list_for_each_entry(ar, &nr->nr_addrranges, ar_link)
if (nr->nr_netstrfns->nf_match_addr(
be32_to_cpu(nid->nid_addr[0]),
&ar->ar_numaddr_ranges))
return 1;
}
return 0;
}
EXPORT_SYMBOL(cfs_match_nid);
/**
* cfs_print_network() - Print the network part of the nidrange @nr into the
* specified @buffer.
* @buffer: Buffer holding network part of nidrange [out]
* @count: Max lenght to print
* @nr: nidrange
*
* Return number of characters written
*/
static int
cfs_print_network(char *buffer, int count, struct nidrange *nr)
{
struct netstrfns *nf = nr->nr_netstrfns;
if (nr->nr_netnum == 0)
return scnprintf(buffer, count, "@%s", nf->nf_name);
else
return scnprintf(buffer, count, "@%s%u",
nf->nf_name, nr->nr_netnum);
}
/**
* cfs_print_addrranges() - Print a list of addrrange (@addrranges) into the
* specified @buffer.
* @buffer: Buffer holding the addrrange
* @count: Max length to print
* @addrranges: list of addrrange
* @nr: nidrange
*
* Print a list of addrrange (@addrranges) into the specified @buffer.
* At max @count characters can be printed into @buffer.
*
* Return number of characters written
*/
static int
cfs_print_addrranges(char *buffer, int count, struct list_head *addrranges,
struct nidrange *nr)
{
int i = 0;
struct addrrange *ar;
struct netstrfns *nf = nr->nr_netstrfns;
list_for_each_entry(ar, addrranges, ar_link) {
if (i != 0)
i += scnprintf(buffer + i, count - i, " ");
i += nf->nf_print_addrlist(buffer + i, count - i,
&ar->ar_numaddr_ranges);
i += cfs_print_network(buffer + i, count - i, nr);
}
return i;
}
static int
print_nidmask(char *buffer, int count, struct nidmask *nm, struct nidrange *nr,
bool omit_prefix)
{
int i = 0;
struct sockaddr_storage sa = {};
u8 max;
/* parse_nidmask() ensures nm_family is set to either AF_INET
* or AF_INET6
*/
sa.ss_family = nm->nm_family;
if (nm->nm_family == AF_INET) {
memcpy(&((struct sockaddr_in *)(&sa))->sin_addr.s_addr,
&nm->nm_addr.ipv4, sizeof(struct in_addr));
max = 32;
} else {
memcpy(&((struct sockaddr_in6 *)(&sa))->sin6_addr.s6_addr,
&nm->nm_addr.ipv6, sizeof(struct in6_addr));
max = 128;
}
i += rpc_ntop((struct sockaddr *)&sa, buffer + i, count - i);
if (nm->nm_prefix_len < max && !omit_prefix)
i += scnprintf(buffer + i, count - i, "/%u",
nm->nm_prefix_len);
i += cfs_print_network(buffer + i, count - i, nr);
return i;
}
static int
cfs_print_nidmasks(char *buffer, int count, struct list_head *nidmasks,
struct nidrange *nr)
{
int i = 0;
struct nidmask *nm;
list_for_each_entry(nm, nidmasks, nm_link) {
if (i != 0)
i += scnprintf(buffer + i, count - i, " ");
i += print_nidmask(buffer + i, count - i, nm, nr, false);
}
return i;
}
/**
* cfs_print_nidlist() - Print list of @nidranges into the specified @buffer.
* @buffer: buffer holding list of @nidranges
* @count: Max characters can be printed into @buffer.
* @nidlist: Nidranges are separated by a space character.
*
* Return number of characters written
*/
int cfs_print_nidlist(char *buffer, int count, struct list_head *nidlist)
{
int i = 0;
struct nidrange *nr;
bool need_space = false;
if (count <= 0)
return 0;
list_for_each_entry(nr, nidlist, nr_link) {
if (i != 0)
i += scnprintf(buffer + i, count - i, " ");
if (nr->nr_all) {
LASSERT(list_empty(&nr->nr_addrranges));
LASSERT(list_empty(&nr->nr_nidmasks));
i += scnprintf(buffer + i, count - i, "*");
i += cfs_print_network(buffer + i, count - i, nr);
continue;
}
if (!list_empty(&nr->nr_nidmasks)) {
i += cfs_print_nidmasks(buffer + i, count - i,
&nr->nr_nidmasks, nr);
need_space = true;
}
if (!list_empty(&nr->nr_addrranges)) {
if (need_space)
i += scnprintf(buffer + i, count - i, " ");
i += cfs_print_addrranges(buffer + i, count - i,
&nr->nr_addrranges, nr);
}
need_space = false;
}
return i;
}
EXPORT_SYMBOL(cfs_print_nidlist);
/* Caller should provide a nidlist with a single nidmask */
u8
cfs_nidmask_get_length(struct list_head *nidlist)
{
struct nidrange *nr;
struct nidmask *nm;
u8 len = 0;
list_for_each_entry(nr, nidlist, nr_link) {
list_for_each_entry(nm, &nr->nr_nidmasks, nm_link) {
if (len)
return 0;
len = nm->nm_prefix_len;
}
}
return len;
}
EXPORT_SYMBOL(cfs_nidmask_get_length);
/* Caller should provide a nidlist with a single nidmask */
int
cfs_nidmask_get_base_nidstr(char *buf, int count, struct list_head *nidlist)
{
struct nidrange *nr;
struct nidmask *nm;
bool found = false;
list_for_each_entry(nr, nidlist, nr_link) {
list_for_each_entry(nm, &nr->nr_nidmasks, nm_link) {
if (found)
return -EINVAL;
print_nidmask(buf, count, nm, nr, true);
found = true;
}
}
return 0;
}
EXPORT_SYMBOL(cfs_nidmask_get_base_nidstr);
static int
libcfs_lo_str2addr(const char *str, int nob, __u32 *addr)
{
*addr = 0;
return 1;
}
static void
libcfs_ip_addr2str(__u32 addr, char *str, size_t size)
{
snprintf(str, size, "%u.%u.%u.%u",
(addr >> 24) & 0xff, (addr >> 16) & 0xff,
(addr >> 8) & 0xff, addr & 0xff);
}
static void
libcfs_ip_addr2str_size(const __be32 *addr, size_t asize,
char *str, size_t size)
{
struct sockaddr_storage sa = {};
switch (asize) {
case 4:
sa.ss_family = AF_INET;
memcpy(&((struct sockaddr_in *)(&sa))->sin_addr.s_addr,
addr, asize);
break;
case 16:
sa.ss_family = AF_INET6;
memcpy(&((struct sockaddr_in6 *)(&sa))->sin6_addr.s6_addr,
addr, asize);
break;
default:
return;
}
rpc_ntop((struct sockaddr *)&sa, str, size);
}
/* CAVEAT EMPTOR XscanfX
* I use "%n" at the end of a sscanf format to detect trailing junk. However
* sscanf may return immediately if it sees the terminating '0' in a string, so
* I initialise the %n variable to the expected length. If sscanf sets it;
* fine, if it doesn't, then the scan ended at the end of the string, which is
* fine too :) */
static int
libcfs_ip_str2addr(const char *str, int nob, __u32 *addr)
{
unsigned int a;
unsigned int b;
unsigned int c;
unsigned int d;
int n = nob; /* XscanfX */
/* numeric IP? */
if (sscanf(str, "%u.%u.%u.%u%n", &a, &b, &c, &d, &n) >= 4 &&
n == nob &&
(a & ~0xff) == 0 && (b & ~0xff) == 0 &&
(c & ~0xff) == 0 && (d & ~0xff) == 0) {
*addr = ((a<<24)|(b<<16)|(c<<8)|d);
return 1;
}
return 0;
}
static int
libcfs_ip_str2addr_size(const char *str, int nob,
__be32 *addr, size_t *alen)
{
struct sockaddr_storage sa;
/* Note: 'net' arg to rpc_pton is only needed for link-local
* addresses. Such addresses would not work with LNet routing,
* so we can assume they aren't used. So it doesn't matter
* which net namespace is passed.
*/
if (rpc_pton(&init_net, str, nob,
(struct sockaddr *)&sa, sizeof(sa)) == 0)
return 0;
if (sa.ss_family == AF_INET6) {
memcpy(addr,
&((struct sockaddr_in6 *)(&sa))->sin6_addr.s6_addr,
16);
*alen = 16;
return 1;
}
if (sa.ss_family == AF_INET) {
memcpy(addr,
&((struct sockaddr_in *)(&sa))->sin_addr.s_addr,
4);
*alen = 4;
return 1;
}
return 0;
}
/* Used by lnet/config.c so it can't be static */
int
cfs_ip_addr_parse(char *str, int len_ignored, struct list_head *list)
{
struct cfs_expr_list *el;
int rc = 0;
int i = 0;
str = strim(str);
while (rc == 0 && str) {
char *res;
res = strsep(&str, ".");
res = strim(res);
if (!*res) {
rc = -EINVAL;
} else if ((rc = cfs_expr_list_parse(res, strlen(res),
0, 255, &el)) == 0) {
list_add_tail(&el->el_link, list);
i++;
}
}
if (rc == 0 && i == 4)
return 0;
cfs_expr_list_free_list(list);
return rc ?: -EINVAL;
}
/**
* cfs_range_expr_print() - Print range expression @expr into specified @buffer
* @buffer: buffer holding list of @expr
* @count: Max characters can be printed into @buffer.
* @expr: range expression list
* @bracketed: if %true, expression does not need additional brackets.
*
* Return number of characters written
*/
static int
cfs_range_expr_print(char *buffer, int count, struct cfs_range_expr *expr,
bool bracketed)
{
int i;
char s[] = "[";
char e[] = "]";
if (bracketed)
s[0] = e[0] = '\0';
if (expr->re_lo == expr->re_hi)
i = scnprintf(buffer, count, "%u", expr->re_lo);
else if (expr->re_stride == 1)
i = scnprintf(buffer, count, "%s%u-%u%s",
s, expr->re_lo, expr->re_hi, e);
else
i = scnprintf(buffer, count, "%s%u-%u/%u%s",
s, expr->re_lo, expr->re_hi,
expr->re_stride, e);
return i;
}
/**
* cfs_expr_list_print() - Print a list of range expressions
* @buffer: Buffer holding range expression for print
* @count: Max length to print
* @expr_list: Range expression
*
* Print a list of range expressions (@expr_list) into specified @buffer.
* If the list contains several expressions, separate them with comma
* and surround the list with brackets.
*
* Returns number of characters written
*/
static int
cfs_expr_list_print(char *buffer, int count, struct cfs_expr_list *expr_list)
{
struct cfs_range_expr *expr;
int i = 0, j = 0;
int numexprs = 0;
if (count <= 0)
return 0;
list_for_each_entry(expr, &expr_list->el_exprs, re_link)
numexprs++;
if (numexprs > 1)
i += scnprintf(buffer + i, count - i, "[");
list_for_each_entry(expr, &expr_list->el_exprs, re_link) {
if (j++ != 0)
i += scnprintf(buffer + i, count - i, ",");
i += cfs_range_expr_print(buffer + i, count - i, expr,
numexprs > 1);
}
if (numexprs > 1)
i += scnprintf(buffer + i, count - i, "]");
return i;
}
static int
libcfs_ip_addr_range_print(char *buffer, int count, struct list_head *list)
{
int i = 0, j = 0;
struct cfs_expr_list *el;
list_for_each_entry(el, list, el_link) {
LASSERT(j++ < 4);
if (i != 0)
i += scnprintf(buffer + i, count - i, ".");
i += cfs_expr_list_print(buffer + i, count - i, el);
}
return i;
}
/**
* cfs_ip_addr_match() - Matches address (@addr) against address set encoded
* in @list.
* @addr: IP address to be checked
* @list: List contains parsed IP address
*
* Return:
* * %1 if @addr matches
* * %0 otherwise
*/
int
cfs_ip_addr_match(__u32 addr, struct list_head *list)
{
struct cfs_expr_list *el;
int i = 0;
list_for_each_entry_reverse(el, list, el_link) {
if (!cfs_expr_list_match(addr & 0xff, el))
return 0;
addr >>= 8;
i++;
}
return i == 4;
}
/**
* libcfs_ip_in_netmask() - Matches address (@addr) against the netmask encoded
* in @netmask and @netaddr.
* @addr: IP address to be checked
* @asize: Size of @addr
* @netmask: Netmask
* @netaddr: Network address
*
* Return:
* * %1 if @addr matches
* * %0 otherwise
*/
int
libcfs_ip_in_netmask(const __be32 *addr, size_t asize, const __be32 *netmask,
const __be32 *netaddr)
{
if (asize == 4) {
struct in_addr nid_addr, masked_addr;
memcpy(&nid_addr.s_addr, addr, asize);
masked_addr.s_addr = nid_addr.s_addr &
(*(struct in_addr *)netmask).s_addr;
return memcmp(&masked_addr, netaddr, sizeof(masked_addr)) == 0;
} else if (asize == 16) {
struct in6_addr nid_addr, masked_addr;
int i;
memcpy(&nid_addr.s6_addr, addr, asize);
for (i = 0; i < 16; i++)
masked_addr.s6_addr[i] =
nid_addr.s6_addr[i] &
(*(struct in6_addr *)netmask).s6_addr[i];
return memcmp(&masked_addr, netaddr, sizeof(masked_addr)) == 0;
}
return 0;
}
/**
* libcfs_decnum_addr2str() - Print the network part of the nidrange @nr into
* the specified buffer(@str)
* @addr: numeric address to be converted
* @str: Converted decimial string will be stored [out]
* @size: length of @str
*
* Returns number of characters written
*/
static void
libcfs_decnum_addr2str(__u32 addr, char *str, size_t size)
{
snprintf(str, size, "%u", addr);
}
static int
libcfs_num_str2addr(const char *str, int nob, __u32 *addr)
{
int n;
n = nob;
if (sscanf(str, "0x%x%n", addr, &n) >= 1 && n == nob)
return 1;
n = nob;
if (sscanf(str, "0X%x%n", addr, &n) >= 1 && n == nob)
return 1;
n = nob;
if (sscanf(str, "%u%n", addr, &n) >= 1 && n == nob)
return 1;
return 0;
}
/**
* libcfs_num_parse() - Nf_parse_addrlist method for networks using numeric
* addresses.
* @str: numeric address information
* @len: length of @str
* @list: Parsed to numeric address will be added [out]
*
* Examples of such networks are gm and elan.
*
* Return:
* * %0 if @str parsed to numeric address on success
* * %errno otherwise
*/
int
libcfs_num_parse(char *str, int len, struct list_head *list)
{
struct cfs_expr_list *el;
int rc;
rc = cfs_expr_list_parse(str, len, 0, MAX_NUMERIC_VALUE, &el);
if (rc == 0)
list_add_tail(&el->el_link, list);
return rc;
}
static int
libcfs_num_addr_range_print(char *buffer, int count, struct list_head *list)
{
int i = 0, j = 0;
struct cfs_expr_list *el;
list_for_each_entry(el, list, el_link) {
LASSERT(j++ < 1);
i += cfs_expr_list_print(buffer + i, count - i, el);
}
return i;
}
/*
* Nf_match_addr method for networks using numeric addresses
*
* \retval 1 on match
* \retval 0 otherwise
*/
static int
libcfs_num_match(__u32 addr, struct list_head *numaddr)
{
struct cfs_expr_list *el;
LASSERT(!list_empty(numaddr));
el = list_first_entry(numaddr, struct cfs_expr_list, el_link);
return cfs_expr_list_match(addr, el);
}
static struct netstrfns libcfs_netstrfns[] = {
{
.nf_type = LOLND,
.nf_name = "lo",
.nf_modname = "klolnd",
.nf_addr2str = libcfs_decnum_addr2str,
.nf_str2addr = libcfs_lo_str2addr,
.nf_parse_addrlist = libcfs_num_parse,
.nf_print_addrlist = libcfs_num_addr_range_print,
.nf_match_addr = libcfs_num_match
},
{ .nf_type = SOCKLND,
.nf_name = "tcp",
.nf_modname = "ksocklnd",
.nf_addr2str = libcfs_ip_addr2str,
.nf_addr2str_size = libcfs_ip_addr2str_size,
.nf_str2addr = libcfs_ip_str2addr,
.nf_str2addr_size = libcfs_ip_str2addr_size,
.nf_parse_addrlist = cfs_ip_addr_parse,
.nf_print_addrlist = libcfs_ip_addr_range_print,
.nf_match_addr = cfs_ip_addr_match,
.nf_match_netmask = libcfs_ip_in_netmask
},
{ .nf_type = O2IBLND,
.nf_name = "o2ib",
.nf_modname = "ko2iblnd",
.nf_addr2str = libcfs_ip_addr2str,
.nf_str2addr = libcfs_ip_str2addr,
.nf_parse_addrlist = cfs_ip_addr_parse,
.nf_print_addrlist = libcfs_ip_addr_range_print,
.nf_match_addr = cfs_ip_addr_match,
.nf_match_netmask = libcfs_ip_in_netmask
},
{
.nf_type = GNILND,
.nf_name = "gni",
.nf_modname = "kgnilnd",
.nf_addr2str = libcfs_decnum_addr2str,
.nf_str2addr = libcfs_num_str2addr,
.nf_parse_addrlist = libcfs_num_parse,
.nf_print_addrlist = libcfs_num_addr_range_print,
.nf_match_addr = libcfs_num_match
},
{
.nf_type = GNIIPLND,
.nf_name = "gip",
.nf_modname = "kgnilnd",
.nf_addr2str = libcfs_ip_addr2str,
.nf_str2addr = libcfs_ip_str2addr,
.nf_parse_addrlist = cfs_ip_addr_parse,
.nf_print_addrlist = libcfs_ip_addr_range_print,
.nf_match_addr = cfs_ip_addr_match
},
{
.nf_type = PTL4LND,
.nf_name = "ptlf",
.nf_modname = "kptl4lnd",
.nf_addr2str = libcfs_decnum_addr2str,
.nf_str2addr = libcfs_num_str2addr,
.nf_parse_addrlist = libcfs_num_parse,
.nf_print_addrlist = libcfs_num_addr_range_print,
.nf_match_addr = libcfs_num_match
},
{
.nf_type = KFILND,
.nf_name = "kfi",
.nf_modname = "kkfilnd",
.nf_addr2str = libcfs_decnum_addr2str,
.nf_str2addr = libcfs_num_str2addr,
.nf_parse_addrlist = libcfs_num_parse,
.nf_print_addrlist = libcfs_num_addr_range_print,
.nf_match_addr = libcfs_num_match
},
{
.nf_type = EFALND,
.nf_name = "efa",
.nf_modname = "kefalnd",
.nf_addr2str = libcfs_ip_addr2str,
.nf_addr2str_size = libcfs_ip_addr2str_size,
.nf_str2addr = libcfs_ip_str2addr,
.nf_str2addr_size = libcfs_ip_str2addr_size,
.nf_parse_addrlist = cfs_ip_addr_parse,
.nf_print_addrlist = libcfs_ip_addr_range_print,
.nf_match_addr = cfs_ip_addr_match
},
{
.nf_type = BXI3LND,
.nf_name = "bxi3f",
.nf_modname = "kbxi3lnd",
.nf_addr2str = libcfs_decnum_addr2str,
.nf_str2addr = libcfs_num_str2addr,
.nf_parse_addrlist = libcfs_num_parse,
.nf_print_addrlist = libcfs_num_addr_range_print,
.nf_match_addr = libcfs_num_match
},
};
static const size_t libcfs_nnetstrfns = ARRAY_SIZE(libcfs_netstrfns);
static struct netstrfns *
type2net_info(__u32 net_type)
{
int i;
for (i = 0; i < libcfs_nnetstrfns; i++) {
if (libcfs_netstrfns[i].nf_type == net_type)
return &libcfs_netstrfns[i];
}
return NULL;
}
int
cfs_match_net(__u32 net_id, __u32 net_type, struct list_head *net_num_list)
{
__u32 net_num;
if (!net_num_list)
return 0;
if (net_type != LNET_NETTYP(net_id))
return 0;
net_num = LNET_NETNUM(net_id);
/* if there is a net number but the list passed in is empty, then
* there is no match.
*/
if (!net_num && list_empty(net_num_list))
return 1;
else if (list_empty(net_num_list))
return 0;
if (!libcfs_num_match(net_num, net_num_list))
return 0;
return 1;
}
int
cfs_match_nid_net(struct lnet_nid *nid, __u32 net_type,
struct list_head *net_num_list,
struct list_head *addr)
{
__u32 address;
struct netstrfns *nf;
if (!addr || list_empty(addr) || !net_num_list)
return 0;
nf = type2net_info(LNET_NETTYP(LNET_NID_NET(nid)));
if (!nf)
return 0;
/* FIXME handle long-addr nid */
address = LNET_NIDADDR(lnet_nid_to_nid4(nid));
/* if either the address or net number don't match then no match */
if (!nf->nf_match_addr(address, addr) ||
!cfs_match_net(LNET_NID_NET(nid), net_type, net_num_list))
return 0;
return 1;
}
EXPORT_SYMBOL(cfs_match_nid_net);
static struct netstrfns *
libcfs_lnd2netstrfns(__u32 lnd)
{
int i;
for (i = 0; i < libcfs_nnetstrfns; i++)
if (lnd == libcfs_netstrfns[i].nf_type)
return &libcfs_netstrfns[i];
return NULL;
}
static struct netstrfns *
libcfs_namenum2netstrfns(const char *name)
{
struct netstrfns *nf;
int i;
for (i = 0; i < libcfs_nnetstrfns; i++) {
nf = &libcfs_netstrfns[i];
if (!strncmp(name, nf->nf_name, strlen(nf->nf_name)))
return nf;
}
return NULL;
}
static struct netstrfns *
libcfs_name2netstrfns(const char *name)
{
int i;
for (i = 0; i < libcfs_nnetstrfns; i++)
if (!strcmp(libcfs_netstrfns[i].nf_name, name))
return &libcfs_netstrfns[i];
return NULL;
}
int
libcfs_isknown_lnd(__u32 lnd)
{
return libcfs_lnd2netstrfns(lnd) != NULL;
}
EXPORT_SYMBOL(libcfs_isknown_lnd);
char *
libcfs_lnd2modname(__u32 lnd)
{
struct netstrfns *nf = libcfs_lnd2netstrfns(lnd);
return (nf == NULL) ? NULL : nf->nf_modname;
}
EXPORT_SYMBOL(libcfs_lnd2modname);
int
libcfs_str2lnd(const char *str)
{
struct netstrfns *nf = libcfs_name2netstrfns(str);
if (nf != NULL)
return nf->nf_type;
return -ENXIO;
}
EXPORT_SYMBOL(libcfs_str2lnd);
char *
libcfs_lnd2str_r(__u32 lnd, char *buf, size_t buf_size)
{
struct netstrfns *nf;
nf = libcfs_lnd2netstrfns(lnd);
if (nf == NULL)
snprintf(buf, buf_size, "?%u?", lnd);
else
snprintf(buf, buf_size, "%s", nf->nf_name);
return buf;
}
EXPORT_SYMBOL(libcfs_lnd2str_r);
char *
libcfs_net2str_r(__u32 net, char *buf, size_t buf_size)
{
__u32 nnum = LNET_NETNUM(net);
__u32 lnd = LNET_NETTYP(net);
struct netstrfns *nf;
nf = libcfs_lnd2netstrfns(lnd);
if (nf == NULL)
snprintf(buf, buf_size, "<%u:%u>", lnd, nnum);
else if (nnum == 0)
snprintf(buf, buf_size, "%s", nf->nf_name);
else
snprintf(buf, buf_size, "%s%u", nf->nf_name, nnum);
return buf;
}
EXPORT_SYMBOL(libcfs_net2str_r);
char *
libcfs_nid2str_r(lnet_nid_t nid, char *buf, size_t buf_size)
{
__u32 addr = LNET_NIDADDR(nid);
__u32 net = LNET_NIDNET(nid);
__u32 nnum = LNET_NETNUM(net);
__u32 lnd = LNET_NETTYP(net);
struct netstrfns *nf;
if (nid == LNET_NID_ANY) {
strncpy(buf, "<?>", buf_size);
buf[buf_size - 1] = '\0';
return buf;
}
nf = libcfs_lnd2netstrfns(lnd);
if (nf == NULL) {
snprintf(buf, buf_size, "%x@<%u:%u>", addr, lnd, nnum);
} else {
size_t addr_len;
nf->nf_addr2str(addr, buf, buf_size);
addr_len = strlen(buf);
if (nnum == 0)
snprintf(buf + addr_len, buf_size - addr_len, "@%s",
nf->nf_name);
else
snprintf(buf + addr_len, buf_size - addr_len, "@%s%u",
nf->nf_name, nnum);
}
return buf;
}
EXPORT_SYMBOL(libcfs_nid2str_r);
char *
libcfs_nidstr_r(const struct lnet_nid *nid, char *buf, size_t buf_size)
{
__u32 nnum;
__u32 lnd;
struct netstrfns *nf;
if (LNET_NID_IS_ANY(nid)) {
strncpy(buf, "<?>", buf_size);
buf[buf_size - 1] = '\0';
return buf;
}
nnum = be16_to_cpu(nid->nid_num);
lnd = nid->nid_type;
nf = libcfs_lnd2netstrfns(lnd);
if (nf) {
size_t addr_len;
if (nf->nf_addr2str_size)
nf->nf_addr2str_size(nid->nid_addr, NID_ADDR_BYTES(nid),
buf, buf_size);
else
nf->nf_addr2str(ntohl(nid->nid_addr[0]), buf, buf_size);
addr_len = strlen(buf);
if (nnum == 0)
snprintf(buf + addr_len, buf_size - addr_len, "@%s",
nf->nf_name);
else
snprintf(buf + addr_len, buf_size - addr_len, "@%s%u",
nf->nf_name, nnum);
} else {
int l = 0;
int words = DIV_ROUND_UP(NID_ADDR_BYTES(nid), 4);
int i;
for (i = 0; i < words && i < 4; i++)
l = snprintf(buf+l, buf_size-l, "%s%x",
i ? ":" : "", ntohl(nid->nid_addr[i]));
snprintf(buf+l, buf_size-l, "@<%u:%u>", lnd, nnum);
}
return buf;
}
EXPORT_SYMBOL(libcfs_nidstr_r);
static struct netstrfns *
libcfs_str2net_internal(const char *str, __u32 *net)
{
struct netstrfns *nf = NULL;
int nob;
unsigned int netnum;
int i;
for (i = 0; i < libcfs_nnetstrfns; i++) {
nf = &libcfs_netstrfns[i];
if (!strncmp(str, nf->nf_name, strlen(nf->nf_name)))
break;
}
if (i == libcfs_nnetstrfns)
return NULL;
nob = strlen(nf->nf_name);
if (strlen(str) == (unsigned int)nob) {
netnum = 0;
} else {
if (nf->nf_type == LOLND) /* net number not allowed */
return NULL;
str += nob;
i = strlen(str);
if (sscanf(str, "%u%n", &netnum, &i) < 1 ||
i != (int)strlen(str))
return NULL;
}
*net = LNET_MKNET(nf->nf_type, netnum);
return nf;
}
__u32
libcfs_str2net(const char *str)
{
__u32 net;
if (libcfs_str2net_internal(str, &net) != NULL)
return net;
return LNET_NET_ANY;
}
EXPORT_SYMBOL(libcfs_str2net);
lnet_nid_t
libcfs_str2nid(const char *str)
{
const char *sep = strchr(str, '@');
struct netstrfns *nf;
__u32 net;
__u32 addr;
if (sep != NULL) {
nf = libcfs_str2net_internal(sep + 1, &net);
if (nf == NULL)
return LNET_NID_ANY;
} else {
sep = str + strlen(str);
net = LNET_MKNET(SOCKLND, 0);
nf = libcfs_lnd2netstrfns(SOCKLND);
LASSERT(nf != NULL);
}
if (!nf->nf_str2addr(str, (int)(sep - str), &addr))
return LNET_NID_ANY;
return LNET_MKNID(net, addr);
}
EXPORT_SYMBOL(libcfs_str2nid);
int
libcfs_strnid(struct lnet_nid *nid, const char *str)
{
const char *sep = strchr(str, '@');
struct netstrfns *nf;
__u32 net;
if (sep != NULL) {
nf = libcfs_str2net_internal(sep + 1, &net);
if (nf == NULL)
return -EINVAL;
} else {
if (strcmp(str, "<?>") == 0) {
memcpy(nid, &LNET_ANY_NID, sizeof(*nid));
return 0;
}
sep = str + strlen(str);
net = LNET_MKNET(SOCKLND, 0);
nf = libcfs_lnd2netstrfns(SOCKLND);
LASSERT(nf != NULL);
}
memset(nid, 0, sizeof(*nid));
nid->nid_type = LNET_NETTYP(net);
nid->nid_num = htons(LNET_NETNUM(net));
if (nf->nf_str2addr_size) {
size_t asize = 0;
if (!nf->nf_str2addr_size(str, (int)(sep - str),
nid->nid_addr, &asize))
return -EINVAL;
nid->nid_size = asize - 4;
} else {
__u32 addr;
if (!nf->nf_str2addr(str, (int)(sep - str), &addr))
return -EINVAL;
nid->nid_addr[0] = htonl(addr);
nid->nid_size = 0;
}
return 0;
}
EXPORT_SYMBOL(libcfs_strnid);
char *
libcfs_id2str(struct lnet_process_id id)
{
char *str = libcfs_next_nidstring();
if (id.pid == LNET_PID_ANY) {
snprintf(str, LNET_NIDSTR_SIZE,
"LNET_PID_ANY-%s", libcfs_nid2str(id.nid));
return str;
}
snprintf(str, LNET_NIDSTR_SIZE, "%s%u-%s",
((id.pid & LNET_PID_USERFLAG) != 0) ? "U" : "",
(id.pid & ~LNET_PID_USERFLAG), libcfs_nid2str(id.nid));
return str;
}
EXPORT_SYMBOL(libcfs_id2str);
char *
libcfs_idstr(struct lnet_processid *id)
{
char *str = libcfs_next_nidstring();
if (id->pid == LNET_PID_ANY) {
snprintf(str, LNET_NIDSTR_SIZE,
"LNET_PID_ANY-%s", libcfs_nidstr(&id->nid));
return str;
}
snprintf(str, LNET_NIDSTR_SIZE, "%s%u-%s",
((id->pid & LNET_PID_USERFLAG) != 0) ? "U" : "",
(id->pid & ~LNET_PID_USERFLAG), libcfs_nidstr(&id->nid));
return str;
}
EXPORT_SYMBOL(libcfs_idstr);
int
libcfs_strid(struct lnet_processid *id, const char *str)
{
char *tmp = strchr(str, '-');
id->pid = LNET_PID_LUSTRE;
if (tmp &&
strncmp("LNET_PID_ANY-", str, tmp - str) != 0) {
char pid[LNET_NIDSTR_SIZE];
int rc;
strscpy(pid, str, tmp - str);
rc = kstrtou32(pid, 10, &id->pid);
if (rc < 0)
return rc;
tmp++;
} else {
tmp = (char *)str;
}
return libcfs_strnid(&id->nid, tmp);
}
EXPORT_SYMBOL(libcfs_strid);
int
libcfs_str2anynid(lnet_nid_t *nidp, const char *str)
{
if (!strcmp(str, "*")) {
*nidp = LNET_NID_ANY;
return 1;
}
*nidp = libcfs_str2nid(str);
return *nidp != LNET_NID_ANY;
}
EXPORT_SYMBOL(libcfs_str2anynid);
int
libcfs_stranynid(struct lnet_nid *nid, const char *str)
{
if (!strcmp(str, "*")) {
*nid = LNET_ANY_NID;
return 1;
}
if (libcfs_strnid(nid, str))
*nid = LNET_ANY_NID;
return !LNET_NID_IS_ANY(nid);
}
EXPORT_SYMBOL(libcfs_stranynid);