blob: 7c8ba9d7378b6dc2ef360c17e4c1d3160d8d3762 [file] [log] [blame]
Thomas Gleixnerc942fdd2019-05-27 08:55:06 +02001/* SPDX-License-Identifier: GPL-2.0-or-later */
Mike Christiea081c132008-12-02 00:32:11 -06002/*
3 * iSCSI over TCP/IP Data-Path lib
4 *
5 * Copyright (C) 2008 Mike Christie
6 * Copyright (C) 2008 Red Hat, Inc. All rights reserved.
7 * maintained by open-iscsi@googlegroups.com
Mike Christiea081c132008-12-02 00:32:11 -06008 */
9
10#ifndef LIBISCSI_TCP_H
11#define LIBISCSI_TCP_H
12
13#include <scsi/libiscsi.h>
14
15struct iscsi_tcp_conn;
16struct iscsi_segment;
17struct sk_buff;
Herbert Xu5d6ac292016-01-24 21:19:41 +080018struct ahash_request;
Mike Christiea081c132008-12-02 00:32:11 -060019
20typedef int iscsi_segment_done_fn_t(struct iscsi_tcp_conn *,
21 struct iscsi_segment *);
22
23struct iscsi_segment {
24 unsigned char *data;
25 unsigned int size;
26 unsigned int copied;
27 unsigned int total_size;
28 unsigned int total_copied;
29
Herbert Xu5d6ac292016-01-24 21:19:41 +080030 struct ahash_request *hash;
Karen Xie28568302009-01-10 19:06:07 -080031 unsigned char padbuf[ISCSI_PAD_LEN];
Mike Christiea081c132008-12-02 00:32:11 -060032 unsigned char recv_digest[ISCSI_DIGEST_SIZE];
33 unsigned char digest[ISCSI_DIGEST_SIZE];
34 unsigned int digest_len;
35
36 struct scatterlist *sg;
37 void *sg_mapped;
38 unsigned int sg_offset;
Mike Christie70c7c882011-03-17 16:22:17 -050039 bool atomic_mapped;
Mike Christiea081c132008-12-02 00:32:11 -060040
41 iscsi_segment_done_fn_t *done;
42};
43
Lucas De Marchi25985ed2011-03-30 22:57:33 -030044/* Socket connection receive helper */
Mike Christiea081c132008-12-02 00:32:11 -060045struct iscsi_tcp_recv {
46 struct iscsi_hdr *hdr;
47 struct iscsi_segment segment;
48
49 /* Allocate buffer for BHS + AHS */
50 uint32_t hdr_buf[64];
51
52 /* copied and flipped values */
53 int datalen;
54};
55
56struct iscsi_tcp_conn {
57 struct iscsi_conn *iscsi_conn;
58 void *dd_data;
59 int stop_stage; /* conn_stop() flag: *
60 * stop to recover, *
61 * stop to terminate */
62 /* control data */
63 struct iscsi_tcp_recv in; /* TCP receive context */
64 /* CRC32C (Rx) LLD should set this is they do not offload */
Herbert Xu5d6ac292016-01-24 21:19:41 +080065 struct ahash_request *rx_hash;
Mike Christiea081c132008-12-02 00:32:11 -060066};
67
68struct iscsi_tcp_task {
69 uint32_t exp_datasn; /* expected target's R2TSN/DataSN */
70 int data_offset;
71 struct iscsi_r2t_info *r2t; /* in progress solict R2T */
72 struct iscsi_pool r2tpool;
Stefani Seibold45465482009-12-21 14:37:26 -080073 struct kfifo r2tqueue;
Mike Christiea081c132008-12-02 00:32:11 -060074 void *dd_data;
Shlomo Pongratz659743b2014-02-07 00:41:38 -060075 spinlock_t pool2queue;
76 spinlock_t queue2pool;
Mike Christiea081c132008-12-02 00:32:11 -060077};
78
79enum {
80 ISCSI_TCP_SEGMENT_DONE, /* curr seg has been processed */
81 ISCSI_TCP_SKB_DONE, /* skb is out of data */
82 ISCSI_TCP_CONN_ERR, /* iscsi layer has fired a conn err */
83 ISCSI_TCP_SUSPENDED, /* conn is suspended */
84};
85
86extern void iscsi_tcp_hdr_recv_prep(struct iscsi_tcp_conn *tcp_conn);
87extern int iscsi_tcp_recv_skb(struct iscsi_conn *conn, struct sk_buff *skb,
88 unsigned int offset, bool offloaded, int *status);
89extern void iscsi_tcp_cleanup_task(struct iscsi_task *task);
90extern int iscsi_tcp_task_init(struct iscsi_task *task);
91extern int iscsi_tcp_task_xmit(struct iscsi_task *task);
92
93/* segment helpers */
94extern int iscsi_tcp_recv_segment_is_hdr(struct iscsi_tcp_conn *tcp_conn);
Mike Christie6df19a72008-12-02 00:32:16 -060095extern int iscsi_tcp_segment_done(struct iscsi_tcp_conn *tcp_conn,
96 struct iscsi_segment *segment, int recv,
Mike Christiea081c132008-12-02 00:32:11 -060097 unsigned copied);
98extern void iscsi_tcp_segment_unmap(struct iscsi_segment *segment);
99
100extern void iscsi_segment_init_linear(struct iscsi_segment *segment,
101 void *data, size_t size,
102 iscsi_segment_done_fn_t *done,
Herbert Xu5d6ac292016-01-24 21:19:41 +0800103 struct ahash_request *hash);
Mike Christiea081c132008-12-02 00:32:11 -0600104extern int
105iscsi_segment_seek_sg(struct iscsi_segment *segment,
106 struct scatterlist *sg_list, unsigned int sg_count,
107 unsigned int offset, size_t size,
Herbert Xu5d6ac292016-01-24 21:19:41 +0800108 iscsi_segment_done_fn_t *done,
109 struct ahash_request *hash);
Mike Christiea081c132008-12-02 00:32:11 -0600110
111/* digest helpers */
Herbert Xu5d6ac292016-01-24 21:19:41 +0800112extern void iscsi_tcp_dgst_header(struct ahash_request *hash, const void *hdr,
Mike Christiea081c132008-12-02 00:32:11 -0600113 size_t hdrlen,
114 unsigned char digest[ISCSI_DIGEST_SIZE]);
115extern struct iscsi_cls_conn *
116iscsi_tcp_conn_setup(struct iscsi_cls_session *cls_session, int dd_data_size,
117 uint32_t conn_idx);
118extern void iscsi_tcp_conn_teardown(struct iscsi_cls_conn *cls_conn);
119
120/* misc helpers */
121extern int iscsi_tcp_r2tpool_alloc(struct iscsi_session *session);
122extern void iscsi_tcp_r2tpool_free(struct iscsi_session *session);
Mike Christie1304be52012-01-26 21:13:10 -0600123extern int iscsi_tcp_set_max_r2t(struct iscsi_conn *conn, char *buf);
Mike Christiea081c132008-12-02 00:32:11 -0600124extern void iscsi_tcp_conn_get_stats(struct iscsi_cls_conn *cls_conn,
125 struct iscsi_stats *stats);
126#endif /* LIBISCSI_TCP_H */