2008-12-02 00:32:11 -06:00
/*
* iSCSI over TCP / IP Data - Path lib
*
* Copyright ( C ) 2008 Mike Christie
* Copyright ( C ) 2008 Red Hat , Inc . All rights reserved .
* maintained by open - iscsi @ googlegroups . com
*
* 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 .
*
* See the file COPYING included with this distribution for more details .
*/
# ifndef LIBISCSI_TCP_H
# define LIBISCSI_TCP_H
# include <scsi/libiscsi.h>
struct iscsi_tcp_conn ;
struct iscsi_segment ;
struct sk_buff ;
struct hash_desc ;
typedef int iscsi_segment_done_fn_t ( struct iscsi_tcp_conn * ,
struct iscsi_segment * ) ;
struct iscsi_segment {
unsigned char * data ;
unsigned int size ;
unsigned int copied ;
unsigned int total_size ;
unsigned int total_copied ;
struct hash_desc * hash ;
unsigned char recv_digest [ ISCSI_DIGEST_SIZE ] ;
unsigned char digest [ ISCSI_DIGEST_SIZE ] ;
unsigned int digest_len ;
struct scatterlist * sg ;
void * sg_mapped ;
unsigned int sg_offset ;
iscsi_segment_done_fn_t * done ;
} ;
/* Socket connection recieve helper */
struct iscsi_tcp_recv {
struct iscsi_hdr * hdr ;
struct iscsi_segment segment ;
/* Allocate buffer for BHS + AHS */
uint32_t hdr_buf [ 64 ] ;
/* copied and flipped values */
int datalen ;
} ;
struct iscsi_tcp_conn {
struct iscsi_conn * iscsi_conn ;
void * dd_data ;
int stop_stage ; /* conn_stop() flag: *
* stop to recover , *
* stop to terminate */
/* control data */
struct iscsi_tcp_recv in ; /* TCP receive context */
/* CRC32C (Rx) LLD should set this is they do not offload */
struct hash_desc * rx_hash ;
} ;
struct iscsi_tcp_task {
uint32_t exp_datasn ; /* expected target's R2TSN/DataSN */
int data_offset ;
struct iscsi_r2t_info * r2t ; /* in progress solict R2T */
struct iscsi_pool r2tpool ;
struct kfifo * r2tqueue ;
void * dd_data ;
} ;
enum {
ISCSI_TCP_SEGMENT_DONE , /* curr seg has been processed */
ISCSI_TCP_SKB_DONE , /* skb is out of data */
ISCSI_TCP_CONN_ERR , /* iscsi layer has fired a conn err */
ISCSI_TCP_SUSPENDED , /* conn is suspended */
} ;
extern void iscsi_tcp_hdr_recv_prep ( struct iscsi_tcp_conn * tcp_conn ) ;
extern int iscsi_tcp_recv_skb ( struct iscsi_conn * conn , struct sk_buff * skb ,
unsigned int offset , bool offloaded , int * status ) ;
extern void iscsi_tcp_cleanup_task ( struct iscsi_task * task ) ;
extern int iscsi_tcp_task_init ( struct iscsi_task * task ) ;
extern int iscsi_tcp_task_xmit ( struct iscsi_task * task ) ;
/* segment helpers */
extern int iscsi_tcp_recv_segment_is_hdr ( struct iscsi_tcp_conn * tcp_conn ) ;
2008-12-02 00:32:16 -06:00
extern int iscsi_tcp_segment_done ( struct iscsi_tcp_conn * tcp_conn ,
struct iscsi_segment * segment , int recv ,
2008-12-02 00:32:11 -06:00
unsigned copied ) ;
extern void iscsi_tcp_segment_unmap ( struct iscsi_segment * segment ) ;
extern void iscsi_segment_init_linear ( struct iscsi_segment * segment ,
void * data , size_t size ,
iscsi_segment_done_fn_t * done ,
struct hash_desc * hash ) ;
extern int
iscsi_segment_seek_sg ( struct iscsi_segment * segment ,
struct scatterlist * sg_list , unsigned int sg_count ,
unsigned int offset , size_t size ,
iscsi_segment_done_fn_t * done , struct hash_desc * hash ) ;
/* digest helpers */
extern void iscsi_tcp_dgst_header ( struct hash_desc * hash , const void * hdr ,
size_t hdrlen ,
unsigned char digest [ ISCSI_DIGEST_SIZE ] ) ;
extern struct iscsi_cls_conn *
iscsi_tcp_conn_setup ( struct iscsi_cls_session * cls_session , int dd_data_size ,
uint32_t conn_idx ) ;
extern void iscsi_tcp_conn_teardown ( struct iscsi_cls_conn * cls_conn ) ;
/* misc helpers */
extern int iscsi_tcp_r2tpool_alloc ( struct iscsi_session * session ) ;
extern void iscsi_tcp_r2tpool_free ( struct iscsi_session * session ) ;
extern void iscsi_tcp_conn_get_stats ( struct iscsi_cls_conn * cls_conn ,
struct iscsi_stats * stats ) ;
# endif /* LIBISCSI_TCP_H */