Blame libopeniscsiusr/session.c

Packit Service 646995
/*
Packit Service 646995
 * Copyright (C) 2017 Red Hat, Inc.
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 by
Packit Service 646995
 * the Free Software Foundation, either version 3 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,
Packit Service 646995
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit Service 646995
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
Packit Service 646995
 * GNU General Public License for more details.
Packit Service 646995
 *
Packit Service 646995
 * You should have received a copy of the GNU General Public License
Packit Service 646995
 * along with this program.  If not, see <http://www.gnu.org/licenses/>.
Packit Service 646995
 *
Packit Service 646995
 * Author: Gris Ge <fge@redhat.com>
Packit Service 646995
 */
Packit Service 646995
#ifndef _GNU_SOURCE
Packit Service 646995
#define _GNU_SOURCE			/* For NI_MAXHOST */
Packit Service 646995
#endif
Packit Service 646995
Packit Service 646995
#include <stdint.h>
Packit Service 646995
#include <stdlib.h>
Packit Service 646995
#include <stdio.h>
Packit Service 646995
#include <assert.h>
Packit Service 646995
#include <dirent.h>
Packit Service 646995
#include <limits.h>
Packit Service 646995
#include <stdint.h>
Packit Service 646995
#include <inttypes.h>
Packit Service 646995
#include <netdb.h>
Packit Service 646995
#include <string.h>
Packit Service 646995
#include <errno.h>
Packit Service 646995
Packit Service 646995
#include "libopeniscsiusr/libopeniscsiusr.h"
Packit Service 646995
#include "misc.h"
Packit Service 646995
#include "sysfs.h"
Packit Service 646995
#include "iface.h"
Packit Service 646995
Packit Service 646995
#define _ISCSI_NAME_MAX_LEN		223
Packit Service 646995
/* ^ RFC 3720:
Packit Service 646995
 *	Each iSCSI node, whether an initiator or target, MUST have an iSCSI
Packit Service 646995
 *	name.
Packit Service 646995
 *
Packit Service 646995
 *	Initiators and targets MUST support the receipt of iSCSI names of up
Packit Service 646995
 *	to the maximum length of 223 bytes.
Packit Service 646995
 */
Packit Service 646995
Packit Service 646995
#define _ISCSI_CHAP_AUTH_STR_MAX_LEN	256
Packit Service 646995
/* ^ No official document found for this value, just copy from usr/auth.h
Packit Service 646995
 */
Packit Service 646995
Packit Service 646995
struct iscsi_session {
Packit Service 646995
	uint32_t sid;
Packit Service 646995
	/* ^ It's actually a int according to Linux kernel code but
Packit Service 646995
	 * the dev_set_name() in iscsi_add_session() of scsi_transport_iscsi.c
Packit Service 646995
	 * are using %u to output this.
Packit Service 646995
	 */
Packit Service 646995
	char persistent_address[NI_MAXHOST + 1];
Packit Service 646995
	int32_t persistent_port;
Packit Service 646995
	char target_name[_ISCSI_NAME_MAX_LEN + 1];
Packit Service 646995
	char username[_ISCSI_CHAP_AUTH_STR_MAX_LEN];
Packit Service 646995
	char password[_ISCSI_CHAP_AUTH_STR_MAX_LEN];
Packit Service 646995
	char username_in[_ISCSI_CHAP_AUTH_STR_MAX_LEN];
Packit Service 646995
	char password_in[_ISCSI_CHAP_AUTH_STR_MAX_LEN];
Packit Service 646995
	int32_t recovery_tmo;
Packit Service 646995
	/* ^ It's actually a int according to Linux kernel code.
Packit Service 646995
	 */
Packit Service 646995
	int32_t lu_reset_tmo;
Packit Service 646995
	/* ^ It's actually a int according to Linux kernel code.
Packit Service 646995
	 */
Packit Service 646995
	int32_t tgt_reset_tmo;
Packit Service 646995
	/* ^ It's actually a int according to Linux kernel code.
Packit Service 646995
	 */
Packit Service 646995
	int32_t abort_tmo;
Packit Service 646995
	/* ^ It's actually a int according to Linux kernel code.
Packit Service 646995
	 */
Packit Service 646995
	int32_t tpgt;
Packit Service 646995
	/* ^ It's actually a int according to Linux kernel code.
Packit Service 646995
	 */
Packit Service 646995
	char address[NI_MAXHOST + 1];
Packit Service 646995
Packit Service 646995
	int32_t port;
Packit Service 646995
	struct iscsi_iface *iface;
Packit Service 646995
};
Packit Service 646995
Packit Service 646995
_iscsi_getter_func_gen(iscsi_session, sid, uint32_t);
Packit Service 646995
_iscsi_getter_func_gen(iscsi_session, persistent_address, const char *);
Packit Service 646995
_iscsi_getter_func_gen(iscsi_session, persistent_port, int32_t);
Packit Service 646995
_iscsi_getter_func_gen(iscsi_session, target_name, const char *);
Packit Service 646995
_iscsi_getter_func_gen(iscsi_session, username, const char *);
Packit Service 646995
_iscsi_getter_func_gen(iscsi_session, password, const char *);
Packit Service 646995
_iscsi_getter_func_gen(iscsi_session, username_in, const char *);
Packit Service 646995
_iscsi_getter_func_gen(iscsi_session, password_in, const char *);
Packit Service 646995
_iscsi_getter_func_gen(iscsi_session, recovery_tmo, int32_t);
Packit Service 646995
_iscsi_getter_func_gen(iscsi_session, lu_reset_tmo, int32_t);
Packit Service 646995
_iscsi_getter_func_gen(iscsi_session, tgt_reset_tmo, int32_t);
Packit Service 646995
_iscsi_getter_func_gen(iscsi_session, abort_tmo, int32_t);
Packit Service 646995
_iscsi_getter_func_gen(iscsi_session, tpgt, int32_t);
Packit Service 646995
_iscsi_getter_func_gen(iscsi_session, address, const char *);
Packit Service 646995
_iscsi_getter_func_gen(iscsi_session, port, int32_t);
Packit Service 646995
_iscsi_getter_func_gen(iscsi_session, iface, struct iscsi_iface *);
Packit Service 646995
Packit Service 9d3ea9
int _iscsi_session_get(struct iscsi_context *ctx, uint32_t sid,
Packit Service 9d3ea9
		      struct iscsi_session **se, bool verbose)
Packit Service 646995
{
Packit Service 646995
	int rc = LIBISCSI_OK;
Packit Service 646995
	char *sysfs_se_dir_path = NULL;
Packit Service 646995
	char *sysfs_con_dir_path = NULL;
Packit Service 646995
	uint32_t host_id = 0;
Packit Service 646995
Packit Service 646995
	assert(ctx != NULL);
Packit Service 646995
	assert(se != NULL);
Packit Service 646995
Packit Service 646995
	_debug(ctx, "Querying iSCSI session for sid %" PRIu32, sid);
Packit Service 646995
Packit Service 646995
	_good(_asprintf(&sysfs_se_dir_path, "%s/session%" PRIu32,
Packit Service 646995
			_ISCSI_SYS_SESSION_DIR, sid), rc, out);
Packit Service 646995
	_good(_asprintf(&sysfs_con_dir_path, "%s/connection%" PRIu32 ":0",
Packit Service 646995
			_ISCSI_SYS_CONNECTION_DIR, sid), rc, out);
Packit Service 646995
	/* ^ BUG(Gris Ge): ':0' here in kernel is referred as connection id.
Packit Service 646995
	 *		   but the open-iscsi assuming it's always 0, need
Packit Service 646995
	 *		   investigation.
Packit Service 646995
	 */
Packit Service 646995
Packit Service 646995
	*se = (struct iscsi_session *) calloc(1, sizeof(struct iscsi_session));
Packit Service 646995
	_alloc_null_check(ctx, *se , rc, out);
Packit Service 646995
Packit Service 646995
	if (! _file_exists(sysfs_se_dir_path)) {
Packit Service a62078
		_info(ctx, "Sysfs path '%s' does not exist",
Packit Service 646995
		      sysfs_se_dir_path);
Packit Service 646995
		rc = LIBISCSI_ERR_SESS_NOT_FOUND;
Packit Service 646995
	}
Packit Service 646995
	if (! _file_exists(sysfs_con_dir_path)) {
Packit Service a62078
		_info(ctx, "Sysfs path '%s' does not exist",
Packit Service 646995
		      sysfs_se_dir_path);
Packit Service 646995
		rc = LIBISCSI_ERR_SESS_NOT_FOUND;
Packit Service 646995
	}
Packit Service 646995
	if (rc == LIBISCSI_ERR_SESS_NOT_FOUND) {
Packit Service 9d3ea9
		/* don't complain loudly if called through iscsi_sessions_get()
Packit Service 9d3ea9
		 * the caller is not looking for a specific session,
Packit Service 9d3ea9
		 * and the list could be changing as we work through it
Packit Service 9d3ea9
		 */
Packit Service 9d3ea9
		if (verbose) {
Packit Service 9d3ea9
			_error(ctx, "Specified SID %" PRIu32 " does not exist",
Packit Service 9d3ea9
			       sid);
Packit Service 9d3ea9
		}
Packit Service 646995
		goto out;
Packit Service 646995
	}
Packit Service 646995
Packit Service 646995
	(*se)->sid = sid;
Packit Service 646995
	_good(_sysfs_prop_get_str(ctx, sysfs_se_dir_path, "targetname",
Packit Service 646995
				 (*se)->target_name,
Packit Service 646995
				 sizeof((*se)->target_name) / sizeof(char),
Packit Service 646995
				 NULL), rc, out);
Packit Service 646995
Packit Service 646995
	_good(_sysfs_prop_get_str(ctx, sysfs_se_dir_path, "username",
Packit Service 646995
				  (*se)->username,
Packit Service 646995
				  sizeof((*se)->username) / sizeof(char),
Packit Service 646995
				  ""),
Packit Service 646995
	      rc, out);
Packit Service 646995
Packit Service 646995
	_good(_sysfs_prop_get_str(ctx, sysfs_se_dir_path, "password",
Packit Service 646995
				  (*se)->password,
Packit Service 646995
				  sizeof((*se)->password) / sizeof(char),
Packit Service 646995
				  ""),
Packit Service 646995
	      rc, out);
Packit Service 646995
Packit Service 646995
	_good(_sysfs_prop_get_str(ctx, sysfs_se_dir_path, "username_in",
Packit Service 646995
				  (*se)->username_in,
Packit Service 646995
				  sizeof((*se)->username_in) / sizeof(char),
Packit Service 646995
				  ""),
Packit Service 646995
	      rc, out);
Packit Service 646995
Packit Service 646995
	_good(_sysfs_prop_get_str(ctx, sysfs_se_dir_path, "password_in",
Packit Service 646995
				  (*se)->password_in,
Packit Service 646995
				  sizeof((*se)->password_in) / sizeof(char),
Packit Service 646995
				  ""),
Packit Service 646995
	      rc, out);
Packit Service 646995
Packit Service 646995
	_good(_sysfs_prop_get_i32(ctx, sysfs_se_dir_path, "recovery_tmo",
Packit Service 646995
				  &((*se)->recovery_tmo), -1, true),
Packit Service 646995
	      rc, out);
Packit Service 646995
Packit Service 646995
	_good(_sysfs_prop_get_i32(ctx, sysfs_se_dir_path, "lu_reset_tmo",
Packit Service 646995
				  &((*se)->lu_reset_tmo), -1, true),
Packit Service 646995
	      rc, out);
Packit Service 646995
Packit Service 646995
	_good(_sysfs_prop_get_i32(ctx, sysfs_se_dir_path, "tgt_reset_tmo",
Packit Service 646995
				  &((*se)->tgt_reset_tmo), -1, true),
Packit Service 646995
	      rc, out);
Packit Service 646995
Packit Service 646995
	_good(_sysfs_prop_get_i32(ctx, sysfs_se_dir_path, "abort_tmo",
Packit Service 646995
				  &((*se)->abort_tmo), -1, true),
Packit Service 646995
	      rc, out);
Packit Service 646995
Packit Service 646995
	_good(_sysfs_prop_get_i32(ctx, sysfs_se_dir_path, "tpgt",
Packit Service 646995
				  &((*se)->tpgt), -1, true),
Packit Service 646995
	      rc, out);
Packit Service 646995
Packit Service 646995
	_good(_sysfs_prop_get_str(ctx, sysfs_con_dir_path, "persistent_address",
Packit Service 646995
				  (*se)->persistent_address,
Packit Service 646995
				  sizeof((*se)->persistent_address) /
Packit Service 646995
				  sizeof(char), ""),
Packit Service 646995
	      rc, out);
Packit Service 646995
Packit Service 646995
	_good(_sysfs_prop_get_i32(ctx, sysfs_con_dir_path, "persistent_port",
Packit Service 646995
				  &((*se)->persistent_port), -1, true),
Packit Service 646995
	      rc, out);
Packit Service 646995
Packit Service 646995
	_sysfs_prop_get_str(ctx, sysfs_con_dir_path, "address", (*se)->address,
Packit Service 646995
			    sizeof((*se)->address) / sizeof(char), "");
Packit Service 646995
Packit Service 646995
	_sysfs_prop_get_i32(ctx, sysfs_con_dir_path, "port",
Packit Service 646995
			    &((*se)->port), -1, true);
Packit Service 646995
Packit Service 646995
	if ((strcmp((*se)->address, "") != 0) &&
Packit Service 646995
	    (strcmp((*se)->persistent_address, "") == 0))
Packit Service 646995
		_strncpy((*se)->persistent_address, (*se)->address,
Packit Service 646995
			 sizeof((*se)->persistent_address) / sizeof(char));
Packit Service 646995
	else if ((strcmp((*se)->address, "") == 0) &&
Packit Service 646995
	    (strcmp((*se)->persistent_address, "") != 0))
Packit Service 646995
		_strncpy((*se)->address, (*se)->persistent_address,
Packit Service 646995
			 sizeof((*se)->address) / sizeof(char));
Packit Service 646995
Packit Service 646995
	if (((*se)->persistent_port == -1) &&
Packit Service 646995
	    ((*se)->port != -1))
Packit Service 646995
		(*se)->persistent_port = (*se)->port;
Packit Service 646995
	else if (((*se)->persistent_port != -1) &&
Packit Service 646995
		 ((*se)->port == -1))
Packit Service 646995
		(*se)->port = (*se)->persistent_port;
Packit Service 646995
Packit Service 646995
	_good(_iscsi_host_id_of_session(ctx, sid, &host_id), rc, out);
Packit Service 646995
Packit Service 646995
	/* does this need to the correct iface_kern_id for the session? */
Packit Service 646995
	_good(_iscsi_iface_get_from_sysfs(ctx, host_id, sid, NULL, &((*se)->iface)),
Packit Service 646995
	      rc, out);
Packit Service 646995
Packit Service 646995
out:
Packit Service 646995
	if (rc != LIBISCSI_OK) {
Packit Service 646995
		iscsi_session_free(*se);
Packit Service 646995
		*se = NULL;
Packit Service 646995
	}
Packit Service 646995
	free(sysfs_se_dir_path);
Packit Service 646995
	free(sysfs_con_dir_path);
Packit Service 646995
	return rc;
Packit Service 646995
}
Packit Service 646995
Packit Service 9d3ea9
int iscsi_session_get(struct iscsi_context *ctx, uint32_t sid,
Packit Service 9d3ea9
		      struct iscsi_session **se) {
Packit Service 9d3ea9
	return _iscsi_session_get(ctx, sid, se, true);
Packit Service 9d3ea9
}
Packit Service 9d3ea9
Packit Service 646995
int iscsi_sessions_get(struct iscsi_context *ctx,
Packit Service 646995
		       struct iscsi_session ***sessions,
Packit Service 646995
		       uint32_t *session_count)
Packit Service 646995
{
Packit Service 646995
	int rc = LIBISCSI_OK;
Packit Service 646995
	uint32_t i = 0;
Packit Service 1e753c
	uint32_t j = 0;
Packit Service 646995
	uint32_t *sids = NULL;
Packit Service 646995
Packit Service 646995
	assert(ctx != NULL);
Packit Service 646995
	assert(sessions != NULL);
Packit Service 646995
	assert(session_count != NULL);
Packit Service 646995
Packit Service 646995
	*sessions = NULL;
Packit Service 646995
	*session_count = 0;
Packit Service 646995
Packit Service 646995
	_good(_iscsi_sids_get(ctx, &sids, session_count), rc ,out);
Packit Service 646995
Packit Service 646995
	*sessions = calloc (*session_count, sizeof(struct iscsi_session *));
Packit Service 646995
	_alloc_null_check(ctx, *sessions, rc, out);
Packit Service 646995
Packit Service 646995
	for (i = 0; i < *session_count; ++i) {
Packit Service 646995
		_debug(ctx, "sid %" PRIu32, sids[i]);
Packit Service 9d3ea9
		rc = _iscsi_session_get(ctx, sids[i], &((*sessions)[j]), false);
Packit Service 1e753c
		if (rc == LIBISCSI_OK) {
Packit Service 1e753c
			/* if session info was successfully read from sysfs, advance the sessions pointer */
Packit Service 1e753c
			j++;
Packit Service 1e753c
		} else {
Packit Service 1e753c
			/* if not, just ignore the issue and keep trying with the next session ID,
Packit Service 1e753c
			 * there's always going to be an inherent race against session removal when collecting
Packit Service 1e753c
			 * attribute data from sysfs
Packit Service 1e753c
			 */
Packit Service 1e753c
			_debug(ctx, "Problem reading session %" PRIu32 ", skipping.", sids[i]);
Packit Service 1e753c
			rc = LIBISCSI_OK;
Packit Service 1e753c
		}
Packit Service 646995
	}
Packit Service 1e753c
	/* reset session count and sessions array length to what we were able to read from sysfs */
Packit Service 1e753c
	*session_count = j;
Packit Service 1e753c
	*sessions = reallocarray(*sessions, *session_count, sizeof(struct iscsi_session *));
Packit Service 646995
Packit Service 646995
out:
Packit Service 646995
	free(sids);
Packit Service 646995
	if (rc != LIBISCSI_OK) {
Packit Service 646995
		iscsi_sessions_free(*sessions, *session_count);
Packit Service 646995
		*sessions = NULL;
Packit Service 646995
		*session_count = 0;
Packit Service 646995
	}
Packit Service 646995
	return rc;
Packit Service 646995
}
Packit Service 646995
Packit Service 646995
void iscsi_session_free(struct iscsi_session *se)
Packit Service 646995
{
Packit Service 646995
	if (se != NULL)
Packit Service 646995
		iscsi_iface_free(se->iface);
Packit Service 646995
	free(se);
Packit Service 646995
}
Packit Service 646995
Packit Service 646995
void iscsi_sessions_free(struct iscsi_session **ses, uint32_t se_count)
Packit Service 646995
{
Packit Service 646995
	uint32_t i = 0;
Packit Service 646995
Packit Service 646995
	if ((ses == NULL) || (se_count == 0))
Packit Service 646995
		return;
Packit Service 646995
Packit Service 646995
	for (i = 0; i < se_count; ++i)
Packit Service 646995
		iscsi_session_free(ses[i]);
Packit Service 646995
	free (ses);
Packit Service 646995
}