/* * Event notification code. * * Copyright (C) 2005 FUJITA Tomonori * Copyright (C) 2007 - 2008 Vladislav Bolkhovitin * Copyright (C) 2007 - 2008 CMS Distribution Limited * * This program is free software; you can redistribute it and/or * modify it under the terms of the GNU General Public License * as published by the Free Software Foundation. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * Some functions are based on open-iscsi code * written by Dmitry Yusupov, Alex Aizman. */ #include #include #include #include #include #include #include #include #include "iscsid.h" static struct sockaddr_nl src_addr, dest_addr; static int nl_write(int fd, void *data, int len) { struct iovec iov[2]; struct msghdr msg; struct nlmsghdr nlh; iov[0].iov_base = &nlh; iov[0].iov_len = sizeof(nlh); iov[1].iov_base = data; iov[1].iov_len = NLMSG_SPACE(len) - sizeof(nlh); nlh.nlmsg_len = NLMSG_SPACE(len); nlh.nlmsg_pid = getpid(); nlh.nlmsg_flags = 0; nlh.nlmsg_type = 0; memset(&msg, 0, sizeof(msg)); msg.msg_name= (void *)&dest_addr; msg.msg_namelen = sizeof(dest_addr); msg.msg_iov = iov; msg.msg_iovlen = 2; return sendmsg(fd, &msg, 0); } static int nl_read(int fd, void *data, int len) { struct iovec iov[2]; struct msghdr msg; struct nlmsghdr nlh; iov[0].iov_base = &nlh; iov[0].iov_len = sizeof(nlh); iov[1].iov_base = data; iov[1].iov_len = len; memset(&msg, 0, sizeof(msg)); msg.msg_name= (void *)&src_addr; msg.msg_namelen = sizeof(src_addr); msg.msg_iov = iov; msg.msg_iovlen = 2; return recvmsg(fd, &msg, MSG_DONTWAIT); } void handle_iscsi_events(int fd) { struct session *session; struct connection *conn; struct iscsi_kern_event event; int res; retry: if ((res = nl_read(fd, &event, sizeof(event))) < 0) { if (errno == EAGAIN) return; if (errno == EINTR) goto retry; log_error("read netlink fd (%d)", errno); exit(1); } log_debug(1, "conn %u session %#" PRIx64 " target %u, state %u", event.cid, event.sid, event.tid, event.state); switch (event.state) { case E_CONN_CLOSE: session = session_find_id(event.tid, event.sid); if (session == NULL) { log_error("Session %#" PRIx64 " not found", event.sid); goto retry; } conn = conn_find(session, event.cid); if (conn == NULL) { log_error("Connection %x for session %#" PRIx64 " not " "found", event.cid, event.sid); goto retry; } conn_free(conn); if (list_empty(&session->conn_list)) session_free(session); break; default: log_warning("%s(%d) %u\n", __FUNCTION__, __LINE__, event.state); exit(-1); break; } } int nl_open(void) { int nl_fd, res; nl_fd = socket(PF_NETLINK, SOCK_RAW, NETLINK_ISCSI_SCST); if (nl_fd == -1) { log_error("%s %d\n", __FUNCTION__, errno); return -1; } memset(&src_addr, 0, sizeof(src_addr)); src_addr.nl_family = AF_NETLINK; src_addr.nl_pid = getpid(); src_addr.nl_groups = 0; /* not in mcast groups */ memset(&dest_addr, 0, sizeof(dest_addr)); dest_addr.nl_family = AF_NETLINK; dest_addr.nl_pid = 0; /* kernel */ dest_addr.nl_groups = 0; /* unicast */ res = nl_write(nl_fd, NULL, 0); if (res < 0) { log_error("%s %d\n", __FUNCTION__, res); return res; } return nl_fd; }