/******************************************************************************* * This file contains the iSCSI Target DataIN value generation functions. * * (c) Copyright 2007-2013 Datera, Inc. * * Author: Nicholas A. Bellinger <nab@linux-iscsi.org> * * 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; either version 2 of the License, or * (at your option) any later version. * * 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. ******************************************************************************/ #include <scsi/iscsi_proto.h> #include "iscsi_target_core.h" #include "iscsi_target_seq_pdu_list.h" #include "iscsi_target_erl1.h" #include "iscsi_target_util.h" #include "iscsi_target.h" #include "iscsi_target_datain_values.h" struct iscsi_datain_req *iscsit_allocate_datain_req(void) { struct iscsi_datain_req *dr; dr = kmem_cache_zalloc(lio_dr_cache, GFP_ATOMIC); if (!dr) { pr_err("Unable to allocate memory for" " struct iscsi_datain_req\n"); return NULL; } INIT_LIST_HEAD(&dr->cmd_datain_node); return dr; } void iscsit_attach_datain_req(struct iscsi_cmd *cmd, struct iscsi_datain_req *dr) { spin_lock(&cmd->datain_lock); list_add_tail(&dr->cmd_datain_node, &cmd->datain_list); spin_unlock(&cmd->datain_lock); } void iscsit_free_datain_req(struct iscsi_cmd *cmd, struct iscsi_datain_req *dr) { spin_lock(&cmd->datain_lock); list_del(&dr->cmd_datain_node); spin_unlock(&cmd->datain_lock); kmem_cache_free(lio_dr_cache, dr); } void iscsit_free_all_datain_reqs(struct iscsi_cmd *cmd) { struct iscsi_datain_req *dr, *dr_tmp; spin_lock(&cmd->datain_lock); list_for_each_entry_safe(dr, dr_tmp, &cmd->datain_list, cmd_datain_node) { list_del(&dr->cmd_datain_node); kmem_cache_free(lio_dr_cache, dr); } spin_unlock(&cmd->datain_lock); } struct iscsi_datain_req *iscsit_get_datain_req(struct iscsi_cmd *cmd) { if (list_empty(&cmd->datain_list)) { pr_err("cmd->datain_list is empty for ITT:" " 0x%08x\n", cmd->init_task_tag); return NULL; } return list_first_entry(&cmd->datain_list, struct iscsi_datain_req, cmd_datain_node); } /* * For Normal and Recovery DataSequenceInOrder=Yes and DataPDUInOrder=Yes. */ static struct iscsi_datain_req *iscsit_set_datain_values_yes_and_yes( struct iscsi_cmd *cmd, struct iscsi_datain *datain) { u32 next_burst_len, read_data_done, read_data_left; struct iscsi_conn *conn = cmd->conn; struct iscsi_datain_req *dr; dr = iscsit_get_datain_req(cmd); if (!dr) return NULL; if (dr->recovery && dr->generate_recovery_values) { if (iscsit_create_recovery_datain_values_datasequenceinorder_yes( cmd, dr) < 0) return NULL; dr->generate_recovery_values = 0; } next_burst_len = (!dr->recovery) ? cmd->next_burst_len : dr->next_burst_len; read_data_done = (!dr->recovery) ? cmd->read_data_done : dr->read_data_done; read_data_left = (cmd->se_cmd.data_length - read_data_done); if (!read_data_left) { pr_err("ITT: 0x%08x read_data_left is zero!\n", cmd->init_task_tag); return NULL; } if ((read_data_left <= conn->conn_ops->MaxRecvDataSegmentLength) && (read_data_left <= (conn->sess->sess_ops->MaxBurstLength - next_burst_len))) { datain->length = read_data_left; datain->flags |= (ISCSI_FLAG_CMD_FINAL | ISCSI_FLAG_DATA_STATUS); if (conn->sess->sess_ops->ErrorRecoveryLevel > 0) datain->flags |= ISCSI_FLAG_DATA_ACK; } else { if ((next_burst_len + conn->conn_ops->MaxRecvDataSegmentLength) < conn->sess->sess_ops->MaxBurstLength) { datain->length = conn->conn_ops->MaxRecvDataSegmentLength; next_burst_len += datain->length; } else { datain->length = (conn->sess->sess_ops->MaxBurstLength - next_burst_len); next_burst_len = 0; datain->flags |= ISCSI_FLAG_CMD_FINAL; if (conn->sess->sess_ops->ErrorRecoveryLevel > 0) datain->flags |= ISCSI_FLAG_DATA_ACK; } } datain->data_sn = (!dr->recovery) ? cmd->data_sn++ : dr->data_sn++; datain->offset = read_data_done; if (!dr->recovery) { cmd->next_burst_len = next_burst_len; cmd->read_data_done += datain->length; } else { dr->next_burst_len = next_burst_len; dr->read_data_done += datain->length; } if (!dr->recovery) { if (datain->flags & ISCSI_FLAG_DATA_STATUS) dr->dr_complete = DATAIN_COMPLETE_NORMAL; return dr; } if (!dr->runlength) { if (datain->flags & ISCSI_FLAG_DATA_STATUS) { dr->dr_complete = (dr->recovery == DATAIN_WITHIN_COMMAND_RECOVERY) ? DATAIN_COMPLETE_WITHIN_COMMAND_RECOVERY : DATAIN_COMPLETE_CONNECTION_RECOVERY; } } else { if ((dr->begrun + dr->runlength) == dr->data_sn) { dr->dr_complete = (dr->recovery == DATAIN_WITHIN_COMMAND_RECOVERY) ? DATAIN_COMPLETE_WITHIN_COMMAND_RECOVERY : DATAIN_COMPLETE_CONNECTION_RECOVERY; } } return dr; } /* * For Normal and Recovery DataSequenceInOrder=No and DataPDUInOrder=Yes. */ static struct iscsi_datain_req *iscsit_set_datain_values_no_and_yes( struct iscsi_cmd *cmd, struct iscsi_datain *datain) { u32 offset, read_data_done, read_data_left, seq_send_order; struct iscsi_conn *conn = cmd->conn; struct iscsi_datain_req *dr; struct iscsi_seq *seq; dr = iscsit_get_datain_req(cmd); if (!dr) return NULL; if (dr->recovery && dr->generate_recovery_values) { if (iscsit_create_recovery_datain_values_datasequenceinorder_no( cmd, dr) < 0) return NULL; dr->generate_recovery_values = 0; } read_data_done = (!dr->recovery) ? cmd->read_data_done : dr->read_data_done; seq_send_order = (!dr->recovery) ? cmd->seq_send_order : dr->seq_send_order; read_data_left = (cmd->se_cmd.data_length - read_data_done); if (!read_data_left) { pr_err("ITT: 0x%08x read_data_left is zero!\n", cmd->init_task_tag); return NULL; } seq = iscsit_get_seq_holder_for_datain(cmd, seq_send_order); if (!seq) return NULL; seq->sent = 1; if (!dr->recovery && !seq->next_burst_len) seq->first_datasn = cmd->data_sn; offset = (seq->offset + seq->next_burst_len); if ((offset + conn->conn_ops->MaxRecvDataSegmentLength) >= cmd->se_cmd.data_length) { datain->length = (cmd->se_cmd.data_length - offset); datain->offset = offset; datain->flags |= ISCSI_FLAG_CMD_FINAL; if (conn->sess->sess_ops->ErrorRecoveryLevel > 0) datain->flags |= ISCSI_FLAG_DATA_ACK; seq->next_burst_len = 0; seq_send_order++; } else { if ((seq->next_burst_len + conn->conn_ops->MaxRecvDataSegmentLength) < conn->sess->sess_ops->MaxBurstLength) { datain->length = conn->conn_ops->MaxRecvDataSegmentLength; datain->offset = (seq->offset + seq->next_burst_len); seq->next_burst_len += datain->length; } else { datain->length = (conn->sess->sess_ops->MaxBurstLength - seq->next_burst_len); datain->offset = (seq->offset + seq->next_burst_len); datain->flags |= ISCSI_FLAG_CMD_FINAL; if (conn->sess->sess_ops->ErrorRecoveryLevel > 0) datain->flags |= ISCSI_FLAG_DATA_ACK; seq->next_burst_len = 0; seq_send_order++; } } if ((read_data_done + datain->length) == cmd->se_cmd.data_length) datain->flags |= ISCSI_FLAG_DATA_STATUS; datain->data_sn = (!dr->recovery) ? cmd->data_sn++ : dr->data_sn++; if (!dr->recovery) { cmd->seq_send_order = seq_send_order; cmd->read_data_done += datain->length; } else { dr->seq_send_order = seq_send_order; dr->read_data_done += datain->length; } if (!dr->recovery) { if (datain->flags & ISCSI_FLAG_CMD_FINAL) seq->last_datasn = datain->data_sn; if (datain->flags & ISCSI_FLAG_DATA_STATUS) dr->dr_complete = DATAIN_COMPLETE_NORMAL; return dr; } if (!dr->runlength) { if (datain->flags & ISCSI_FLAG_DATA_STATUS) { dr->dr_complete = (dr->recovery == DATAIN_WITHIN_COMMAND_RECOVERY) ? DATAIN_COMPLETE_WITHIN_COMMAND_RECOVERY : DATAIN_COMPLETE_CONNECTION_RECOVERY; } } else { if ((dr->begrun + dr->runlength) == dr->data_sn) { dr->dr_complete = (dr->recovery == DATAIN_WITHIN_COMMAND_RECOVERY) ? DATAIN_COMPLETE_WITHIN_COMMAND_RECOVERY : DATAIN_COMPLETE_CONNECTION_RECOVERY; } } return dr; } /* * For Normal and Recovery DataSequenceInOrder=Yes and DataPDUInOrder=No. */ static struct iscsi_datain_req *iscsit_set_datain_values_yes_and_no( struct iscsi_cmd *cmd, struct iscsi_datain *datain) { u32 next_burst_len, read_data_done, read_data_left; struct iscsi_conn *conn = cmd->conn; struct iscsi_datain_req *dr; struct iscsi_pdu *pdu; dr = iscsit_get_datain_req(cmd); if (!dr) return NULL; if (dr->recovery && dr->generate_recovery_values) { if (iscsit_create_recovery_datain_values_datasequenceinorder_yes( cmd, dr) < 0) return NULL; dr->generate_recovery_values = 0; } next_burst_len = (!dr->recovery) ? cmd->next_burst_len : dr->next_burst_len; read_data_done = (!dr->recovery) ? cmd->read_data_done : dr->read_data_done; read_data_left = (cmd->se_cmd.data_length - read_data_done); if (!read_data_left) { pr_err("ITT: 0x%08x read_data_left is zero!\n", cmd->init_task_tag); return dr; } pdu = iscsit_get_pdu_holder_for_seq(cmd, NULL); if (!pdu) return dr; if ((read_data_done + pdu->length) == cmd->se_cmd.data_length) { pdu->flags |= (ISCSI_FLAG_CMD_FINAL | ISCSI_FLAG_DATA_STATUS); if (conn->sess->sess_ops->ErrorRecoveryLevel > 0) pdu->flags |= ISCSI_FLAG_DATA_ACK; next_burst_len = 0; } else { if ((next_burst_len + conn->conn_ops->MaxRecvDataSegmentLength) < conn->sess->sess_ops->MaxBurstLength) next_burst_len += pdu->length; else { pdu->flags |= ISCSI_FLAG_CMD_FINAL; if (conn->sess->sess_ops->ErrorRecoveryLevel > 0) pdu->flags |= ISCSI_FLAG_DATA_ACK; next_burst_len = 0; } } pdu->data_sn = (!dr->recovery) ? cmd->data_sn++ : dr->data_sn++; if (!dr->recovery) { cmd->next_burst_len = next_burst_len; cmd->read_data_done += pdu->length; } else { dr->next_burst_len = next_burst_len; dr->read_data_done += pdu->length; } datain->flags = pdu->flags; datain->length = pdu->length; datain->offset = pdu->offset; datain->data_sn = pdu->data_sn; if (!dr->recovery) { if (datain->flags & ISCSI_FLAG_DATA_STATUS) dr->dr_complete = DATAIN_COMPLETE_NORMAL; return dr; } if (!dr->runlength) { if (datain->flags & ISCSI_FLAG_DATA_STATUS) { dr->dr_complete = (dr->recovery == DATAIN_WITHIN_COMMAND_RECOVERY) ? DATAIN_COMPLETE_WITHIN_COMMAND_RECOVERY : DATAIN_COMPLETE_CONNECTION_RECOVERY; } } else { if ((dr->begrun + dr->runlength) == dr->data_sn) { dr->dr_complete = (dr->recovery == DATAIN_WITHIN_COMMAND_RECOVERY) ? DATAIN_COMPLETE_WITHIN_COMMAND_RECOVERY : DATAIN_COMPLETE_CONNECTION_RECOVERY; } } return dr; } /* * For Normal and Recovery DataSequenceInOrder=No and DataPDUInOrder=No. */ static struct iscsi_datain_req *iscsit_set_datain_values_no_and_no( struct iscsi_cmd *cmd, struct iscsi_datain *datain) { u32 read_data_done, read_data_left, seq_send_order; struct iscsi_conn *conn = cmd->conn; struct iscsi_datain_req *dr; struct iscsi_pdu *pdu; struct iscsi_seq *seq = NULL; dr = iscsit_get_datain_req(cmd); if (!dr) return NULL; if (dr->recovery && dr->generate_recovery_values) { if (iscsit_create_recovery_datain_values_datasequenceinorder_no( cmd, dr) < 0) return NULL; dr->generate_recovery_values = 0; } read_data_done = (!dr->recovery) ? cmd->read_data_done : dr->read_data_done; seq_send_order = (!dr->recovery) ? cmd->seq_send_order : dr->seq_send_order; read_data_left = (cmd->se_cmd.data_length - read_data_done); if (!read_data_left) { pr_err("ITT: 0x%08x read_data_left is zero!\n", cmd->init_task_tag); return NULL; } seq = iscsit_get_seq_holder_for_datain(cmd, seq_send_order); if (!seq) return NULL; seq->sent = 1; if (!dr->recovery && !seq->next_burst_len) seq->first_datasn = cmd->data_sn; pdu = iscsit_get_pdu_holder_for_seq(cmd, seq); if (!pdu) return NULL; if (seq->pdu_send_order == seq->pdu_count) { pdu->flags |= ISCSI_FLAG_CMD_FINAL; if (conn->sess->sess_ops->ErrorRecoveryLevel > 0) pdu->flags |= ISCSI_FLAG_DATA_ACK; seq->next_burst_len = 0; seq_send_order++; } else seq->next_burst_len += pdu->length; if ((read_data_done + pdu->length) == cmd->se_cmd.data_length) pdu->flags |= ISCSI_FLAG_DATA_STATUS; pdu->data_sn = (!dr->recovery) ? cmd->data_sn++ : dr->data_sn++; if (!dr->recovery) { cmd->seq_send_order = seq_send_order; cmd->read_data_done += pdu->length; } else { dr->seq_send_order = seq_send_order; dr->read_data_done += pdu->length; } datain->flags = pdu->flags; datain->length = pdu->length; datain->offset = pdu->offset; datain->data_sn = pdu->data_sn; if (!dr->recovery) { if (datain->flags & ISCSI_FLAG_CMD_FINAL) seq->last_datasn = datain->data_sn; if (datain->flags & ISCSI_FLAG_DATA_STATUS) dr->dr_complete = DATAIN_COMPLETE_NORMAL; return dr; } if (!dr->runlength) { if (datain->flags & ISCSI_FLAG_DATA_STATUS) { dr->dr_complete = (dr->recovery == DATAIN_WITHIN_COMMAND_RECOVERY) ? DATAIN_COMPLETE_WITHIN_COMMAND_RECOVERY : DATAIN_COMPLETE_CONNECTION_RECOVERY; } } else { if ((dr->begrun + dr->runlength) == dr->data_sn) { dr->dr_complete = (dr->recovery == DATAIN_WITHIN_COMMAND_RECOVERY) ? DATAIN_COMPLETE_WITHIN_COMMAND_RECOVERY : DATAIN_COMPLETE_CONNECTION_RECOVERY; } } return dr; } struct iscsi_datain_req *iscsit_get_datain_values( struct iscsi_cmd *cmd, struct iscsi_datain *datain) { struct iscsi_conn *conn = cmd->conn; if (conn->sess->sess_ops->DataSequenceInOrder && conn->sess->sess_ops->DataPDUInOrder) return iscsit_set_datain_values_yes_and_yes(cmd, datain); else if (!conn->sess->sess_ops->DataSequenceInOrder && conn->sess->sess_ops->DataPDUInOrder) return iscsit_set_datain_values_no_and_yes(cmd, datain); else if (conn->sess->sess_ops->DataSequenceInOrder && !conn->sess->sess_ops->DataPDUInOrder) return iscsit_set_datain_values_yes_and_no(cmd, datain); else if (!conn->sess->sess_ops->DataSequenceInOrder && !conn->sess->sess_ops->DataPDUInOrder) return iscsit_set_datain_values_no_and_no(cmd, datain); return NULL; }