aboutsummaryrefslogtreecommitdiffstats
path: root/fs/cifs/connect.c
diff options
context:
space:
mode:
Diffstat (limited to 'fs/cifs/connect.c')
-rw-r--r--fs/cifs/connect.c295
1 files changed, 149 insertions, 146 deletions
diff --git a/fs/cifs/connect.c b/fs/cifs/connect.c
index 8c5d310514ea..419f145c80b5 100644
--- a/fs/cifs/connect.c
+++ b/fs/cifs/connect.c
@@ -294,165 +294,168 @@ cifs_demultiplex_thread(struct TCP_Server_Info *server)
294 csocket = server->ssocket; 294 csocket = server->ssocket;
295 wake_up(&server->response_q); 295 wake_up(&server->response_q);
296 continue; 296 continue;
297 } else if (length > 3) { 297 } else if (length < 4) {
298 pdu_length = ntohl(smb_buffer->smb_buf_length); 298 cFYI(1,
299 /* Only read pdu_length after below checks for too short (due 299 ("Frame less than four bytes received %d bytes long.",
300 to e.g. int overflow) and too long ie beyond end of buf */ 300 length));
301 cFYI(1,("rfc1002 length(big endian)0x%x)", 301 cifs_reconnect(server);
302 pdu_length+4)); 302 csocket = server->ssocket;
303 303 wake_up(&server->response_q);
304 temp = (char *) smb_buffer; 304 continue;
305 if (temp[0] == (char) RFC1002_SESSION_KEEP_ALIVE) { 305 }
306 cFYI(0,("Received 4 byte keep alive packet")); 306
307 } else if (temp[0] == 307 /* the right amount was read from socket - 4 bytes */
308 (char) RFC1002_POSITIVE_SESSION_RESPONSE) { 308
309 cFYI(1,("Good RFC 1002 session rsp")); 309 pdu_length = ntohl(smb_buffer->smb_buf_length);
310 } else if (temp[0] == 310 cFYI(1,("rfc1002 length(big endian)0x%x)", pdu_length+4));
311 (char)RFC1002_NEGATIVE_SESSION_RESPONSE) { 311
312 /* we get this from Windows 98 instead of 312 temp = (char *) smb_buffer;
313 an error on SMB negprot response */ 313 if (temp[0] == (char) RFC1002_SESSION_KEEP_ALIVE) {
314 cFYI(1,("Negative RFC 1002 Session Response Error 0x%x)",temp[4])); 314 cFYI(0,("Received 4 byte keep alive packet"));
315 if(server->tcpStatus == CifsNew) { 315 } else if (temp[0] == (char)RFC1002_POSITIVE_SESSION_RESPONSE) {
316 /* if nack on negprot (rather than 316 cFYI(1,("Good RFC 1002 session rsp"));
317 ret of smb negprot error) reconnecting 317 } else if (temp[0] == (char)RFC1002_NEGATIVE_SESSION_RESPONSE) {
318 not going to help, ret error to mount */ 318 /* we get this from Windows 98 instead of
319 break; 319 an error on SMB negprot response */
320 } else { 320 cFYI(1,("Negative RFC 1002 Session Response Error 0x%x)",temp[4]));
321 /* give server a second to 321 if(server->tcpStatus == CifsNew) {
322 clean up before reconnect attempt */ 322 /* if nack on negprot (rather than
323 msleep(1000); 323 ret of smb negprot error) reconnecting
324 /* always try 445 first on reconnect 324 not going to help, ret error to mount */
325 since we get NACK on some if we ever 325 break;
326 connected to port 139 (the NACK is 326 } else {
327 since we do not begin with RFC1001 327 /* give server a second to
328 session initialize frame) */ 328 clean up before reconnect attempt */
329 server->addr.sockAddr.sin_port = 329 msleep(1000);
330 htons(CIFS_PORT); 330 /* always try 445 first on reconnect
331 cifs_reconnect(server); 331 since we get NACK on some if we ever
332 csocket = server->ssocket; 332 connected to port 139 (the NACK is
333 wake_up(&server->response_q); 333 since we do not begin with RFC1001
334 continue; 334 session initialize frame) */
335 } 335 server->addr.sockAddr.sin_port =
336 } else if (temp[0] != (char) 0) { 336 htons(CIFS_PORT);
337 cERROR(1,("Unknown RFC 1002 frame"));
338 cifs_dump_mem(" Received Data: ", temp, length);
339 cifs_reconnect(server); 337 cifs_reconnect(server);
340 csocket = server->ssocket; 338 csocket = server->ssocket;
339 wake_up(&server->response_q);
341 continue; 340 continue;
342 } else { 341 }
343 if((pdu_length > CIFSMaxBufSize + 342 } else if (temp[0] != (char) 0) {
344 MAX_CIFS_HDR_SIZE - 4) || 343 cERROR(1,("Unknown RFC 1002 frame"));
345 (pdu_length < sizeof (struct smb_hdr) - 1 - 4)) { 344 cifs_dump_mem(" Received Data: ", temp, length);
346 cERROR(1, 345 cifs_reconnect(server);
347 ("Invalid size SMB length %d and pdu_length %d", 346 csocket = server->ssocket;
348 length, pdu_length+4)); 347 continue;
349 cifs_reconnect(server); 348 } else { /* we have an SMB response */
350 csocket = server->ssocket; 349 if((pdu_length > CIFSMaxBufSize +
351 wake_up(&server->response_q); 350 MAX_CIFS_HDR_SIZE - 4) ||
352 continue; 351 (pdu_length < sizeof (struct smb_hdr) - 1 - 4)) {
353 } else { /* length ok */ 352 cERROR(1,
354 if(pdu_length > MAX_CIFS_HDR_SIZE - 4) { 353 ("Invalid size SMB length %d and pdu_length %d",
355 isLargeBuf = TRUE; 354 length, pdu_length+4));
356 memcpy(bigbuf, smallbuf, 4); 355 cifs_reconnect(server);
357 smb_buffer = bigbuf; 356 csocket = server->ssocket;
358 } 357 wake_up(&server->response_q);
359 length = 0; 358 continue;
360 iov.iov_base = 4 + (char *)smb_buffer; 359 } else { /* length ok */
361 iov.iov_len = pdu_length; 360 int reconnect = 0;
362 for (total_read = 0; 361
363 total_read < pdu_length; 362 if(pdu_length > MAX_CIFS_HDR_SIZE - 4) {
364 total_read += length) { 363 isLargeBuf = TRUE;
365 length = kernel_recvmsg(csocket, &smb_msg, 364 memcpy(bigbuf, smallbuf, 4);
366 &iov, 1, 365 smb_buffer = bigbuf;
367 pdu_length - total_read, 0); 366 }
368 if((server->tcpStatus == CifsExiting) || 367 length = 0;
369 (length == -EINTR)) { 368 iov.iov_base = 4 + (char *)smb_buffer;
370 /* then will exit */ 369 iov.iov_len = pdu_length;
371 goto dmx_loop_end; 370 for (total_read = 0;
372 } else if (server->tcpStatus == 371 total_read < pdu_length;
373 CifsNeedReconnect) { 372 total_read += length) {
374 cifs_reconnect(server); 373 length = kernel_recvmsg(csocket, &smb_msg,
375 csocket = server->ssocket; 374 &iov, 1,
376 /* Reconnect wakes up rspns q */ 375 pdu_length - total_read, 0);
376 if((server->tcpStatus == CifsExiting) ||
377 (length == -EINTR)) {
378 /* then will exit */
379 reconnect = 2;
380 break;
381 } else if (server->tcpStatus ==
382 CifsNeedReconnect) {
383 cifs_reconnect(server);
384 csocket = server->ssocket;
385 /* Reconnect wakes up rspns q */
377 /* Now we will reread sock */ 386 /* Now we will reread sock */
378 goto dmx_loop_end; 387 reconnect = 1;
379 } else if ((length == -ERESTARTSYS) || 388 break;
380 (length == -EAGAIN)) { 389 } else if ((length == -ERESTARTSYS) ||
381 msleep(1); /* minimum sleep to prevent looping 390 (length == -EAGAIN)) {
382 allowing socket to clear and app threads to set 391 msleep(1); /* minimum sleep to prevent looping
383 tcpStatus CifsNeedReconnect if server hung */ 392 allowing socket to clear and app threads to set
384 continue; 393 tcpStatus CifsNeedReconnect if server hung */
385 } else if (length <= 0) { 394 continue;
386 cERROR(1, 395 } else if (length <= 0) {
387 ("Received no data, expecting %d", 396 cERROR(1,("Received no data, expecting %d",
388 pdu_length - total_read)); 397 pdu_length - total_read));
389 cifs_reconnect(server); 398 cifs_reconnect(server);
390 csocket = server->ssocket; 399 csocket = server->ssocket;
391 goto dmx_loop_end; 400 reconnect = 1;
392 } 401 break;
393 } 402 }
394 length += 4; /* account for rfc1002 hdr */
395 } 403 }
404 if(reconnect == 2)
405 break;
406 else if(reconnect == 1)
407 continue;
396 408
397 dump_smb(smb_buffer, length); 409 length += 4; /* account for rfc1002 hdr */
398 if (checkSMB 410 }
399 (smb_buffer, smb_buffer->Mid, total_read+4)) {
400 cERROR(1, ("Bad SMB Received "));
401 continue;
402 }
403 411
404 /* BB FIXME - add checkTrans2SMBSecondary() */ 412 dump_smb(smb_buffer, length);
405 413 if (checkSMB
406 task_to_wake = NULL; 414 (smb_buffer, smb_buffer->Mid, total_read+4)) {
407 spin_lock(&GlobalMid_Lock); 415 cERROR(1, ("Bad SMB Received "));
408 list_for_each(tmp, &server->pending_mid_q) { 416 continue;
409 mid_entry = list_entry(tmp, struct 417 }
410 mid_q_entry, 418
411 qhead); 419
412 420 task_to_wake = NULL;
413 if ((mid_entry->mid == smb_buffer->Mid) 421 spin_lock(&GlobalMid_Lock);
414 && (mid_entry->midState == 422 list_for_each(tmp, &server->pending_mid_q) {
415 MID_REQUEST_SUBMITTED) 423 mid_entry = list_entry(tmp, struct mid_q_entry,
416 && (mid_entry->command == 424 qhead);
417 smb_buffer->Command)) { 425
418 cFYI(1,("Found Mid 0x%x wake up" 426 if ((mid_entry->mid == smb_buffer->Mid)
419 ,mid_entry->mid)); 427 && (mid_entry->midState ==
420 task_to_wake = mid_entry->tsk; 428 MID_REQUEST_SUBMITTED)
421 mid_entry->resp_buf = 429 && (mid_entry->command ==
422 smb_buffer; 430 smb_buffer->Command)) {
423 mid_entry->midState = 431 cFYI(1,("Found Mid 0x%x wake up"
424 MID_RESPONSE_RECEIVED; 432 ,mid_entry->mid));
425 if(isLargeBuf) 433 /* BB FIXME - missing code here BB */
426 mid_entry->largeBuf = 1; 434 /* check_2nd_t2(smb_buffer); */
427 else 435 task_to_wake = mid_entry->tsk;
428 mid_entry->largeBuf = 0; 436 mid_entry->resp_buf =
429 } 437 smb_buffer;
430 } 438 mid_entry->midState =
431 spin_unlock(&GlobalMid_Lock); 439 MID_RESPONSE_RECEIVED;
432 if (task_to_wake) {
433 if(isLargeBuf) 440 if(isLargeBuf)
434 bigbuf = NULL; 441 mid_entry->largeBuf = 1;
435 else 442 else
436 smallbuf = NULL; 443 mid_entry->largeBuf = 0;
437 smb_buffer = NULL; /* will be freed by users thread after he is done */
438 wake_up_process(task_to_wake);
439 } else if (is_valid_oplock_break(smb_buffer) == FALSE) {
440 cERROR(1, ("No task to wake, unknown frame rcvd!"));
441 cifs_dump_mem("Received Data is: ",temp,sizeof(struct smb_hdr));
442 } 444 }
443 } 445 }
444 } else { 446 spin_unlock(&GlobalMid_Lock);
445 cFYI(1, 447 if (task_to_wake) {
446 ("Frame less than four bytes received %d bytes long.", 448 if(isLargeBuf)
447 length)); 449 bigbuf = NULL;
448 cifs_reconnect(server); 450 else
449 csocket = server->ssocket; 451 smallbuf = NULL;
450 wake_up(&server->response_q); 452 smb_buffer = NULL; /* will be freed by users thread after he is done */
451 continue; 453 wake_up_process(task_to_wake);
454 } else if (is_valid_oplock_break(smb_buffer) == FALSE) {
455 cERROR(1, ("No task to wake, unknown frame rcvd!"));
456 cifs_dump_mem("Received Data is: ",temp,sizeof(struct smb_hdr));
457 }
452 } 458 }
453dmx_loop_end:
454 cFYI(1,("Exiting cifsd loop"));
455
456 } 459 }
457 spin_lock(&GlobalMid_Lock); 460 spin_lock(&GlobalMid_Lock);
458 server->tcpStatus = CifsExiting; 461 server->tcpStatus = CifsExiting;