/*
* Copyright (c) 2019, Mellanox Technologies. All rights reserved.
*
* This software is available to you under a choice of one of two
* licenses. You may choose to be licensed under the terms of the GNU
* General Public License (GPL) Version 2, available from the file
* COPYING in the main directory of this source tree, or the
* OpenIB.org BSD license below:
*
* Redistribution and use in source and binary forms, with or
* without modification, are permitted provided that the following
* conditions are met:
*
* - Redistributions of source code must retain the above
* copyright notice, this list of conditions and the following
* disclaimer.
*
* - Redistributions in binary form must reproduce the above
* copyright notice, this list of conditions and the following
* disclaimer in the documentation and/or other materials
* provided with the distribution.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
* NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS
* BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN
* ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
* CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
* SOFTWARE.
*/
#define _GNU_SOURCE
#include <stdio.h>
#include <stdlib.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <unistd.h>
#include <sys/timerfd.h>
#include <sys/inotify.h>
#include <sys/sysmacros.h>
#include <poll.h>
#include <util/util.h>
#include <config.h>
static int open_cdev_internal(const char *path, dev_t cdev)
{
struct stat st;
int fd;
fd = open(path, O_RDWR | O_CLOEXEC);
if (fd == -1)
return -1;
if (fstat(fd, &st) || !S_ISCHR(st.st_mode) ||
(cdev != 0 && st.st_rdev != cdev)) {
close(fd);
return -1;
}
return fd;
}
/*
* In case the cdev was not exactly where we should be, use this more
* elaborate approach to find it. This is designed to resolve a race with
* module autoloading where udev is concurrently creately the cdev as we are
* looking for it. udev has 5 seconds to create the link or we fail.
*
* Modern userspace and kernels create the /dev/infiniband/X synchronously via
* devtmpfs before returning from the netlink query, so they should never use
* this path.
*/
static int open_cdev_robust(const char *devname_hint, dev_t cdev)
{
struct itimerspec ts = { .it_value = { .tv_sec = 5 } };
struct inotify_event buf[16];
struct pollfd fds[2];
char *devpath;
int res = -1;
int ifd;
int tfd;
/*
* This assumes that udev is being used and is creating the /dev/char/
* symlinks.
*/
if (asprintf(&devpath, "/dev/char/%u:%u", major(cdev), minor(cdev)) < 0)
return -1;
/* Use inotify to speed up the resolution time. */
ifd = inotify_init1(IN_CLOEXEC | IN_NONBLOCK);
if (ifd == -1)
goto err_mem;
if (inotify_add_watch(ifd, "/dev/char/", IN_CREATE) == -1)
goto err_inotify;
/* Timerfd is simpler than working with relative time outs */
tfd = timerfd_create(CLOCK_MONOTONIC, TFD_CLOEXEC);
if (tfd == -1)
goto err_inotify;
if (timerfd_settime(tfd, 0, &ts, NULL) == -1)
goto out_timer;
res = open_cdev_internal(devpath, cdev);
if (res != -1)
goto out_timer;
fds[0].fd = ifd;
fds[0].events = POLLIN;
fds[1].fd = tfd;
fds[1].events = POLLIN;
while (poll(fds, 2, -1) > 0) {
res = open_cdev_internal(devpath, cdev);
if (res != -1)
goto out_timer;
if (fds[0].revents) {
if (read(ifd, buf, sizeof(buf)) == -1)
goto out_timer;
}
if (fds[1].revents)
goto out_timer;
}
out_timer:
close(tfd);
err_inotify:
close(ifd);
err_mem:
free(devpath);
return res;
}
int open_cdev(const char *devname_hint, dev_t cdev)
{
char *devpath;
int fd;
if (asprintf(&devpath, RDMA_CDEV_DIR "/%s", devname_hint) < 0)
return -1;
fd = open_cdev_internal(devpath, cdev);
free(devpath);
if (fd == -1 && cdev != 0)
return open_cdev_robust(devname_hint, cdev);
return fd;
}