summaryrefslogtreecommitdiffstats
path: root/src/proto
diff options
context:
space:
mode:
authorMichael Brown2006-05-13 13:45:49 +0200
committerMichael Brown2006-05-13 13:45:49 +0200
commit21493646c2a6be4673439e46ffd1015a58ee6347 (patch)
tree81130af48f90d4d1300ca967d152ee3eca5ea088 /src/proto
parentProvide access to rm_cs and rm_ds from protected-mode code. (diff)
downloadipxe-21493646c2a6be4673439e46ffd1015a58ee6347.tar.gz
ipxe-21493646c2a6be4673439e46ffd1015a58ee6347.tar.xz
ipxe-21493646c2a6be4673439e46ffd1015a58ee6347.zip
Moved to net/tcp/iscsi.c.
Diffstat (limited to 'src/proto')
-rw-r--r--src/proto/iscsi.c545
1 files changed, 0 insertions, 545 deletions
diff --git a/src/proto/iscsi.c b/src/proto/iscsi.c
deleted file mode 100644
index f4fc91a7..00000000
--- a/src/proto/iscsi.c
+++ /dev/null
@@ -1,545 +0,0 @@
-#include <stddef.h>
-#include <string.h>
-#include <vsprintf.h>
-#include <assert.h>
-#include <byteswap.h>
-#include <gpxe/iscsi.h>
-
-/** @file
- *
- * iSCSI protocol
- *
- */
-
-/****************************************************************************
- *
- * Utility functions
- *
- */
-
-/**
- * Start up a new TX PDU
- *
- * @v iscsi iSCSI session
- *
- * This initiates the process of sending a new PDU. Only one PDU may
- * be in transit at any one time.
- */
-static void iscsi_start_tx ( struct iscsi_session *iscsi ) {
- assert ( iscsi->tx_state == ISCSI_TX_IDLE );
- iscsi->tx_state = ISCSI_TX_BHS;
- iscsi->tx_offset = 0;
-}
-
-/**
- * Mark session as failed
- *
- * @v iscsi iSCSI session
- *
- * This marks the session as permanently failed. The session will not
- * be automatically logged back in.
- */
-static void iscsi_fail ( struct iscsi_session *iscsi ) {
- iscsi->state = ISCSI_STATE_FAILED;
- tcp_close ( &iscsi->tcp );
-}
-
-/****************************************************************************
- *
- * iSCSI SCSI command issuing
- *
- */
-
-/**
- * Start up a block read
- *
- * @v iscsi iSCSI session
- *
- */
-static void iscsi_start_read_block ( struct iscsi_session *iscsi ) {
- struct iscsi_bhs_scsi_command *command = &iscsi->tx_bhs.scsi_command;
- struct scsi_cdb_read_10 *read = &command->cdb.read_10;
-
- assert ( iscsi->block_size != 0 );
- assert ( iscsi->block_count != 0 );
- assert ( iscsi->block_read_callback != NULL );
-
- /* Construct BHS */
- memset ( command, 0, sizeof ( *command ) );
- command->opcode = ISCSI_OPCODE_SCSI_COMMAND;
- command->flags = ( ISCSI_FLAG_FINAL |
- ISCSI_COMMAND_FLAG_READ |
- ISCSI_COMMAND_ATTR_SIMPLE );
- /* lengths left as zero */
- /* lun left as zero, on the assumption that no-one uses LUNs > 0 */
- command->itt = htonl ( iscsi->itt );
- command->exp_len = htonl ( iscsi->block_count * iscsi->block_size );
- command->cmdsn = htonl ( iscsi->cmdsn );
- command->expstatsn = htonl ( iscsi->statsn + 1 );
- read->opcode = SCSI_OPCODE_READ_10;
- read->lba = htonl ( iscsi->block_start );
- read->len = htons ( iscsi->block_count );
-
- iscsi->state = ISCSI_STATE_READING_DATA;
- iscsi_start_tx ( iscsi );
-}
-
-/**
- * Receive data segment of an iSCSI data-in PDU
- *
- * @v iscsi iSCSI session
- * @v data Received data
- * @v len Length of received data
- * @v remaining Data remaining after this data
- *
- */
-static void iscsi_rx_data_in ( struct iscsi_session *iscsi, void *data,
- size_t len, size_t remaining ) {
- struct iscsi_bhs_data_in *data_in = &iscsi->rx_bhs.data_in;
- unsigned long offset;
-
- /* Update cmdsn and statsn */
- iscsi->cmdsn = ntohl ( data_in->expcmdsn );
- iscsi->statsn = ntohl ( data_in->statsn );
-
- /* Process data via callback */
- offset = ntohl ( data_in->offset ) + iscsi->rx_offset;
- iscsi->block_read_callback ( iscsi->block_read_private,
- data, offset, len );
-
- /* If this is the end, mark state as idle */
- if ( ( data_in->flags & ISCSI_FLAG_FINAL ) && ( remaining == 0 ) )
- iscsi->state = ISCSI_STATE_IDLE;
-}
-
-/****************************************************************************
- *
- * iSCSI login
- *
- */
-
-/**
- * Build iSCSI login request strings
- *
- * @v iscsi iSCSI session
- *
- * These are the initial set of strings sent in the first login
- * request PDU.
- */
-static int iscsi_build_login_request_strings ( struct iscsi_session *iscsi,
- void *data, size_t len ) {
- return snprintf ( data, len,
- "InitiatorName=%s:initiator%c"
- "TargetName=%s%c"
- "MaxRecvDataSegmentLength=512%c"
- "SessionType=Normal%c"
- "DataDigest=None%c"
- "HeaderDigest=None%c",
- iscsi->initiator, 0, iscsi->target, 0,
- 0, 0, 0, 0 );
-}
-
-/**
- * Transmit data segment of an iSCSI login request PDU
- *
- * @v iscsi iSCSI session
- *
- * For login requests, the data segment consists of the login strings.
- */
-static void iscsi_tx_login_request ( struct iscsi_session *iscsi ) {
- int len;
-
- len = iscsi_build_login_request_strings ( iscsi, tcp_buffer,
- tcp_buflen );
- tcp_send ( &iscsi->tcp, tcp_buffer + iscsi->tx_offset,
- len - iscsi->tx_offset );
-}
-
-/**
- * Start up a login request
- *
- * @v iscsi iSCSI session
- *
- */
-static void iscsi_start_login ( struct iscsi_session *iscsi ) {
- struct iscsi_bhs_login_request *request = &iscsi->tx_bhs.login_request;
- int len;
-
- /* Construct login request BHS */
- memset ( request, 0, sizeof ( *request ) );
- request->opcode = ( ISCSI_OPCODE_LOGIN_REQUEST |
- ISCSI_FLAG_IMMEDIATE );
- request->flags = ( ISCSI_LOGIN_FLAG_TRANSITION |
- ISCSI_LOGIN_CSG_OPERATIONAL_NEGOTIATION |
- ISCSI_LOGIN_NSG_FULL_FEATURE_PHASE );
- /* version_max and version_min left as zero */
- len = iscsi_build_login_request_strings ( iscsi, NULL, 0 );
- ISCSI_SET_LENGTHS ( request->lengths, 0, len );
- request->isid_iana_en = htonl ( ISCSI_ISID_IANA |
- IANA_EN_FEN_SYSTEMS );
- /* isid_iana_qual left as zero */
- request->tsih = htons ( iscsi->tsih );
- /* itt left as zero */
- /* cid left as zero */
- request->cmdsn = htonl ( iscsi->cmdsn );
- request->expstatsn = htonl ( iscsi->statsn + 1 );
-
- iscsi->state = ISCSI_STATE_LOGGING_IN;
- iscsi_start_tx ( iscsi );
-}
-
-/**
- * Receive data segment of an iSCSI login response PDU
- *
- * @v iscsi iSCSI session
- * @v data Received data
- * @v len Length of received data
- * @v remaining Data remaining after this data
- *
- */
-static void iscsi_rx_login_response ( struct iscsi_session *iscsi,
- void *data __unused,
- size_t len __unused,
- size_t remaining __unused ) {
- struct iscsi_bhs_login_request *request = &iscsi->tx_bhs.login_request;
- struct iscsi_bhs_login_response *response
- = &iscsi->rx_bhs.login_response;
-
- /* Sanity check */
- if ( iscsi->state != ISCSI_STATE_LOGGING_IN ) {
- printf ( "Spurious iSCSI login response\n" );
- iscsi_fail ( iscsi );
- return;
- }
-
- /* Check for fatal errors */
- if ( response->status_class != 0 ) {
- printf ( "iSCSI login failure: class %02x detail %02x\n",
- response->status_class, response->status_detail );
- iscsi_fail ( iscsi );
- return;
- }
-
- /* Update cmdsn and statsn */
- iscsi->cmdsn = ntohl ( response->expcmdsn );
- iscsi->statsn = ntohl ( response->statsn );
-
- /* If server did not transition, we send it another login
- * request with empty strings.
- */
- if ( ! ( response->flags & ISCSI_LOGIN_FLAG_TRANSITION ) ) {
- ISCSI_SET_LENGTHS ( request->lengths, 0, 0 );
- iscsi_start_tx ( iscsi );
- return;
- }
-
- /* Record TSIH for future reference */
- iscsi->tsih = ntohl ( response->tsih );
-
- /* Start reading data */
- iscsi_start_read_block ( iscsi );
-}
-
-/****************************************************************************
- *
- * iSCSI to TCP interface
- *
- */
-
-static inline struct iscsi_session *
-tcp_to_iscsi ( struct tcp_connection *conn ) {
- return container_of ( conn, struct iscsi_session, tcp );
-}
-
-static void iscsi_closed ( struct tcp_connection *conn, int status ) {
- struct iscsi_session *iscsi = tcp_to_iscsi ( conn );
-
-}
-
-static void iscsi_connected ( struct tcp_connection *conn ) {
- struct iscsi_session *iscsi = tcp_to_iscsi ( conn );
-
- /* Prepare to receive PDUs. */
- iscsi->rx_state = ISCSI_RX_BHS;
- iscsi->rx_offset = 0;
-
- /* TX state should already have been set up */
- assert ( iscsi->tx_state != ISCSI_TX_IDLE );
- assert ( iscsi->tx_offset == 0 );
-}
-
-/**
- * Transmit data segment of an iSCSI PDU
- *
- * @v iscsi iSCSI session
- *
- * Handle transmission of part of a PDU data segment. iscsi::tx_bhs
- * will be valid when this is called.
- */
-static void iscsi_tx_data ( struct iscsi_session *iscsi ) {
- struct iscsi_bhs_common *common = &iscsi->tx_bhs.common;
-
- switch ( common->opcode & ISCSI_OPCODE_MASK ) {
- case ISCSI_OPCODE_LOGIN_REQUEST:
- iscsi_tx_login_request ( iscsi );
- break;
- default:
- assert ( 0 );
- break;
- }
-}
-
-/**
- * Handle TCP ACKs
- *
- * @v iscsi iSCSI session
- *
- * Updates iscsi->tx_offset and, if applicable, transitions to the
- * next TX state.
- */
-static void iscsi_acked ( struct tcp_connection *conn, size_t len ) {
- struct iscsi_session *iscsi = tcp_to_iscsi ( conn );
- struct iscsi_bhs_common *common = &iscsi->tx_bhs.common;
- size_t max_tx_offset;
- enum iscsi_tx_state next_state;
-
- iscsi->tx_offset += len;
- while ( 1 ) {
- switch ( iscsi->tx_state ) {
- case ISCSI_TX_BHS:
- max_tx_offset = sizeof ( iscsi->tx_bhs );
- next_state = ISCSI_TX_AHS;
- break;
- case ISCSI_TX_AHS:
- max_tx_offset = 4 * ISCSI_AHS_LEN ( common->lengths );
- next_state = ISCSI_TX_DATA;
- break;
- case ISCSI_TX_DATA:
- max_tx_offset = ISCSI_DATA_LEN ( common->lengths );
- next_state = ISCSI_TX_DATA_PADDING;
- break;
- case ISCSI_TX_DATA_PADDING:
- max_tx_offset = ISCSI_DATA_PAD_LEN ( common->lengths );
- next_state = ISCSI_TX_IDLE;
- break;
- case ISCSI_TX_IDLE:
- return;
- default:
- assert ( 0 );
- return;
- }
- assert ( iscsi->tx_offset <= max_tx_offset );
-
- /* If the whole of the current portion has not yet
- * been acked, stay in this state for now.
- */
- if ( iscsi->tx_offset != max_tx_offset )
- return;
-
- iscsi->tx_state = next_state;
- iscsi->tx_offset = 0;
- }
-}
-
-/**
- * Transmit iSCSI PDU
- *
- * @v iscsi iSCSI session
- *
- * Constructs data to be sent for the current TX state
- */
-static void iscsi_senddata ( struct tcp_connection *conn ) {
- struct iscsi_session *iscsi = tcp_to_iscsi ( conn );
- struct iscsi_bhs_common *common = &iscsi->tx_bhs.common;
- static const char pad[] = { '\0', '\0', '\0' };
-
- switch ( iscsi->tx_state ) {
- case ISCSI_TX_IDLE:
- /* Do nothing */
- break;
- case ISCSI_TX_BHS:
- tcp_send ( conn, &iscsi->tx_bhs.bytes[iscsi->tx_offset],
- ( sizeof ( iscsi->tx_bhs ) - iscsi->tx_offset ) );
- break;
- case ISCSI_TX_AHS:
- /* We don't yet have an AHS transmission mechanism */
- assert ( 0 );
- break;
- case ISCSI_TX_DATA:
- iscsi_tx_data ( iscsi );
- break;
- case ISCSI_TX_DATA_PADDING:
- tcp_send ( conn, pad, ( ISCSI_DATA_PAD_LEN ( common->lengths )
- - iscsi->tx_offset ) );
- break;
- default:
- assert ( 0 );
- break;
- }
-}
-
-/**
- * Receive data segment of an iSCSI PDU
- *
- * @v iscsi iSCSI session
- * @v data Received data
- * @v len Length of received data
- * @v remaining Data remaining after this data
- *
- * Handle processing of part of a PDU data segment. iscsi::rx_bhs
- * will be valid when this is called.
- */
-static void iscsi_rx_data ( struct iscsi_session *iscsi, void *data,
- size_t len, size_t remaining ) {
- struct iscsi_bhs_common *common = &iscsi->rx_bhs.common;
-
- switch ( common->opcode & ISCSI_OPCODE_MASK ) {
- case ISCSI_OPCODE_LOGIN_RESPONSE:
- iscsi_rx_login_response ( iscsi, data, len, remaining );
- break;
- case ISCSI_OPCODE_DATA_IN:
- iscsi_rx_data_in ( iscsi, data, len, remaining );
- break;
- default:
- printf ( "Unknown iSCSI opcode %02x\n", common->opcode );
- break;
- }
-}
-
-/**
- * Discard portion of an iSCSI PDU.
- *
- * @v iscsi iSCSI session
- * @v data Received data
- * @v len Length of received data
- * @v remaining Data remaining after this data
- *
- * This discards data from a portion of a received PDU.
- */
-static void iscsi_rx_discard ( struct iscsi_session *iscsi __unused,
- void *data __unused, size_t len __unused,
- size_t remaining __unused ) {
- /* Do nothing */
-}
-
-/**
- * Receive basic header segment of an iSCSI PDU
- *
- * @v iscsi iSCSI session
- * @v data Received data
- * @v len Length of received data
- * @v remaining Data remaining after this data
- *
- * This fills in iscsi::rx_bhs with the data from the BHS portion of
- * the received PDU.
- */
-static void iscsi_rx_bhs ( struct iscsi_session *iscsi, void *data,
- size_t len, size_t remaining __unused ) {
- memcpy ( &iscsi->rx_bhs.bytes[iscsi->rx_offset], data, len );
-}
-
-/**
- * Receive new data
- *
- * @v tcp TCP connection
- * @v data Received data
- * @v len Length of received data
- *
- * This handles received PDUs. The receive strategy is to fill in
- * iscsi::rx_bhs with the contents of the BHS portion of the PDU,
- * throw away any AHS portion, and then process each part of the data
- * portion as it arrives. The data processing routine therefore
- * always has a full copy of the BHS available, even for portions of
- * the data in different packets to the BHS.
- */
-static void iscsi_newdata ( struct tcp_connection *conn, void *data,
- size_t len ) {
- struct iscsi_session *iscsi = tcp_to_iscsi ( conn );
- struct iscsi_bhs_common *common = &iscsi->rx_bhs.common;
- void ( *process ) ( struct iscsi_session *iscsi, void *data,
- size_t len, size_t remaining );
- size_t max_rx_offset;
- enum iscsi_rx_state next_state;
- size_t frag_len;
- size_t remaining;
-
- while ( 1 ) {
- switch ( iscsi->rx_state ) {
- case ISCSI_RX_BHS:
- process = iscsi_rx_bhs;
- max_rx_offset = sizeof ( iscsi->rx_bhs );
- next_state = ISCSI_RX_AHS;
- break;
- case ISCSI_RX_AHS:
- process = iscsi_rx_discard;
- max_rx_offset = 4 * ISCSI_AHS_LEN ( common->lengths );
- next_state = ISCSI_RX_DATA;
- break;
- case ISCSI_RX_DATA:
- process = iscsi_rx_data;
- max_rx_offset = ISCSI_DATA_LEN ( common->lengths );
- next_state = ISCSI_RX_DATA_PADDING;
- break;
- case ISCSI_RX_DATA_PADDING:
- process = iscsi_rx_discard;
- max_rx_offset = ISCSI_DATA_PAD_LEN ( common->lengths );
- next_state = ISCSI_RX_BHS;
- break;
- default:
- assert ( 0 );
- return;
- }
-
- frag_len = max_rx_offset - iscsi->rx_offset;
- if ( frag_len > len )
- frag_len = len;
- remaining = max_rx_offset - iscsi->rx_offset - frag_len;
- process ( iscsi, data, frag_len, remaining );
-
- iscsi->rx_offset += frag_len;
- data += frag_len;
- len -= frag_len;
-
- /* If all the data for this state has not yet been
- * received, stay in this state for now.
- */
- if ( iscsi->rx_offset != max_rx_offset )
- return;
-
- iscsi->rx_state = next_state;
- iscsi->rx_offset = 0;
- }
-}
-
-/** iSCSI TCP operations */
-static struct tcp_operations iscsi_tcp_operations = {
- .closed = iscsi_closed,
- .connected = iscsi_connected,
- .acked = iscsi_acked,
- .newdata = iscsi_newdata,
- .senddata = iscsi_senddata,
-};
-
-/**
- * Wake up session
- *
- * @v iscsi iSCSI session
- *
- */
-void iscsi_wakeup ( struct iscsi_session *iscsi ) {
- iscsi->tcp.tcp_op = &iscsi_tcp_operations;
-
- switch ( iscsi->state ) {
- case ISCSI_STATE_NOT_CONNECTED:
- case ISCSI_STATE_FAILED:
- tcp_connect ( &iscsi->tcp );
- iscsi_start_login ( iscsi );
- break;
- case ISCSI_STATE_IDLE:
- iscsi_start_read_block ( iscsi );
- break;
- default:
- /* Stay in same state */
- break;
- }
-}