2008-05-22 21:41:43 +04:00
|
|
|
/*
|
2008-05-22 22:02:20 +04:00
|
|
|
* Copyright (c) 2008 Chelsio, Inc. All rights reserved.
|
2008-05-22 21:41:43 +04:00
|
|
|
* Copyright (c) 2008 Cisco Systems, Inc. All rights reserved.
|
|
|
|
*
|
|
|
|
* Additional copyrights may follow
|
|
|
|
*
|
|
|
|
* $HEADER$
|
|
|
|
*
|
|
|
|
* @file
|
|
|
|
*/
|
|
|
|
|
2008-05-08 18:38:14 +04:00
|
|
|
#include "ompi_config.h"
|
2008-05-22 21:41:43 +04:00
|
|
|
|
2008-05-13 20:01:58 +04:00
|
|
|
#include <infiniband/verbs.h>
|
2008-05-08 18:38:14 +04:00
|
|
|
|
2008-05-22 21:41:43 +04:00
|
|
|
#if OMPI_HAVE_RDMACM
|
2008-05-08 18:38:14 +04:00
|
|
|
#include <rdma/rdma_cma.h>
|
|
|
|
#include <malloc.h>
|
2008-10-06 04:46:02 +04:00
|
|
|
#include <stdio.h>
|
2008-05-08 18:38:14 +04:00
|
|
|
|
|
|
|
#include "opal/util/argv.h"
|
2008-05-13 22:34:22 +04:00
|
|
|
#include "opal/util/if.h"
|
2008-05-08 18:38:14 +04:00
|
|
|
|
|
|
|
#include "connect/connect.h"
|
2008-05-22 21:41:43 +04:00
|
|
|
#endif
|
|
|
|
/* Always want to include this file */
|
2008-10-08 13:50:54 +04:00
|
|
|
#include "btl_openib_endpoint.h"
|
2008-05-08 18:38:14 +04:00
|
|
|
#include "btl_openib_iwarp.h"
|
2008-05-22 21:41:43 +04:00
|
|
|
#if OMPI_HAVE_RDMACM
|
2008-05-08 18:38:14 +04:00
|
|
|
|
|
|
|
/*
|
|
|
|
* The cruft below maintains the linked list of rdma ipv4 addresses and their
|
|
|
|
* associated rdma device names and device port numbers.
|
|
|
|
*/
|
|
|
|
struct rdma_addr_list {
|
|
|
|
opal_list_item_t super;
|
|
|
|
uint32_t addr;
|
|
|
|
uint32_t subnet;
|
|
|
|
char addr_str[16];
|
|
|
|
char dev_name[IBV_SYSFS_NAME_MAX];
|
|
|
|
uint8_t dev_port;
|
|
|
|
};
|
|
|
|
typedef struct rdma_addr_list rdma_addr_list_t;
|
|
|
|
|
|
|
|
static OBJ_CLASS_INSTANCE(rdma_addr_list_t, opal_list_item_t,
|
|
|
|
NULL, NULL);
|
2008-06-24 23:16:06 +04:00
|
|
|
static opal_list_t *myaddrs = NULL;
|
2008-05-08 18:38:14 +04:00
|
|
|
|
2008-10-06 04:46:02 +04:00
|
|
|
#if OMPI_ENABLE_DEBUG
|
|
|
|
static char *stringify(uint32_t addr)
|
|
|
|
{
|
|
|
|
static char line[64];
|
|
|
|
memset(line, 0, sizeof(line));
|
|
|
|
snprintf(line, sizeof(line) - 1, "%d.%d.%d.%d (0x%x)",
|
|
|
|
#if defined(WORDS_BIGENDIAN)
|
|
|
|
(addr >> 24),
|
|
|
|
(addr >> 16) & 0xff,
|
|
|
|
(addr >> 8) & 0xff,
|
|
|
|
addr & 0xff,
|
|
|
|
#else
|
|
|
|
addr & 0xff,
|
|
|
|
(addr >> 8) & 0xff,
|
|
|
|
(addr >> 16) & 0xff,
|
|
|
|
(addr >> 24),
|
|
|
|
#endif
|
|
|
|
addr);
|
|
|
|
return line;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2008-05-21 03:28:59 +04:00
|
|
|
uint64_t mca_btl_openib_get_iwarp_subnet_id(struct ibv_device *ib_dev)
|
2008-05-08 18:38:14 +04:00
|
|
|
{
|
|
|
|
opal_list_item_t *item;
|
|
|
|
|
2008-10-06 04:46:02 +04:00
|
|
|
/* In the off chance that the user forces non-rdmacm cpc and
|
|
|
|
* iwarp, the list will be uninitialized. Return 0 to prevent
|
|
|
|
* crashes, and the lack of it actually working will be caught at
|
|
|
|
* a later stage.
|
2008-06-24 23:16:06 +04:00
|
|
|
*/
|
|
|
|
if (NULL == myaddrs) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (item = opal_list_get_first(myaddrs);
|
|
|
|
item != opal_list_get_end(myaddrs);
|
2008-05-08 18:38:14 +04:00
|
|
|
item = opal_list_get_next(item)) {
|
|
|
|
struct rdma_addr_list *addr = (struct rdma_addr_list *)item;
|
|
|
|
if (!strcmp(addr->dev_name, ib_dev->name)) {
|
|
|
|
return addr->subnet;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2008-10-06 04:46:02 +04:00
|
|
|
uint32_t mca_btl_openib_rdma_get_ipv4addr(struct ibv_context *verbs,
|
|
|
|
uint8_t port)
|
2008-05-08 18:38:14 +04:00
|
|
|
{
|
|
|
|
opal_list_item_t *item;
|
|
|
|
|
2008-10-06 04:46:02 +04:00
|
|
|
/* Sanity check */
|
|
|
|
if (NULL == myaddrs) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
BTL_VERBOSE(("Looking for %s:%d in IP address list",
|
|
|
|
ibv_get_device_name(verbs->device), port));
|
2008-06-24 23:16:06 +04:00
|
|
|
for (item = opal_list_get_first(myaddrs);
|
|
|
|
item != opal_list_get_end(myaddrs);
|
2008-05-08 18:38:14 +04:00
|
|
|
item = opal_list_get_next(item)) {
|
|
|
|
struct rdma_addr_list *addr = (struct rdma_addr_list *)item;
|
|
|
|
if (!strcmp(addr->dev_name, verbs->device->name) &&
|
|
|
|
port == addr->dev_port) {
|
2008-10-06 04:46:02 +04:00
|
|
|
BTL_VERBOSE(("FOUND: %s:%d is %s",
|
|
|
|
ibv_get_device_name(verbs->device), port,
|
|
|
|
stringify(addr->addr)));
|
2008-05-08 18:38:14 +04:00
|
|
|
return addr->addr;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int dev_specified(char *name, int port)
|
|
|
|
{
|
|
|
|
char **list;
|
|
|
|
|
|
|
|
if (NULL != mca_btl_openib_component.if_include) {
|
|
|
|
int i;
|
|
|
|
|
|
|
|
list = opal_argv_split(mca_btl_openib_component.if_include, ',');
|
|
|
|
for (i = 0; NULL != list[i]; i++) {
|
|
|
|
char **temp = opal_argv_split(list[i], ':');
|
|
|
|
if (0 == strcmp(name, temp[0]) &&
|
|
|
|
(NULL == temp[1] || port == atoi(temp[1]))) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (NULL != mca_btl_openib_component.if_exclude) {
|
|
|
|
int i;
|
|
|
|
|
|
|
|
list = opal_argv_split(mca_btl_openib_component.if_exclude, ',');
|
|
|
|
for (i = 0; NULL != list[i]; i++) {
|
|
|
|
char **temp = opal_argv_split(list[i], ':');
|
|
|
|
if (0 == strcmp(name, temp[0]) &&
|
|
|
|
(NULL == temp[1] || port == atoi(temp[1]))) {
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2008-05-13 22:34:22 +04:00
|
|
|
static int add_rdma_addr(struct sockaddr *ipaddr, uint32_t netmask)
|
2008-05-08 18:38:14 +04:00
|
|
|
{
|
|
|
|
struct sockaddr_in *sinp;
|
|
|
|
struct rdma_cm_id *cm_id;
|
|
|
|
struct rdma_event_channel *ch;
|
|
|
|
int rc = OMPI_SUCCESS;
|
|
|
|
struct rdma_addr_list *myaddr;
|
|
|
|
|
|
|
|
ch = rdma_create_event_channel();
|
|
|
|
if (NULL == ch) {
|
2008-10-06 04:46:02 +04:00
|
|
|
BTL_VERBOSE(("failed creating RDMA CM event channel"));
|
2008-05-08 18:38:14 +04:00
|
|
|
rc = OMPI_ERROR;
|
|
|
|
goto out1;
|
|
|
|
}
|
|
|
|
|
|
|
|
rc = rdma_create_id(ch, &cm_id, NULL, RDMA_PS_TCP);
|
|
|
|
if (rc) {
|
2008-10-06 04:46:02 +04:00
|
|
|
BTL_VERBOSE(("rdma_create_id returned %d", rc));
|
2008-05-08 18:38:14 +04:00
|
|
|
rc = OMPI_ERROR;
|
|
|
|
goto out2;
|
|
|
|
}
|
|
|
|
|
2008-05-13 22:34:22 +04:00
|
|
|
rc = rdma_bind_addr(cm_id, ipaddr);
|
2008-05-08 18:38:14 +04:00
|
|
|
if (rc) {
|
|
|
|
rc = OMPI_SUCCESS;
|
|
|
|
goto out3;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!cm_id->verbs ||
|
2008-05-13 22:34:22 +04:00
|
|
|
0 == ((struct sockaddr_in *)ipaddr)->sin_addr.s_addr ||
|
2008-05-08 18:38:14 +04:00
|
|
|
dev_specified(cm_id->verbs->device->name, cm_id->port_num)) {
|
|
|
|
goto out3;
|
|
|
|
}
|
|
|
|
|
|
|
|
myaddr = OBJ_NEW(rdma_addr_list_t);
|
|
|
|
if (NULL == myaddr) {
|
|
|
|
BTL_ERROR(("malloc failed!"));
|
|
|
|
rc = OMPI_ERROR;
|
|
|
|
goto out3;
|
|
|
|
}
|
|
|
|
|
2008-05-13 22:34:22 +04:00
|
|
|
sinp = (struct sockaddr_in *)ipaddr;
|
2008-05-08 18:38:14 +04:00
|
|
|
myaddr->addr = sinp->sin_addr.s_addr;
|
2008-05-13 22:34:22 +04:00
|
|
|
myaddr->subnet = myaddr->addr & netmask;
|
2008-05-08 18:38:14 +04:00
|
|
|
inet_ntop(sinp->sin_family, &sinp->sin_addr,
|
2008-10-06 04:46:02 +04:00
|
|
|
myaddr->addr_str, sizeof(myaddr->addr_str));
|
2008-05-08 18:38:14 +04:00
|
|
|
memcpy(myaddr->dev_name, cm_id->verbs->device->name, IBV_SYSFS_NAME_MAX);
|
|
|
|
myaddr->dev_port = cm_id->port_num;
|
2008-10-06 04:46:02 +04:00
|
|
|
BTL_VERBOSE(("Adding addr %s (0x%x) as %s:%d",
|
|
|
|
myaddr->addr_str, myaddr->addr,
|
|
|
|
myaddr->dev_name, myaddr->dev_port));
|
2008-05-08 18:38:14 +04:00
|
|
|
|
2008-06-24 23:16:06 +04:00
|
|
|
opal_list_append(myaddrs, &(myaddr->super));
|
2008-05-08 18:38:14 +04:00
|
|
|
|
|
|
|
out3:
|
|
|
|
rdma_destroy_id(cm_id);
|
|
|
|
out2:
|
|
|
|
rdma_destroy_event_channel(ch);
|
|
|
|
out1:
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
2008-05-21 03:28:59 +04:00
|
|
|
int mca_btl_openib_build_rdma_addr_list(void)
|
2008-05-08 18:38:14 +04:00
|
|
|
{
|
2008-06-04 23:03:02 +04:00
|
|
|
int rc = OMPI_SUCCESS, i;
|
2008-05-08 18:38:14 +04:00
|
|
|
|
2008-06-24 23:16:06 +04:00
|
|
|
myaddrs = OBJ_NEW(opal_list_t);
|
|
|
|
if (NULL == myaddrs) {
|
|
|
|
BTL_ERROR(("malloc failed!"));
|
|
|
|
return OMPI_ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
OBJ_CONSTRUCT(myaddrs, opal_list_t);
|
2008-05-08 18:38:14 +04:00
|
|
|
|
2008-05-22 06:10:23 +04:00
|
|
|
for (i = opal_ifbegin(); i >= 0; i = opal_ifnext(i)) {
|
2008-05-13 22:34:22 +04:00
|
|
|
struct sockaddr ipaddr;
|
|
|
|
uint32_t netmask;
|
|
|
|
|
|
|
|
opal_ifindextoaddr(i, &ipaddr, sizeof(struct sockaddr));
|
|
|
|
opal_ifindextomask(i, &netmask, sizeof(uint32_t));
|
2008-05-08 18:38:14 +04:00
|
|
|
|
2008-05-13 22:34:22 +04:00
|
|
|
if (ipaddr.sa_family == AF_INET) {
|
|
|
|
rc = add_rdma_addr(&ipaddr, netmask);
|
2008-05-08 18:38:14 +04:00
|
|
|
if (OMPI_SUCCESS != rc) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
2008-05-21 03:28:59 +04:00
|
|
|
void mca_btl_openib_free_rdma_addr_list(void)
|
2008-05-08 18:38:14 +04:00
|
|
|
{
|
|
|
|
opal_list_item_t *item;
|
|
|
|
|
2008-06-25 00:38:47 +04:00
|
|
|
if (NULL != myaddrs && 0 != opal_list_get_size(myaddrs)) {
|
2008-06-24 23:16:06 +04:00
|
|
|
for (item = opal_list_get_first(myaddrs);
|
|
|
|
item != opal_list_get_end(myaddrs);
|
2008-05-08 18:38:14 +04:00
|
|
|
item = opal_list_get_next(item)) {
|
2008-06-24 23:16:06 +04:00
|
|
|
struct rdma_addr_list *addr = (struct rdma_addr_list *)item;
|
|
|
|
opal_list_remove_item(myaddrs, item);
|
|
|
|
OBJ_RELEASE(addr);
|
2008-05-08 18:38:14 +04:00
|
|
|
}
|
2008-06-25 00:38:47 +04:00
|
|
|
OBJ_RELEASE(myaddrs);
|
2008-05-08 18:38:14 +04:00
|
|
|
}
|
|
|
|
}
|
2008-10-06 04:46:02 +04:00
|
|
|
|
|
|
|
#else
|
|
|
|
/* !OMPI_HAVE_RDMACM case */
|
|
|
|
|
2008-05-22 06:10:23 +04:00
|
|
|
uint64_t mca_btl_openib_get_iwarp_subnet_id(struct ibv_device *ib_dev)
|
|
|
|
{
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
uint32_t mca_btl_openib_rdma_get_ipv4addr(struct ibv_context *verbs,
|
|
|
|
uint8_t port)
|
|
|
|
{
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int mca_btl_openib_build_rdma_addr_list(void)
|
|
|
|
{
|
2008-10-06 04:46:02 +04:00
|
|
|
return OMPI_SUCCESS;
|
2008-05-22 06:10:23 +04:00
|
|
|
}
|
|
|
|
|
2008-10-06 04:46:02 +04:00
|
|
|
void mca_btl_openib_free_rdma_addr_list(void)
|
|
|
|
{
|
2008-05-22 06:10:23 +04:00
|
|
|
}
|
2008-05-08 18:38:14 +04:00
|
|
|
#endif
|