blob: 7b98792150ea7ebba3d449e994906483dcadf4a3 [file] [log] [blame]
Linus Torvalds1da177e2005-04-16 15:20:36 -07001/*
2 * fs/cifs/transport.c
3 *
Steve Frenchb8643e12005-04-28 22:41:07 -07004 * Copyright (C) International Business Machines Corp., 2002,2005
Linus Torvalds1da177e2005-04-16 15:20:36 -07005 * Author(s): Steve French (sfrench@us.ibm.com)
6 *
7 * This library is free software; you can redistribute it and/or modify
8 * it under the terms of the GNU Lesser General Public License as published
9 * by the Free Software Foundation; either version 2.1 of the License, or
10 * (at your option) any later version.
11 *
12 * This library is distributed in the hope that it will be useful,
13 * but WITHOUT ANY WARRANTY; without even the implied warranty of
14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See
15 * the GNU Lesser General Public License for more details.
16 *
17 * You should have received a copy of the GNU Lesser General Public License
18 * along with this library; if not, write to the Free Software
19 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
20 */
21
22#include <linux/fs.h>
23#include <linux/list.h>
24#include <linux/wait.h>
25#include <linux/net.h>
26#include <linux/delay.h>
27#include <asm/uaccess.h>
28#include <asm/processor.h>
29#include <linux/mempool.h>
30#include "cifspdu.h"
31#include "cifsglob.h"
32#include "cifsproto.h"
33#include "cifs_debug.h"
34
35extern mempool_t *cifs_mid_poolp;
36extern kmem_cache_t *cifs_oplock_cachep;
37
38static struct mid_q_entry *
39AllocMidQEntry(struct smb_hdr *smb_buffer, struct cifsSesInfo *ses)
40{
41 struct mid_q_entry *temp;
42
43 if (ses == NULL) {
Steve French275cde12005-04-28 22:41:10 -070044 cERROR(1, ("Null session passed in to AllocMidQEntry"));
Linus Torvalds1da177e2005-04-16 15:20:36 -070045 return NULL;
46 }
47 if (ses->server == NULL) {
48 cERROR(1, ("Null TCP session in AllocMidQEntry"));
49 return NULL;
50 }
51
Steve Frenchd6e04ae2005-06-13 13:24:43 -050052 temp = (struct mid_q_entry *) mempool_alloc(cifs_mid_poolp,
53 SLAB_KERNEL | SLAB_NOFS);
Linus Torvalds1da177e2005-04-16 15:20:36 -070054 if (temp == NULL)
55 return temp;
56 else {
57 memset(temp, 0, sizeof (struct mid_q_entry));
58 temp->mid = smb_buffer->Mid; /* always LE */
59 temp->pid = current->pid;
60 temp->command = smb_buffer->Command;
61 cFYI(1, ("For smb_command %d", temp->command));
Steve French1047abc2005-10-11 19:58:06 -070062 /* do_gettimeofday(&temp->when_sent);*/ /* easier to use jiffies */
63 /* when mid allocated can be before when sent */
64 temp->when_alloc = jiffies;
Linus Torvalds1da177e2005-04-16 15:20:36 -070065 temp->ses = ses;
66 temp->tsk = current;
67 }
68
69 spin_lock(&GlobalMid_Lock);
70 list_add_tail(&temp->qhead, &ses->server->pending_mid_q);
71 atomic_inc(&midCount);
72 temp->midState = MID_REQUEST_ALLOCATED;
73 spin_unlock(&GlobalMid_Lock);
74 return temp;
75}
76
77static void
78DeleteMidQEntry(struct mid_q_entry *midEntry)
79{
Steve French1047abc2005-10-11 19:58:06 -070080#ifdef CONFIG_CIFS_STATS2
81 unsigned long now;
82#endif
Linus Torvalds1da177e2005-04-16 15:20:36 -070083 spin_lock(&GlobalMid_Lock);
84 midEntry->midState = MID_FREE;
85 list_del(&midEntry->qhead);
86 atomic_dec(&midCount);
87 spin_unlock(&GlobalMid_Lock);
Steve Frenchb8643e12005-04-28 22:41:07 -070088 if(midEntry->largeBuf)
89 cifs_buf_release(midEntry->resp_buf);
90 else
91 cifs_small_buf_release(midEntry->resp_buf);
Steve French1047abc2005-10-11 19:58:06 -070092#ifdef CONFIG_CIFS_STATS2
93 now = jiffies;
94 /* commands taking longer than one second are indications that
95 something is wrong, unless it is quite a slow link or server */
96 if((now - midEntry->when_alloc) > HZ) {
97 if((cifsFYI & CIFS_TIMER) &&
98 (midEntry->command != SMB_COM_LOCKING_ANDX)) {
99 printk(KERN_DEBUG " CIFS slow rsp: cmd %d mid %d",
100 midEntry->command, midEntry->mid);
101 printk(" A: 0x%lx S: 0x%lx R: 0x%lx\n",
102 now - midEntry->when_alloc,
103 now - midEntry->when_sent,
104 now - midEntry->when_received);
105 }
106 }
107#endif
Linus Torvalds1da177e2005-04-16 15:20:36 -0700108 mempool_free(midEntry, cifs_mid_poolp);
109}
110
111struct oplock_q_entry *
112AllocOplockQEntry(struct inode * pinode, __u16 fid, struct cifsTconInfo * tcon)
113{
114 struct oplock_q_entry *temp;
115 if ((pinode== NULL) || (tcon == NULL)) {
116 cERROR(1, ("Null parms passed to AllocOplockQEntry"));
117 return NULL;
118 }
119 temp = (struct oplock_q_entry *) kmem_cache_alloc(cifs_oplock_cachep,
120 SLAB_KERNEL);
121 if (temp == NULL)
122 return temp;
123 else {
124 temp->pinode = pinode;
125 temp->tcon = tcon;
126 temp->netfid = fid;
127 spin_lock(&GlobalMid_Lock);
128 list_add_tail(&temp->qhead, &GlobalOplock_Q);
129 spin_unlock(&GlobalMid_Lock);
130 }
131 return temp;
132
133}
134
135void DeleteOplockQEntry(struct oplock_q_entry * oplockEntry)
136{
137 spin_lock(&GlobalMid_Lock);
138 /* should we check if list empty first? */
139 list_del(&oplockEntry->qhead);
140 spin_unlock(&GlobalMid_Lock);
141 kmem_cache_free(cifs_oplock_cachep, oplockEntry);
142}
143
144int
145smb_send(struct socket *ssocket, struct smb_hdr *smb_buffer,
146 unsigned int smb_buf_length, struct sockaddr *sin)
147{
148 int rc = 0;
149 int i = 0;
150 struct msghdr smb_msg;
151 struct kvec iov;
152 unsigned len = smb_buf_length + 4;
153
154 if(ssocket == NULL)
155 return -ENOTSOCK; /* BB eventually add reconnect code here */
156 iov.iov_base = smb_buffer;
157 iov.iov_len = len;
158
159 smb_msg.msg_name = sin;
160 smb_msg.msg_namelen = sizeof (struct sockaddr);
161 smb_msg.msg_control = NULL;
162 smb_msg.msg_controllen = 0;
163 smb_msg.msg_flags = MSG_DONTWAIT + MSG_NOSIGNAL; /* BB add more flags?*/
164
165 /* smb header is converted in header_assemble. bcc and rest of SMB word
166 area, and byte area if necessary, is converted to littleendian in
167 cifssmb.c and RFC1001 len is converted to bigendian in smb_send
168 Flags2 is converted in SendReceive */
169
170 smb_buffer->smb_buf_length = cpu_to_be32(smb_buffer->smb_buf_length);
Steve French3e844692005-10-03 13:37:24 -0700171 cFYI(1, ("Sending smb of length %d", smb_buf_length));
Linus Torvalds1da177e2005-04-16 15:20:36 -0700172 dump_smb(smb_buffer, len);
173
174 while (len > 0) {
175 rc = kernel_sendmsg(ssocket, &smb_msg, &iov, 1, len);
176 if ((rc == -ENOSPC) || (rc == -EAGAIN)) {
177 i++;
Steve French3e844692005-10-03 13:37:24 -0700178 /* smaller timeout here than send2 since smaller size */
179 /* Although it may not be required, this also is smaller
180 oplock break time */
Steve French68058e72005-10-10 10:34:22 -0700181 if(i > 12) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700182 cERROR(1,
Steve French68058e72005-10-10 10:34:22 -0700183 ("sends on sock %p stuck for 7 seconds",
Linus Torvalds1da177e2005-04-16 15:20:36 -0700184 ssocket));
185 rc = -EAGAIN;
186 break;
187 }
Steve French68058e72005-10-10 10:34:22 -0700188 msleep(1 << i);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700189 continue;
190 }
191 if (rc < 0)
192 break;
Steve French5e1253b2005-10-10 14:06:37 -0700193 else
194 i = 0; /* reset i after each successful send */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700195 iov.iov_base += rc;
196 iov.iov_len -= rc;
197 len -= rc;
198 }
199
200 if (rc < 0) {
Steve French3e844692005-10-03 13:37:24 -0700201 cERROR(1,("Error %d sending data on socket to server", rc));
Linus Torvalds1da177e2005-04-16 15:20:36 -0700202 } else {
203 rc = 0;
204 }
205
206 return rc;
207}
208
Steve Frenchd6e04ae2005-06-13 13:24:43 -0500209static int
Steve French3e844692005-10-03 13:37:24 -0700210smb_send2(struct socket *ssocket, struct kvec *iov, int n_vec,
211 struct sockaddr *sin)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700212{
213 int rc = 0;
214 int i = 0;
215 struct msghdr smb_msg;
Steve French3e844692005-10-03 13:37:24 -0700216 struct smb_hdr *smb_buffer = iov[0].iov_base;
217 unsigned int len = iov[0].iov_len;
218 unsigned int total_len;
219 int first_vec = 0;
Steve Frenchd6e04ae2005-06-13 13:24:43 -0500220
Linus Torvalds1da177e2005-04-16 15:20:36 -0700221 if(ssocket == NULL)
222 return -ENOTSOCK; /* BB eventually add reconnect code here */
Steve French3e844692005-10-03 13:37:24 -0700223
Linus Torvalds1da177e2005-04-16 15:20:36 -0700224 smb_msg.msg_name = sin;
225 smb_msg.msg_namelen = sizeof (struct sockaddr);
226 smb_msg.msg_control = NULL;
227 smb_msg.msg_controllen = 0;
228 smb_msg.msg_flags = MSG_DONTWAIT + MSG_NOSIGNAL; /* BB add more flags?*/
229
230 /* smb header is converted in header_assemble. bcc and rest of SMB word
231 area, and byte area if necessary, is converted to littleendian in
232 cifssmb.c and RFC1001 len is converted to bigendian in smb_send
233 Flags2 is converted in SendReceive */
234
Steve French3e844692005-10-03 13:37:24 -0700235
236 total_len = 0;
237 for (i = 0; i < n_vec; i++)
238 total_len += iov[i].iov_len;
239
Linus Torvalds1da177e2005-04-16 15:20:36 -0700240 smb_buffer->smb_buf_length = cpu_to_be32(smb_buffer->smb_buf_length);
Steve French3e844692005-10-03 13:37:24 -0700241 cFYI(1, ("Sending smb: total_len %d", total_len));
Linus Torvalds1da177e2005-04-16 15:20:36 -0700242 dump_smb(smb_buffer, len);
243
Steve French3e844692005-10-03 13:37:24 -0700244 while (total_len) {
245 rc = kernel_sendmsg(ssocket, &smb_msg, &iov[first_vec],
246 n_vec - first_vec, total_len);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700247 if ((rc == -ENOSPC) || (rc == -EAGAIN)) {
248 i++;
Steve French68058e72005-10-10 10:34:22 -0700249 if(i >= 14) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700250 cERROR(1,
Steve French68058e72005-10-10 10:34:22 -0700251 ("sends on sock %p stuck for 15 seconds",
Linus Torvalds1da177e2005-04-16 15:20:36 -0700252 ssocket));
253 rc = -EAGAIN;
254 break;
255 }
Steve French68058e72005-10-10 10:34:22 -0700256 msleep(1 << i);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700257 continue;
258 }
259 if (rc < 0)
260 break;
Steve French3e844692005-10-03 13:37:24 -0700261
262 if (rc >= total_len) {
263 WARN_ON(rc > total_len);
264 break;
Steve Frenchd6e04ae2005-06-13 13:24:43 -0500265 }
Steve French3e844692005-10-03 13:37:24 -0700266 if(rc == 0) {
267 /* should never happen, letting socket clear before
268 retrying is our only obvious option here */
Steve French04c08812005-10-03 19:33:15 -0700269 cERROR(1,("tcp sent no data"));
Steve French3e844692005-10-03 13:37:24 -0700270 msleep(500);
271 continue;
272 }
273 total_len -= rc;
Steve French68058e72005-10-10 10:34:22 -0700274 /* the line below resets i */
Steve French3e844692005-10-03 13:37:24 -0700275 for (i = first_vec; i < n_vec; i++) {
276 if (iov[i].iov_len) {
277 if (rc > iov[i].iov_len) {
278 rc -= iov[i].iov_len;
279 iov[i].iov_len = 0;
280 } else {
281 iov[i].iov_base += rc;
282 iov[i].iov_len -= rc;
283 first_vec = i;
284 break;
285 }
286 }
Steve Frenchd6e04ae2005-06-13 13:24:43 -0500287 }
Steve French5e1253b2005-10-10 14:06:37 -0700288 i = 0; /* in case we get ENOSPC on the next send */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700289 }
290
291 if (rc < 0) {
Steve French3e844692005-10-03 13:37:24 -0700292 cERROR(1,("Error %d sending data on socket to server", rc));
293 } else
Linus Torvalds1da177e2005-04-16 15:20:36 -0700294 rc = 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700295
296 return rc;
297}
298
Linus Torvalds1da177e2005-04-16 15:20:36 -0700299int
Steve Frenchd6e04ae2005-06-13 13:24:43 -0500300SendReceive2(const unsigned int xid, struct cifsSesInfo *ses,
Steve Frenchec637e32005-12-12 20:53:18 -0800301 struct kvec *iov, int n_vec, int * pRespBufType /* ret */,
Steve French3e844692005-10-03 13:37:24 -0700302 const int long_op)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700303{
304 int rc = 0;
Steve Frenchd6e04ae2005-06-13 13:24:43 -0500305 unsigned int receive_len;
306 unsigned long timeout;
307 struct mid_q_entry *midQ;
Steve French3e844692005-10-03 13:37:24 -0700308 struct smb_hdr *in_buf = iov[0].iov_base;
Steve Frenchec637e32005-12-12 20:53:18 -0800309
310 *pRespBufType = CIFS_NO_BUFFER; /* no response buf yet */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700311
312 if (ses == NULL) {
313 cERROR(1,("Null smb session"));
314 return -EIO;
315 }
316 if(ses->server == NULL) {
317 cERROR(1,("Null tcp session"));
318 return -EIO;
319 }
Linus Torvalds1da177e2005-04-16 15:20:36 -0700320
Steve Frenchd6e04ae2005-06-13 13:24:43 -0500321 if(ses->server->tcpStatus == CifsExiting)
Steve French31ca3bc2005-04-28 22:41:11 -0700322 return -ENOENT;
323
Linus Torvalds1da177e2005-04-16 15:20:36 -0700324 /* Ensure that we do not send more than 50 overlapping requests
325 to the same server. We may make this configurable later or
326 use ses->maxReq */
327 if(long_op == -1) {
328 /* oplock breaks must not be held up */
329 atomic_inc(&ses->server->inFlight);
330 } else {
331 spin_lock(&GlobalMid_Lock);
332 while(1) {
Steve Frenchd6e04ae2005-06-13 13:24:43 -0500333 if(atomic_read(&ses->server->inFlight) >=
334 cifs_max_pending){
Linus Torvalds1da177e2005-04-16 15:20:36 -0700335 spin_unlock(&GlobalMid_Lock);
Steve French131afd02005-10-07 09:51:05 -0700336#ifdef CONFIG_CIFS_STATS2
337 atomic_inc(&ses->server->num_waiters);
338#endif
Linus Torvalds1da177e2005-04-16 15:20:36 -0700339 wait_event(ses->server->request_q,
340 atomic_read(&ses->server->inFlight)
341 < cifs_max_pending);
Steve French131afd02005-10-07 09:51:05 -0700342#ifdef CONFIG_CIFS_STATS2
343 atomic_dec(&ses->server->num_waiters);
344#endif
Linus Torvalds1da177e2005-04-16 15:20:36 -0700345 spin_lock(&GlobalMid_Lock);
346 } else {
347 if(ses->server->tcpStatus == CifsExiting) {
348 spin_unlock(&GlobalMid_Lock);
349 return -ENOENT;
350 }
351
352 /* can not count locking commands against total since
353 they are allowed to block on server */
354
355 if(long_op < 3) {
356 /* update # of requests on the wire to server */
357 atomic_inc(&ses->server->inFlight);
358 }
359 spin_unlock(&GlobalMid_Lock);
360 break;
361 }
362 }
363 }
364 /* make sure that we sign in the same order that we send on this socket
365 and avoid races inside tcp sendmsg code that could cause corruption
366 of smb data */
367
368 down(&ses->server->tcpSem);
369
370 if (ses->server->tcpStatus == CifsExiting) {
371 rc = -ENOENT;
Steve Frenchd6e04ae2005-06-13 13:24:43 -0500372 goto out_unlock2;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700373 } else if (ses->server->tcpStatus == CifsNeedReconnect) {
374 cFYI(1,("tcp session dead - return to caller to retry"));
375 rc = -EAGAIN;
Steve Frenchd6e04ae2005-06-13 13:24:43 -0500376 goto out_unlock2;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700377 } else if (ses->status != CifsGood) {
378 /* check if SMB session is bad because we are setting it up */
379 if((in_buf->Command != SMB_COM_SESSION_SETUP_ANDX) &&
380 (in_buf->Command != SMB_COM_NEGOTIATE)) {
381 rc = -EAGAIN;
Steve Frenchd6e04ae2005-06-13 13:24:43 -0500382 goto out_unlock2;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700383 } /* else ok - we are setting up session */
384 }
385 midQ = AllocMidQEntry(in_buf, ses);
386 if (midQ == NULL) {
387 up(&ses->server->tcpSem);
388 /* If not lock req, update # of requests on wire to server */
389 if(long_op < 3) {
390 atomic_dec(&ses->server->inFlight);
391 wake_up(&ses->server->request_q);
392 }
393 return -ENOMEM;
394 }
395
Steve French84afc292005-12-02 13:32:45 -0800396 rc = cifs_sign_smb2(iov, n_vec, ses->server, &midQ->sequence_number);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700397
398 midQ->midState = MID_REQUEST_SUBMITTED;
Steve French131afd02005-10-07 09:51:05 -0700399#ifdef CONFIG_CIFS_STATS2
400 atomic_inc(&ses->server->inSend);
401#endif
Steve French3e844692005-10-03 13:37:24 -0700402 rc = smb_send2(ses->server->ssocket, iov, n_vec,
Steve Frenchd6e04ae2005-06-13 13:24:43 -0500403 (struct sockaddr *) &(ses->server->addr.sockAddr));
Steve French131afd02005-10-07 09:51:05 -0700404#ifdef CONFIG_CIFS_STATS2
405 atomic_dec(&ses->server->inSend);
Steve French1047abc2005-10-11 19:58:06 -0700406 midQ->when_sent = jiffies;
Steve French131afd02005-10-07 09:51:05 -0700407#endif
Linus Torvalds1da177e2005-04-16 15:20:36 -0700408 if(rc < 0) {
409 DeleteMidQEntry(midQ);
410 up(&ses->server->tcpSem);
411 /* If not lock req, update # of requests on wire to server */
412 if(long_op < 3) {
413 atomic_dec(&ses->server->inFlight);
414 wake_up(&ses->server->request_q);
415 }
416 return rc;
417 } else
418 up(&ses->server->tcpSem);
Steve Frenchd6e04ae2005-06-13 13:24:43 -0500419 if (long_op == -1)
420 goto cifs_no_response_exit2;
421 else if (long_op == 2) /* writes past end of file can take loong time */
Steve French37c0eb42005-10-05 14:50:29 -0700422 timeout = 180 * HZ;
Steve Frenchd6e04ae2005-06-13 13:24:43 -0500423 else if (long_op == 1)
424 timeout = 45 * HZ; /* should be greater than
425 servers oplock break timeout (about 43 seconds) */
426 else if (long_op > 2) {
427 timeout = MAX_SCHEDULE_TIMEOUT;
428 } else
429 timeout = 15 * HZ;
430 /* wait for 15 seconds or until woken up due to response arriving or
431 due to last connection to this server being unmounted */
432 if (signal_pending(current)) {
433 /* if signal pending do not hold up user for full smb timeout
434 but we still give response a change to complete */
435 timeout = 2 * HZ;
436 }
437
438 /* No user interrupts in wait - wreaks havoc with performance */
439 if(timeout != MAX_SCHEDULE_TIMEOUT) {
440 timeout += jiffies;
441 wait_event(ses->server->response_q,
442 (!(midQ->midState & MID_REQUEST_SUBMITTED)) ||
443 time_after(jiffies, timeout) ||
444 ((ses->server->tcpStatus != CifsGood) &&
445 (ses->server->tcpStatus != CifsNew)));
446 } else {
447 wait_event(ses->server->response_q,
448 (!(midQ->midState & MID_REQUEST_SUBMITTED)) ||
449 ((ses->server->tcpStatus != CifsGood) &&
450 (ses->server->tcpStatus != CifsNew)));
451 }
452
453 spin_lock(&GlobalMid_Lock);
454 if (midQ->resp_buf) {
455 spin_unlock(&GlobalMid_Lock);
Steve French70ca7342005-09-22 16:32:06 -0700456 receive_len = midQ->resp_buf->smb_buf_length;
Steve Frenchd6e04ae2005-06-13 13:24:43 -0500457 } else {
Steve French37c0eb42005-10-05 14:50:29 -0700458 cERROR(1,("No response to cmd %d mid %d",
459 midQ->command, midQ->mid));
Steve Frenchd6e04ae2005-06-13 13:24:43 -0500460 if(midQ->midState == MID_REQUEST_SUBMITTED) {
461 if(ses->server->tcpStatus == CifsExiting)
462 rc = -EHOSTDOWN;
463 else {
464 ses->server->tcpStatus = CifsNeedReconnect;
465 midQ->midState = MID_RETRY_NEEDED;
466 }
467 }
468
469 if (rc != -EHOSTDOWN) {
470 if(midQ->midState == MID_RETRY_NEEDED) {
471 rc = -EAGAIN;
472 cFYI(1,("marking request for retry"));
473 } else {
474 rc = -EIO;
475 }
476 }
477 spin_unlock(&GlobalMid_Lock);
478 DeleteMidQEntry(midQ);
479 /* If not lock req, update # of requests on wire to server */
480 if(long_op < 3) {
481 atomic_dec(&ses->server->inFlight);
482 wake_up(&ses->server->request_q);
483 }
484 return rc;
485 }
486
487 if (receive_len > CIFSMaxBufSize + MAX_CIFS_HDR_SIZE) {
488 cERROR(1, ("Frame too large received. Length: %d Xid: %d",
489 receive_len, xid));
490 rc = -EIO;
491 } else { /* rcvd frame is ok */
Steve Frenchd6e04ae2005-06-13 13:24:43 -0500492 if (midQ->resp_buf &&
493 (midQ->midState == MID_RESPONSE_RECEIVED)) {
Steve French84afc292005-12-02 13:32:45 -0800494
Steve Frenchec637e32005-12-12 20:53:18 -0800495 iov[0].iov_base = (char *)midQ->resp_buf;
496 if(midQ->largeBuf)
497 *pRespBufType = CIFS_LARGE_BUFFER;
498 else
499 *pRespBufType = CIFS_SMALL_BUFFER;
500 iov[0].iov_len = receive_len + 4;
501 iov[1].iov_len = 0;
Steve Frenchd6e04ae2005-06-13 13:24:43 -0500502
Steve Frenchec637e32005-12-12 20:53:18 -0800503 dump_smb(midQ->resp_buf, 80);
Steve Frenchd6e04ae2005-06-13 13:24:43 -0500504 /* convert the length into a more usable form */
505 if((receive_len > 24) &&
506 (ses->server->secMode & (SECMODE_SIGN_REQUIRED |
507 SECMODE_SIGN_ENABLED))) {
Steve Frenchec637e32005-12-12 20:53:18 -0800508 rc = cifs_verify_signature(midQ->resp_buf,
Steve Frenchd6e04ae2005-06-13 13:24:43 -0500509 ses->server->mac_signing_key,
510 midQ->sequence_number+1);
511 if(rc) {
512 cERROR(1,("Unexpected SMB signature"));
513 /* BB FIXME add code to kill session */
514 }
515 }
516
Steve Frenchd6e04ae2005-06-13 13:24:43 -0500517 /* BB special case reconnect tid and uid here? */
Steve French6ab16d22005-11-29 20:55:11 -0800518 /* BB special case Errbadpassword and pwdexpired here */
Steve Frenchec637e32005-12-12 20:53:18 -0800519 rc = map_smb_to_linux_error(midQ->resp_buf);
Steve Frenchd6e04ae2005-06-13 13:24:43 -0500520
521 /* convert ByteCount if necessary */
522 if (receive_len >=
523 sizeof (struct smb_hdr) -
524 4 /* do not count RFC1001 header */ +
Steve Frenchec637e32005-12-12 20:53:18 -0800525 (2 * midQ->resp_buf->WordCount) + 2 /* bcc */ )
526 BCC(midQ->resp_buf) =
527 le16_to_cpu(BCC_LE(midQ->resp_buf));
528 midQ->resp_buf = NULL; /* mark it so will not be freed
529 by DeleteMidQEntry */
Steve Frenchd6e04ae2005-06-13 13:24:43 -0500530 } else {
531 rc = -EIO;
Steve Frenchab2f2182005-09-15 20:44:50 -0700532 cFYI(1,("Bad MID state?"));
Steve Frenchd6e04ae2005-06-13 13:24:43 -0500533 }
534 }
535cifs_no_response_exit2:
536 DeleteMidQEntry(midQ);
537
Linus Torvalds1da177e2005-04-16 15:20:36 -0700538 if(long_op < 3) {
Steve Frenchd6e04ae2005-06-13 13:24:43 -0500539 atomic_dec(&ses->server->inFlight);
540 wake_up(&ses->server->request_q);
541 }
542
543 return rc;
544
545out_unlock2:
546 up(&ses->server->tcpSem);
547 /* If not lock req, update # of requests on wire to server */
548 if(long_op < 3) {
549 atomic_dec(&ses->server->inFlight);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700550 wake_up(&ses->server->request_q);
551 }
552
553 return rc;
554}
Linus Torvalds1da177e2005-04-16 15:20:36 -0700555
556int
557SendReceive(const unsigned int xid, struct cifsSesInfo *ses,
558 struct smb_hdr *in_buf, struct smb_hdr *out_buf,
559 int *pbytes_returned, const int long_op)
560{
561 int rc = 0;
562 unsigned int receive_len;
563 unsigned long timeout;
564 struct mid_q_entry *midQ;
565
566 if (ses == NULL) {
567 cERROR(1,("Null smb session"));
568 return -EIO;
569 }
570 if(ses->server == NULL) {
571 cERROR(1,("Null tcp session"));
572 return -EIO;
573 }
574
Steve French31ca3bc2005-04-28 22:41:11 -0700575 if(ses->server->tcpStatus == CifsExiting)
576 return -ENOENT;
577
Linus Torvalds1da177e2005-04-16 15:20:36 -0700578 /* Ensure that we do not send more than 50 overlapping requests
579 to the same server. We may make this configurable later or
580 use ses->maxReq */
581 if(long_op == -1) {
582 /* oplock breaks must not be held up */
583 atomic_inc(&ses->server->inFlight);
584 } else {
585 spin_lock(&GlobalMid_Lock);
586 while(1) {
Steve French275cde12005-04-28 22:41:10 -0700587 if(atomic_read(&ses->server->inFlight) >=
588 cifs_max_pending){
Linus Torvalds1da177e2005-04-16 15:20:36 -0700589 spin_unlock(&GlobalMid_Lock);
Steve French131afd02005-10-07 09:51:05 -0700590#ifdef CONFIG_CIFS_STATS2
591 atomic_inc(&ses->server->num_waiters);
592#endif
Linus Torvalds1da177e2005-04-16 15:20:36 -0700593 wait_event(ses->server->request_q,
594 atomic_read(&ses->server->inFlight)
595 < cifs_max_pending);
Steve French131afd02005-10-07 09:51:05 -0700596#ifdef CONFIG_CIFS_STATS2
597 atomic_dec(&ses->server->num_waiters);
598#endif
Linus Torvalds1da177e2005-04-16 15:20:36 -0700599 spin_lock(&GlobalMid_Lock);
600 } else {
601 if(ses->server->tcpStatus == CifsExiting) {
602 spin_unlock(&GlobalMid_Lock);
603 return -ENOENT;
604 }
605
606 /* can not count locking commands against total since
607 they are allowed to block on server */
608
609 if(long_op < 3) {
610 /* update # of requests on the wire to server */
611 atomic_inc(&ses->server->inFlight);
612 }
613 spin_unlock(&GlobalMid_Lock);
614 break;
615 }
616 }
617 }
618 /* make sure that we sign in the same order that we send on this socket
619 and avoid races inside tcp sendmsg code that could cause corruption
620 of smb data */
621
622 down(&ses->server->tcpSem);
623
624 if (ses->server->tcpStatus == CifsExiting) {
625 rc = -ENOENT;
626 goto out_unlock;
627 } else if (ses->server->tcpStatus == CifsNeedReconnect) {
628 cFYI(1,("tcp session dead - return to caller to retry"));
629 rc = -EAGAIN;
630 goto out_unlock;
631 } else if (ses->status != CifsGood) {
632 /* check if SMB session is bad because we are setting it up */
633 if((in_buf->Command != SMB_COM_SESSION_SETUP_ANDX) &&
634 (in_buf->Command != SMB_COM_NEGOTIATE)) {
635 rc = -EAGAIN;
636 goto out_unlock;
637 } /* else ok - we are setting up session */
638 }
639 midQ = AllocMidQEntry(in_buf, ses);
640 if (midQ == NULL) {
641 up(&ses->server->tcpSem);
642 /* If not lock req, update # of requests on wire to server */
643 if(long_op < 3) {
644 atomic_dec(&ses->server->inFlight);
645 wake_up(&ses->server->request_q);
646 }
647 return -ENOMEM;
648 }
649
650 if (in_buf->smb_buf_length > CIFSMaxBufSize + MAX_CIFS_HDR_SIZE - 4) {
651 up(&ses->server->tcpSem);
652 cERROR(1,
653 ("Illegal length, greater than maximum frame, %d ",
654 in_buf->smb_buf_length));
655 DeleteMidQEntry(midQ);
656 /* If not lock req, update # of requests on wire to server */
657 if(long_op < 3) {
658 atomic_dec(&ses->server->inFlight);
659 wake_up(&ses->server->request_q);
660 }
661 return -EIO;
662 }
663
Steve Frenchad009ac2005-04-28 22:41:05 -0700664 rc = cifs_sign_smb(in_buf, ses->server, &midQ->sequence_number);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700665
666 midQ->midState = MID_REQUEST_SUBMITTED;
Steve French131afd02005-10-07 09:51:05 -0700667#ifdef CONFIG_CIFS_STATS2
668 atomic_inc(&ses->server->inSend);
669#endif
Linus Torvalds1da177e2005-04-16 15:20:36 -0700670 rc = smb_send(ses->server->ssocket, in_buf, in_buf->smb_buf_length,
671 (struct sockaddr *) &(ses->server->addr.sockAddr));
Steve French131afd02005-10-07 09:51:05 -0700672#ifdef CONFIG_CIFS_STATS2
673 atomic_dec(&ses->server->inSend);
Steve French1047abc2005-10-11 19:58:06 -0700674 midQ->when_sent = jiffies;
Steve French131afd02005-10-07 09:51:05 -0700675#endif
Linus Torvalds1da177e2005-04-16 15:20:36 -0700676 if(rc < 0) {
677 DeleteMidQEntry(midQ);
678 up(&ses->server->tcpSem);
679 /* If not lock req, update # of requests on wire to server */
680 if(long_op < 3) {
681 atomic_dec(&ses->server->inFlight);
682 wake_up(&ses->server->request_q);
683 }
684 return rc;
685 } else
686 up(&ses->server->tcpSem);
687 if (long_op == -1)
688 goto cifs_no_response_exit;
Steve French275cde12005-04-28 22:41:10 -0700689 else if (long_op == 2) /* writes past end of file can take loong time */
Steve French37c0eb42005-10-05 14:50:29 -0700690 timeout = 180 * HZ;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700691 else if (long_op == 1)
692 timeout = 45 * HZ; /* should be greater than
693 servers oplock break timeout (about 43 seconds) */
694 else if (long_op > 2) {
695 timeout = MAX_SCHEDULE_TIMEOUT;
696 } else
697 timeout = 15 * HZ;
698 /* wait for 15 seconds or until woken up due to response arriving or
699 due to last connection to this server being unmounted */
700 if (signal_pending(current)) {
701 /* if signal pending do not hold up user for full smb timeout
702 but we still give response a change to complete */
703 timeout = 2 * HZ;
704 }
705
706 /* No user interrupts in wait - wreaks havoc with performance */
707 if(timeout != MAX_SCHEDULE_TIMEOUT) {
708 timeout += jiffies;
709 wait_event(ses->server->response_q,
710 (!(midQ->midState & MID_REQUEST_SUBMITTED)) ||
711 time_after(jiffies, timeout) ||
712 ((ses->server->tcpStatus != CifsGood) &&
713 (ses->server->tcpStatus != CifsNew)));
714 } else {
715 wait_event(ses->server->response_q,
716 (!(midQ->midState & MID_REQUEST_SUBMITTED)) ||
717 ((ses->server->tcpStatus != CifsGood) &&
718 (ses->server->tcpStatus != CifsNew)));
719 }
720
721 spin_lock(&GlobalMid_Lock);
722 if (midQ->resp_buf) {
723 spin_unlock(&GlobalMid_Lock);
Steve French70ca7342005-09-22 16:32:06 -0700724 receive_len = midQ->resp_buf->smb_buf_length;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700725 } else {
Steve French37c0eb42005-10-05 14:50:29 -0700726 cERROR(1,("No response for cmd %d mid %d",
727 midQ->command, midQ->mid));
Linus Torvalds1da177e2005-04-16 15:20:36 -0700728 if(midQ->midState == MID_REQUEST_SUBMITTED) {
729 if(ses->server->tcpStatus == CifsExiting)
730 rc = -EHOSTDOWN;
731 else {
732 ses->server->tcpStatus = CifsNeedReconnect;
733 midQ->midState = MID_RETRY_NEEDED;
734 }
735 }
736
737 if (rc != -EHOSTDOWN) {
738 if(midQ->midState == MID_RETRY_NEEDED) {
739 rc = -EAGAIN;
740 cFYI(1,("marking request for retry"));
741 } else {
742 rc = -EIO;
743 }
744 }
745 spin_unlock(&GlobalMid_Lock);
746 DeleteMidQEntry(midQ);
747 /* If not lock req, update # of requests on wire to server */
748 if(long_op < 3) {
749 atomic_dec(&ses->server->inFlight);
750 wake_up(&ses->server->request_q);
751 }
752 return rc;
753 }
754
755 if (receive_len > CIFSMaxBufSize + MAX_CIFS_HDR_SIZE) {
Steve Frenchad009ac2005-04-28 22:41:05 -0700756 cERROR(1, ("Frame too large received. Length: %d Xid: %d",
Linus Torvalds1da177e2005-04-16 15:20:36 -0700757 receive_len, xid));
758 rc = -EIO;
759 } else { /* rcvd frame is ok */
760
761 if (midQ->resp_buf && out_buf
762 && (midQ->midState == MID_RESPONSE_RECEIVED)) {
763 out_buf->smb_buf_length = receive_len;
764 memcpy((char *)out_buf + 4,
765 (char *)midQ->resp_buf + 4,
766 receive_len);
767
768 dump_smb(out_buf, 92);
769 /* convert the length into a more usable form */
770 if((receive_len > 24) &&
Steve Frenchad009ac2005-04-28 22:41:05 -0700771 (ses->server->secMode & (SECMODE_SIGN_REQUIRED |
772 SECMODE_SIGN_ENABLED))) {
773 rc = cifs_verify_signature(out_buf,
774 ses->server->mac_signing_key,
775 midQ->sequence_number+1);
776 if(rc) {
Steve French275cde12005-04-28 22:41:10 -0700777 cERROR(1,("Unexpected SMB signature"));
778 /* BB FIXME add code to kill session */
Steve Frenchad009ac2005-04-28 22:41:05 -0700779 }
Linus Torvalds1da177e2005-04-16 15:20:36 -0700780 }
781
782 *pbytes_returned = out_buf->smb_buf_length;
783
Steve Frenchad009ac2005-04-28 22:41:05 -0700784 /* BB special case reconnect tid and uid here? */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700785 rc = map_smb_to_linux_error(out_buf);
786
787 /* convert ByteCount if necessary */
788 if (receive_len >=
789 sizeof (struct smb_hdr) -
790 4 /* do not count RFC1001 header */ +
791 (2 * out_buf->WordCount) + 2 /* bcc */ )
Steve French0f2b27c2005-11-16 14:25:50 -0800792 BCC(out_buf) = le16_to_cpu(BCC_LE(out_buf));
Linus Torvalds1da177e2005-04-16 15:20:36 -0700793 } else {
794 rc = -EIO;
Steve Frenchec637e32005-12-12 20:53:18 -0800795 cERROR(1,("Bad MID state?"));
Linus Torvalds1da177e2005-04-16 15:20:36 -0700796 }
797 }
798cifs_no_response_exit:
799 DeleteMidQEntry(midQ);
800
801 if(long_op < 3) {
802 atomic_dec(&ses->server->inFlight);
803 wake_up(&ses->server->request_q);
804 }
805
806 return rc;
807
808out_unlock:
809 up(&ses->server->tcpSem);
810 /* If not lock req, update # of requests on wire to server */
811 if(long_op < 3) {
812 atomic_dec(&ses->server->inFlight);
813 wake_up(&ses->server->request_q);
814 }
815
816 return rc;
817}