Blame usr/host.c

Packit Service 646995
/*
Packit Service 646995
 * iSCSI host helpers
Packit Service 646995
 *
Packit Service 646995
 * Copyright (C) 2008 Mike Christie
Packit Service 646995
 * Copyright (C) 2008 Red Hat, Inc. All rights reserved.
Packit Service 646995
 * maintained by open-iscsi@@googlegroups.com
Packit Service 646995
 *
Packit Service 646995
 * This program is free software; you can redistribute it and/or modify
Packit Service 646995
 * it under the terms of the GNU General Public License as published
Packit Service 646995
 * by the Free Software Foundation; either version 2 of the License, or
Packit Service 646995
 * (at your option) any later version.
Packit Service 646995
 *
Packit Service 646995
 * This program is distributed in the hope that it will be useful, but
Packit Service 646995
 * WITHOUT ANY WARRANTY; without even the implied warranty of
Packit Service 646995
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
Packit Service 646995
 * General Public License for more details.
Packit Service 646995
 *
Packit Service 646995
 * See the file COPYING included with this distribution for more details.
Packit Service 646995
 */
Packit Service 646995
#include <stdlib.h>
Packit Service 646995
#include <stdio.h>
Packit Service 646995
#include <errno.h>
Packit Service 646995
Packit Service 646995
#include "list.h"
Packit Service 646995
#include "iscsi_util.h"
Packit Service 646995
#include "log.h"
Packit Service 646995
#include "iscsi_sysfs.h"
Packit Service 646995
#include "version.h"
Packit Service 646995
#include "iscsi_settings.h"
Packit Service 646995
#include "mgmt_ipc.h"
Packit Service 646995
#include "host.h"
Packit Service 646995
#include "session_info.h"
Packit Service 646995
#include "transport.h"
Packit Service 646995
#include "initiator.h"
Packit Service 646995
#include "iface.h"
Packit Service 646995
#include "iscsi_err.h"
Packit Service 646995
#include "iscsi_netlink.h"
Packit Service 646995
Packit Service 646995
struct _host_info_print_tree_arg {
Packit Service 646995
	unsigned int flags;
Packit Service 646995
	struct iscsi_session **ses;
Packit Service 646995
	uint32_t se_count;
Packit Service 646995
};
Packit Service 646995
Packit Service 646995
static int match_host_to_session(uint32_t host_no, struct iscsi_session *se)
Packit Service 646995
{
Packit Service 646995
	uint32_t sid = 0;
Packit Service 646995
	uint32_t info_host_no;
Packit Service 646995
	int rc;
Packit Service 646995
Packit Service 646995
	sid = iscsi_session_sid_get(se);
Packit Service 646995
Packit Service 646995
	info_host_no = iscsi_sysfs_get_host_no_from_sid(sid, &rc);
Packit Service 646995
	if (rc) {
Packit Service 646995
		log_error("could not get host_no for session%d err %d.",
Packit Service 646995
			  sid, rc);
Packit Service 646995
		return 0;
Packit Service 646995
	}
Packit Service 646995
Packit Service 646995
	return host_no == info_host_no;
Packit Service 646995
}
Packit Service 646995
Packit Service 646995
static void print_host_info(struct iface_rec *iface, char *prefix)
Packit Service 646995
{
Packit Service 646995
	if (strlen(iface->transport_name))
Packit Service 646995
		printf("%sTransport: %s\n", prefix,
Packit Service 646995
		      iface->transport_name);
Packit Service 646995
	else
Packit Service 646995
		printf("%sTransport: %s\n", prefix, UNKNOWN_VALUE);
Packit Service 646995
Packit Service 646995
	if (strlen(iface->iname))
Packit Service 646995
		printf("%sInitiatorname: %s\n", prefix,
Packit Service 646995
		      iface->iname);
Packit Service 646995
	else
Packit Service 646995
		printf("%sInitiatorname: %s\n", prefix, UNKNOWN_VALUE);
Packit Service 646995
Packit Service 646995
	if (!strlen(iface->ipaddress))
Packit Service 646995
		printf("%sIPaddress: %s\n", prefix, UNKNOWN_VALUE);
Packit Service 646995
	else if (strchr(iface->ipaddress, '.'))
Packit Service 646995
		printf("%sIPaddress: %s\n", prefix, iface->ipaddress);
Packit Service 646995
	else
Packit Service 646995
		printf("%sIPaddress: [%s]\n", prefix, iface->ipaddress);
Packit Service 646995
Packit Service 646995
	if (strlen(iface->hwaddress))
Packit Service 646995
		printf("%sHWaddress: %s\n", prefix, iface->hwaddress);
Packit Service 646995
	else
Packit Service 646995
		printf("%sHWaddress: %s\n", prefix, UNKNOWN_VALUE);
Packit Service 646995
Packit Service 646995
	if (strlen(iface->netdev))
Packit Service 646995
		printf("%sNetdev: %s\n", prefix, iface->netdev);
Packit Service 646995
	else
Packit Service 646995
		printf("%sNetdev: %s\n", prefix, UNKNOWN_VALUE);
Packit Service 646995
}
Packit Service 646995
Packit Service 646995
static int host_info_print_flat(__attribute__((unused))void *data,
Packit Service 646995
				struct host_info *hinfo)
Packit Service 646995
{
Packit Service 646995
	struct iface_rec *iface = &hinfo->iface;
Packit Service 646995
Packit Service 646995
	if (strlen(iface->transport_name))
Packit Service 646995
		printf("%s: ", iface->transport_name);
Packit Service 646995
	else
Packit Service 646995
		printf("%s: ", UNKNOWN_VALUE);
Packit Service 646995
Packit Service 646995
	printf("[%u] ", hinfo->host_no);
Packit Service 646995
Packit Service 646995
	if (!strlen(iface->ipaddress))
Packit Service 646995
		printf("%s,", UNKNOWN_VALUE);
Packit Service 646995
	else if (strchr(iface->ipaddress, '.'))
Packit Service 646995
		printf("%s,", iface->ipaddress);
Packit Service 646995
	else
Packit Service 646995
		printf("[%s],", iface->ipaddress);
Packit Service 646995
Packit Service 646995
	if (strlen(iface->hwaddress))
Packit Service 646995
		printf("[%s],", iface->hwaddress);
Packit Service 646995
	else
Packit Service 646995
		printf("[%s],", UNKNOWN_VALUE);
Packit Service 646995
Packit Service 646995
	if (strlen(iface->netdev))
Packit Service 646995
		printf("%s ", iface->netdev);
Packit Service 646995
	else
Packit Service 646995
		printf("%s ", UNKNOWN_VALUE);
Packit Service 646995
Packit Service 646995
	if (strlen(iface->iname))
Packit Service 646995
		printf("%s\n", iface->iname);
Packit Service 646995
	else
Packit Service 646995
		printf("%s\n", UNKNOWN_VALUE);
Packit Service 646995
	return 0;
Packit Service 646995
}
Packit Service 646995
Packit Service 646995
static int print_host_iface(void *data, struct iface_rec *iface)
Packit Service 646995
{
Packit Service 646995
	char *prefix = data;
Packit Service 646995
Packit Service 646995
	printf("%s**********\n", prefix);
Packit Service 646995
	printf("%sInterface:\n", prefix);
Packit Service 646995
	printf("%s**********\n", prefix);
Packit Service 646995
Packit Service 646995
	printf("%sKernel Name: %s\n", prefix, iface->name);
Packit Service 646995
Packit Service 646995
	if (!strlen(iface->ipaddress))
Packit Service 646995
		printf("%sIPaddress: %s\n", prefix, UNKNOWN_VALUE);
Packit Service 646995
	else if (strchr(iface->ipaddress, '.')) {
Packit Service 646995
		printf("%sIPaddress: %s\n", prefix, iface->ipaddress);
Packit Service 646995
Packit Service 646995
		if (!strlen(iface->gateway))
Packit Service 646995
			printf("%sGateway: %s\n", prefix, UNKNOWN_VALUE);
Packit Service 646995
		else
Packit Service 646995
			printf("%sGateway: %s\n", prefix, iface->gateway);
Packit Service 646995
		if (!strlen(iface->subnet_mask))
Packit Service 646995
			printf("%sSubnet: %s\n", prefix, UNKNOWN_VALUE);
Packit Service 646995
		else
Packit Service 646995
			printf("%sSubnet: %s\n", prefix, iface->subnet_mask);
Packit Service 646995
		if (!strlen(iface->bootproto))
Packit Service 646995
			printf("%sBootProto: %s\n", prefix, UNKNOWN_VALUE);
Packit Service 646995
		else
Packit Service 646995
			printf("%sBootProto: %s\n", prefix, iface->bootproto);
Packit Service 646995
	} else {
Packit Service 646995
		printf("%sIPaddress: [%s]\n", prefix, iface->ipaddress);
Packit Service 646995
Packit Service 646995
		if (!strlen(iface->ipv6_autocfg))
Packit Service 646995
			printf("%sIPaddress Autocfg: %s\n", prefix,
Packit Service 646995
			       UNKNOWN_VALUE);
Packit Service 646995
		else
Packit Service 646995
			printf("%sIPaddress Autocfg: %s\n", prefix,
Packit Service 646995
			       iface->ipv6_autocfg);
Packit Service 646995
		if (!strlen(iface->ipv6_linklocal))
Packit Service 646995
			printf("%sLink Local Address: %s\n", prefix,
Packit Service 646995
			       UNKNOWN_VALUE);
Packit Service 646995
		else
Packit Service 646995
			printf("%sLink Local Address: [%s]\n", prefix,
Packit Service 646995
			       iface->ipv6_linklocal);
Packit Service 646995
		if (!strlen(iface->linklocal_autocfg))
Packit Service 646995
			printf("%sLink Local Autocfg: %s\n", prefix,
Packit Service 646995
			       UNKNOWN_VALUE);
Packit Service 646995
		else
Packit Service 646995
			printf("%sLink Local Autocfg: %s\n", prefix,
Packit Service 646995
			       iface->linklocal_autocfg);
Packit Service 646995
		if (!strlen(iface->ipv6_router))
Packit Service 646995
			printf("%sRouter Address: %s\n", prefix,
Packit Service 646995
			      UNKNOWN_VALUE);
Packit Service 646995
		else
Packit Service 646995
			printf("%sRouter Address: [%s]\n", prefix,
Packit Service 646995
			       iface->ipv6_router);
Packit Service 646995
	}
Packit Service 646995
Packit Service 646995
	if (!strlen(iface->port_state))
Packit Service 646995
		printf("%sPort State: %s\n", prefix, UNKNOWN_VALUE);
Packit Service 646995
	else
Packit Service 646995
		printf("%sPort State: %s\n", prefix, iface->port_state);
Packit Service 646995
Packit Service 646995
	if (!strlen(iface->port_speed))
Packit Service 646995
		printf("%sPort Speed: %s\n", prefix, UNKNOWN_VALUE);
Packit Service 646995
	else
Packit Service 646995
		printf("%sPort Speed: %s\n", prefix, iface->port_speed);
Packit Service 646995
Packit Service 646995
	if (!iface->port)
Packit Service 646995
		printf("%sPort: %s\n", prefix, UNKNOWN_VALUE);
Packit Service 646995
	else
Packit Service 646995
		printf("%sPort: %u\n", prefix, iface->port);
Packit Service 646995
Packit Service 646995
	if (!iface->mtu)
Packit Service 646995
		printf("%sMTU: %s\n", prefix, UNKNOWN_VALUE);
Packit Service 646995
	else
Packit Service 646995
		printf("%sMTU: %u\n", prefix, iface->mtu);
Packit Service 646995
Packit Service 646995
	if (iface->vlan_id == UINT16_MAX)
Packit Service 646995
		printf("%sVLAN ID: %s\n", prefix, UNKNOWN_VALUE);
Packit Service 646995
	else
Packit Service 646995
		printf("%sVLAN ID: %u\n", prefix, iface->vlan_id);
Packit Service 646995
Packit Service 646995
	if (iface->vlan_priority == UINT8_MAX)
Packit Service 646995
		printf("%sVLAN priority: %s\n", prefix, UNKNOWN_VALUE);
Packit Service 646995
	else
Packit Service 646995
		printf("%sVLAN priority: %u\n", prefix, iface->vlan_priority);
Packit Service 646995
	return 0;
Packit Service 646995
}
Packit Service 646995
Packit Service 646995
static void print_host_ifaces(struct host_info *hinfo, char *prefix)
Packit Service 646995
{
Packit Service 646995
	int nr_found = 0;
Packit Service 646995
Packit Service 646995
	iscsi_sysfs_for_each_iface_on_host(prefix, hinfo->host_no, &nr_found,
Packit Service 646995
					   print_host_iface);
Packit Service 646995
}
Packit Service 646995
Packit Service 646995
static int host_info_print_tree(void *data, struct host_info *hinfo)
Packit Service 646995
{
Packit Service 646995
	unsigned int session_info_flags = 0;
Packit Service 646995
	struct _host_info_print_tree_arg *arg = data;
Packit Service 646995
	struct iscsi_session **ses = NULL;
Packit Service 646995
	struct iscsi_session **matched_ses = NULL;
Packit Service 646995
	uint32_t se_count = 0;
Packit Service 646995
	uint32_t matched_se_count = 0;
Packit Service 646995
	uint32_t i = 0;
Packit Service 646995
	char state[SCSI_MAX_STATE_VALUE];
Packit Service 646995
Packit Service 646995
	if (arg == NULL)
Packit Service 646995
		return -EINVAL;
Packit Service 646995
Packit Service 646995
	session_info_flags = arg->flags;
Packit Service 646995
	ses = arg->ses;
Packit Service 646995
	se_count = arg->se_count;
Packit Service 646995
Packit Service 646995
	printf("Host Number: %u\n", hinfo->host_no);
Packit Service 646995
	if (!iscsi_sysfs_get_host_state(state, hinfo->host_no))
Packit Service 646995
		printf("\tState: %s\n", state);
Packit Service 646995
	else
Packit Service 646995
		printf("\tState: Unknown\n");
Packit Service 646995
	print_host_info(&hinfo->iface, "\t");
Packit Service 646995
Packit Service 646995
	print_host_ifaces(hinfo, "\t");
Packit Service 646995
Packit Service 646995
	if ((!session_info_flags) || (!se_count))
Packit Service 646995
		return 0;
Packit Service 646995
Packit Service 646995
	matched_ses = calloc(se_count, sizeof(struct iscsi_session *));
Packit Service 646995
	if (matched_ses == NULL)
Packit Service 646995
		return -ENOMEM;
Packit Service 646995
Packit Service 646995
	for (i = 0; i < se_count; ++i)
Packit Service 646995
		if (match_host_to_session(hinfo->host_no, ses[i]))
Packit Service 646995
			matched_ses[matched_se_count++] = ses[i];
Packit Service 646995
Packit Service 646995
	if (!matched_se_count)
Packit Service 646995
		goto out;
Packit Service 646995
Packit Service 646995
	printf("\t*********\n");
Packit Service 646995
	printf("\tSessions:\n");
Packit Service 646995
	printf("\t*********\n");
Packit Service 646995
	session_info_print_tree(matched_ses, matched_se_count, "\t",
Packit Service 646995
				session_info_flags, 0/* don't show password */);
Packit Service 646995
out:
Packit Service 646995
	free(matched_ses);
Packit Service 646995
	return 0;
Packit Service 646995
}
Packit Service 646995
Packit Service 646995
int host_info_print(int info_level, uint32_t host_no,
Packit Service 646995
		    struct iscsi_session **ses, uint32_t se_count)
Packit Service 646995
Packit Service 646995
{
Packit Service 646995
	int num_found = 0, err = 0;
Packit Service 646995
	char *version;
Packit Service 646995
	unsigned int flags = 0;
Packit Service 646995
	struct _host_info_print_tree_arg arg;
Packit Service 646995
Packit Service 646995
	switch (info_level) {
Packit Service 646995
	case 0:
Packit Service 646995
	case -1:
Packit Service 646995
		err = iscsi_sysfs_for_each_host(NULL, &num_found,
Packit Service 646995
						host_info_print_flat);
Packit Service 646995
		break;
Packit Service 646995
	case 4:
Packit Service 646995
		version = iscsi_sysfs_get_iscsi_kernel_version();
Packit Service 646995
		if (version) {
Packit Service 646995
			printf("iSCSI Transport Class version %s\n",
Packit Service 646995
			       version);
Packit Service 646995
			printf("version %s\n", ISCSI_VERSION_STR);
Packit Service 646995
			free(version);
Packit Service 646995
		}
Packit Service 646995
Packit Service 646995
		flags |= SESSION_INFO_SCSI_DEVS;
Packit Service 646995
		/* fall through */
Packit Service 646995
	case 3:
Packit Service 646995
		flags |= SESSION_INFO_ISCSI_PARAMS;
Packit Service 646995
		/* fall through */
Packit Service 646995
	case 2:
Packit Service 646995
		flags |= SESSION_INFO_ISCSI_STATE | SESSION_INFO_IFACE;
Packit Service 646995
		/* fall through */
Packit Service 646995
	case 1:
Packit Service 646995
		arg.flags = flags;
Packit Service 646995
		arg.ses = ses;
Packit Service 646995
		arg.se_count = se_count;
Packit Service 646995
		/* set host_no if not yet done */
Packit Service 646995
		if (host_no > MAX_HOST_NO) {
Packit Service 646995
			struct host_info hinfo;
Packit Service 646995
Packit Service 646995
			memset(&hinfo, 0, sizeof(struct host_info));
Packit Service 646995
			hinfo.host_no = host_no;
Packit Service 646995
			iscsi_sysfs_get_hostinfo_by_host_no(&hinfo);
Packit Service 646995
			host_info_print_tree(&arg, &hinfo);
Packit Service 646995
			num_found = 1;
Packit Service 646995
			break;
Packit Service 646995
		}
Packit Service 646995
Packit Service 646995
		transport_probe_for_offload();
Packit Service 646995
		err = iscsi_sysfs_for_each_host(&arg, &num_found,
Packit Service 646995
						host_info_print_tree);
Packit Service 646995
		break;
Packit Service 646995
	default:
Packit Service 646995
		log_error("Invalid info level %d. Try 0 - 4.", info_level);
Packit Service 646995
		return ISCSI_ERR_INVAL;
Packit Service 646995
	}
Packit Service 646995
Packit Service 646995
	if (err) {
Packit Service 646995
		log_error("Can not get list of iSCSI hosts: %s",
Packit Service 646995
			  iscsi_err_to_str(err));
Packit Service 646995
		return err;
Packit Service 646995
	} else if (!num_found) {
Packit Service 646995
		log_error("No iSCSI hosts.");
Packit Service 646995
		return ISCSI_ERR_NO_OBJS_FOUND;
Packit Service 646995
	}
Packit Service 646995
	return 0;
Packit Service 646995
}
Packit Service 646995
Packit Service 646995
static int chap_fill_param_uint(struct iovec *iov, int param,
Packit Service 646995
				uint32_t param_val, int param_len)
Packit Service 646995
{
Packit Service 646995
	struct iscsi_param_info *param_info;
Packit Service 646995
	struct nlattr *attr;
Packit Service 646995
	int len;
Packit Service 646995
	uint8_t val8 = 0;
Packit Service 646995
	uint16_t val16 = 0;
Packit Service 646995
	uint32_t val32 = 0;
Packit Service 646995
	char *val = NULL;
Packit Service 646995
Packit Service 646995
	len = sizeof(struct iscsi_param_info) + param_len;
Packit Service 646995
	iov->iov_base = iscsi_nla_alloc(param, len);
Packit Service 646995
	if (!iov->iov_base)
Packit Service 646995
		return 1;
Packit Service 646995
Packit Service 646995
	attr = iov->iov_base;
Packit Service 646995
	iov->iov_len = NLA_ALIGN(attr->nla_len);
Packit Service 646995
Packit Service 646995
	param_info = (struct iscsi_param_info *)ISCSI_NLA_DATA(attr);
Packit Service 646995
	param_info->param = param;
Packit Service 646995
	param_info->len = param_len;
Packit Service 646995
Packit Service 646995
	switch (param_len) {
Packit Service 646995
	case 1:
Packit Service 646995
		val8 = (uint8_t)param_val;
Packit Service 646995
		val = (char *)&val;;
Packit Service 646995
		break;
Packit Service 646995
Packit Service 646995
	case 2:
Packit Service 646995
		val16 = (uint16_t)param_val;
Packit Service 646995
		val = (char *)&val16;
Packit Service 646995
		break;
Packit Service 646995
Packit Service 646995
	case 4:
Packit Service 646995
		val32 = (uint32_t)param_val;
Packit Service 646995
		val = (char *)&val32;
Packit Service 646995
		break;
Packit Service 646995
Packit Service 646995
	default:
Packit Service 646995
		goto free;
Packit Service 646995
	}
Packit Service 646995
	memcpy(param_info->value, val, param_len);
Packit Service 646995
Packit Service 646995
	return 0;
Packit Service 646995
Packit Service 646995
free:
Packit Service 646995
	free(iov->iov_base);
Packit Service 646995
	iov->iov_base = NULL;
Packit Service 646995
	iov->iov_len = 0;
Packit Service 646995
	return 1;
Packit Service 646995
}
Packit Service 646995
Packit Service 646995
static int chap_fill_param_str(struct iovec *iov, int param, char *param_val,
Packit Service 646995
			       int param_len)
Packit Service 646995
{
Packit Service 646995
	struct iscsi_param_info *param_info;
Packit Service 646995
	struct nlattr *attr;
Packit Service 646995
	int len;
Packit Service 646995
Packit Service 646995
	len = sizeof(struct iscsi_param_info) + param_len;
Packit Service 646995
	iov->iov_base = iscsi_nla_alloc(param, len);
Packit Service 646995
	if (!iov->iov_base)
Packit Service 646995
		return 1;
Packit Service 646995
Packit Service 646995
	attr = iov->iov_base;
Packit Service 646995
	iov->iov_len = NLA_ALIGN(attr->nla_len);
Packit Service 646995
Packit Service 646995
	param_info = (struct iscsi_param_info *)ISCSI_NLA_DATA(attr);
Packit Service 646995
	param_info->param = param;
Packit Service 646995
	param_info->len = param_len;
Packit Service 646995
	memcpy(param_info->value, param_val, param_len);
Packit Service 646995
	return 0;
Packit Service 646995
}
Packit Service 646995
Packit Service 646995
int chap_build_config(struct iscsi_chap_rec *crec, struct iovec *iovs)
Packit Service 646995
{
Packit Service 646995
	struct iovec *iov = NULL;
Packit Service 646995
	int count = 0;
Packit Service 646995
Packit Service 646995
	/* start at 2, because 0 is for nlmsghdr and 1 for event */
Packit Service 646995
	iov = iovs + 2;
Packit Service 646995
Packit Service 646995
	if (!chap_fill_param_uint(&iov[count], ISCSI_CHAP_PARAM_INDEX,
Packit Service 646995
				  crec->chap_tbl_idx,
Packit Service 646995
				  sizeof(crec->chap_tbl_idx)))
Packit Service 646995
		count++;
Packit Service 646995
Packit Service 646995
	if (!chap_fill_param_uint(&iov[count], ISCSI_CHAP_PARAM_CHAP_TYPE,
Packit Service 646995
				  crec->chap_type, sizeof(crec->chap_type)))
Packit Service 646995
		count++;
Packit Service 646995
Packit Service 646995
	if (!chap_fill_param_str(&iov[count], ISCSI_CHAP_PARAM_USERNAME,
Packit Service 646995
				 crec->username, strlen(crec->username)))
Packit Service 646995
		count++;
Packit Service 646995
Packit Service 646995
	if (!chap_fill_param_str(&iov[count], ISCSI_CHAP_PARAM_PASSWORD,
Packit Service 646995
				 (char *)crec->password,
Packit Service 646995
				 strlen((char *)crec->password)))
Packit Service 646995
		count++;
Packit Service 646995
Packit Service 646995
	if (!chap_fill_param_uint(&iov[count], ISCSI_CHAP_PARAM_PASSWORD_LEN,
Packit Service 646995
				  crec->password_length,
Packit Service 646995
				  sizeof(crec->password_length)))
Packit Service 646995
		count++;
Packit Service 646995
Packit Service 646995
	return count;
Packit Service 646995
}