diff options
author | Mike Christie <michaelc@cs.wisc.edu> | 2008-12-02 01:32:16 -0500 |
---|---|---|
committer | James Bottomley <James.Bottomley@HansenPartnership.com> | 2008-12-29 12:24:23 -0500 |
commit | 6df19a791bdd5d820cccd8c7a12679888ae62099 (patch) | |
tree | 5afebc101c362d0a41275337dced617fb06f8a89 /drivers/scsi/libiscsi_tcp.c | |
parent | ae15f80172d95f978b60d40408353943d5bc099b (diff) |
[SCSI] libiscsi_tcp: support padding offload
cxgb3i does not offload the processing of the header,
but it will always process the padding. This patch
adds a padding offload flag to detect when the LLD
supports this.
The patch also modifies the header processing so that
we do not try to read/bypass the header dugest in the
skb. cxgb3i will not include it with the header like
with other offload cards.
Signed-off-by: Mike Christie <michaelc@cs.wisc.edu>
Signed-off-by: James Bottomley <James.Bottomley@HansenPartnership.com>
Diffstat (limited to 'drivers/scsi/libiscsi_tcp.c')
-rw-r--r-- | drivers/scsi/libiscsi_tcp.c | 37 |
1 files changed, 20 insertions, 17 deletions
diff --git a/drivers/scsi/libiscsi_tcp.c b/drivers/scsi/libiscsi_tcp.c index 9df6b3436e4a..a745f91d2928 100644 --- a/drivers/scsi/libiscsi_tcp.c +++ b/drivers/scsi/libiscsi_tcp.c | |||
@@ -159,6 +159,7 @@ iscsi_tcp_segment_splice_digest(struct iscsi_segment *segment, void *digest) | |||
159 | 159 | ||
160 | /** | 160 | /** |
161 | * iscsi_tcp_segment_done - check whether the segment is complete | 161 | * iscsi_tcp_segment_done - check whether the segment is complete |
162 | * @tcp_conn: iscsi tcp connection | ||
162 | * @segment: iscsi segment to check | 163 | * @segment: iscsi segment to check |
163 | * @recv: set to one of this is called from the recv path | 164 | * @recv: set to one of this is called from the recv path |
164 | * @copied: number of bytes copied | 165 | * @copied: number of bytes copied |
@@ -172,7 +173,8 @@ iscsi_tcp_segment_splice_digest(struct iscsi_segment *segment, void *digest) | |||
172 | * | 173 | * |
173 | * This function must be re-entrant. | 174 | * This function must be re-entrant. |
174 | */ | 175 | */ |
175 | int iscsi_tcp_segment_done(struct iscsi_segment *segment, int recv, | 176 | int iscsi_tcp_segment_done(struct iscsi_tcp_conn *tcp_conn, |
177 | struct iscsi_segment *segment, int recv, | ||
176 | unsigned copied) | 178 | unsigned copied) |
177 | { | 179 | { |
178 | static unsigned char padbuf[ISCSI_PAD_LEN]; | 180 | static unsigned char padbuf[ISCSI_PAD_LEN]; |
@@ -225,13 +227,15 @@ int iscsi_tcp_segment_done(struct iscsi_segment *segment, int recv, | |||
225 | } | 227 | } |
226 | 228 | ||
227 | /* Do we need to handle padding? */ | 229 | /* Do we need to handle padding? */ |
228 | pad = iscsi_padding(segment->total_copied); | 230 | if (!(tcp_conn->iscsi_conn->session->tt->caps & CAP_PADDING_OFFLOAD)) { |
229 | if (pad != 0) { | 231 | pad = iscsi_padding(segment->total_copied); |
230 | debug_tcp("consume %d pad bytes\n", pad); | 232 | if (pad != 0) { |
231 | segment->total_size += pad; | 233 | debug_tcp("consume %d pad bytes\n", pad); |
232 | segment->size = pad; | 234 | segment->total_size += pad; |
233 | segment->data = padbuf; | 235 | segment->size = pad; |
234 | return 0; | 236 | segment->data = padbuf; |
237 | return 0; | ||
238 | } | ||
235 | } | 239 | } |
236 | 240 | ||
237 | /* | 241 | /* |
@@ -273,7 +277,7 @@ iscsi_tcp_segment_recv(struct iscsi_tcp_conn *tcp_conn, | |||
273 | { | 277 | { |
274 | unsigned int copy = 0, copied = 0; | 278 | unsigned int copy = 0, copied = 0; |
275 | 279 | ||
276 | while (!iscsi_tcp_segment_done(segment, 1, copy)) { | 280 | while (!iscsi_tcp_segment_done(tcp_conn, segment, 1, copy)) { |
277 | if (copied == len) { | 281 | if (copied == len) { |
278 | debug_tcp("iscsi_tcp_segment_recv copied %d bytes\n", | 282 | debug_tcp("iscsi_tcp_segment_recv copied %d bytes\n", |
279 | len); | 283 | len); |
@@ -794,7 +798,8 @@ iscsi_tcp_hdr_recv_done(struct iscsi_tcp_conn *tcp_conn, | |||
794 | /* We're done processing the header. See if we're doing | 798 | /* We're done processing the header. See if we're doing |
795 | * header digests; if so, set up the recv_digest buffer | 799 | * header digests; if so, set up the recv_digest buffer |
796 | * and go back for more. */ | 800 | * and go back for more. */ |
797 | if (conn->hdrdgst_en) { | 801 | if (conn->hdrdgst_en && |
802 | !(conn->session->tt->caps & CAP_DIGEST_OFFLOAD)) { | ||
798 | if (segment->digest_len == 0) { | 803 | if (segment->digest_len == 0) { |
799 | /* | 804 | /* |
800 | * Even if we offload the digest processing we | 805 | * Even if we offload the digest processing we |
@@ -806,14 +811,12 @@ iscsi_tcp_hdr_recv_done(struct iscsi_tcp_conn *tcp_conn, | |||
806 | return 0; | 811 | return 0; |
807 | } | 812 | } |
808 | 813 | ||
809 | if (!(conn->session->tt->caps & CAP_DIGEST_OFFLOAD)) { | 814 | iscsi_tcp_dgst_header(tcp_conn->rx_hash, hdr, |
810 | iscsi_tcp_dgst_header(tcp_conn->rx_hash, hdr, | 815 | segment->total_copied - ISCSI_DIGEST_SIZE, |
811 | segment->total_copied - ISCSI_DIGEST_SIZE, | 816 | segment->digest); |
812 | segment->digest); | ||
813 | 817 | ||
814 | if (!iscsi_tcp_dgst_verify(tcp_conn, segment)) | 818 | if (!iscsi_tcp_dgst_verify(tcp_conn, segment)) |
815 | return ISCSI_ERR_HDR_DGST; | 819 | return ISCSI_ERR_HDR_DGST; |
816 | } | ||
817 | } | 820 | } |
818 | 821 | ||
819 | tcp_conn->in.hdr = hdr; | 822 | tcp_conn->in.hdr = hdr; |