net.c
上传用户:tsgydb
上传日期:2007-04-14
资源大小:10674k
文件大小:18k
源码类别:

MySQL数据库

开发平台:

Visual C++

  1. /* Copyright (C) 2000 MySQL AB & MySQL Finland AB & TCX DataKonsult AB
  2.    
  3.    This library is free software; you can redistribute it and/or
  4.    modify it under the terms of the GNU Library General Public
  5.    License as published by the Free Software Foundation; either
  6.    version 2 of the License, or (at your option) any later version.
  7.    
  8.    This library is distributed in the hope that it will be useful,
  9.    but WITHOUT ANY WARRANTY; without even the implied warranty of
  10.    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
  11.    Library General Public License for more details.
  12.    
  13.    You should have received a copy of the GNU Library General Public
  14.    License along with this library; if not, write to the Free
  15.    Software Foundation, Inc., 59 Temple Place - Suite 330, Boston,
  16.    MA 02111-1307, USA */
  17. /* Write and read of logical packets to/from socket
  18. ** Writes are cached into net_buffer_length big packets.
  19. ** Read packets are reallocated dynamicly when reading big packets.
  20. ** Each logical packet has the following pre-info:
  21. ** 3 byte length & 1 byte package-number.
  22. */
  23. #ifdef __WIN__
  24. #include <winsock.h>
  25. #endif
  26. #include <global.h>
  27. #include <violite.h>
  28. #include <my_sys.h>
  29. #include <m_string.h>
  30. #include "mysql.h"
  31. #include "mysqld_error.h"
  32. #include <signal.h>
  33. #include <errno.h>
  34. #include <sys/types.h>
  35. #include <violite.h>
  36. #ifdef MYSQL_SERVER
  37. ulong max_allowed_packet=65536;
  38. extern ulong net_read_timeout,net_write_timeout;
  39. extern uint test_flags;
  40. #else
  41. ulong max_allowed_packet=16*1024*1024L;
  42. ulong net_read_timeout=  NET_READ_TIMEOUT;
  43. ulong net_write_timeout= NET_WRITE_TIMEOUT;
  44. #endif
  45. ulong net_buffer_length=8192; /* Default length. Enlarged if necessary */
  46. #if !defined(__WIN__) && !defined(MSDOS)
  47. #include <sys/socket.h>
  48. #else
  49. #undef MYSQL_SERVER /* Win32 can't handle interrupts */
  50. #endif
  51. #if !defined(MSDOS) && !defined(__WIN__) && !defined(HAVE_BROKEN_NETINET_INCLUDES) && !defined(__BEOS__)
  52. #include <netinet/in_systm.h>
  53. #include <netinet/in.h>
  54. #include <netinet/ip.h>
  55. #if !defined(alpha_linux_port)
  56. #include <netinet/tcp.h>
  57. #endif
  58. #endif
  59. #include "mysqld_error.h"
  60. #ifdef MYSQL_SERVER
  61. #include "my_pthread.h"
  62. #include "thr_alarm.h"
  63. void sql_print_error(const char *format,...);
  64. #define RETRY_COUNT mysqld_net_retry_count
  65. extern ulong mysqld_net_retry_count;
  66. #else
  67. typedef my_bool thr_alarm_t;
  68. typedef my_bool ALARM;
  69. #define thr_alarm_init(A) (*A)=0
  70. #define thr_alarm_in_use(A) (*(A))
  71. #define thr_end_alarm(A)
  72. #define thr_alarm(A,B,C) local_thr_alarm((A),(B),(C))
  73. static inline int local_thr_alarm(my_bool *A,int B __attribute__((unused)),ALARM *C __attribute__((unused)))
  74. {
  75.   *A=1;
  76.   return 0;
  77. }
  78. #define thr_got_alarm(A) 0
  79. #define RETRY_COUNT 1
  80. #endif
  81. #ifdef MYSQL_SERVER
  82. extern ulong bytes_sent, bytes_received; 
  83. extern pthread_mutex_t LOCK_bytes_sent , LOCK_bytes_received;
  84. #else
  85. #undef statistic_add
  86. #define statistic_add(A,B,C)
  87. #endif
  88. /*
  89. ** Give error if a too big packet is found
  90. ** The server can change this with the -O switch, but because the client
  91. ** can't normally do this the client should have a bigger max-buffer.
  92. */
  93. #define TEST_BLOCKING 8
  94. static int net_write_buff(NET *net,const char *packet,uint len);
  95. /* Init with packet info */
  96. int my_net_init(NET *net, Vio* vio)
  97. {
  98.   if (!(net->buff=(uchar*) my_malloc(net_buffer_length,MYF(MY_WME))))
  99.     return 1;
  100.   if (net_buffer_length > max_allowed_packet)
  101.     max_allowed_packet=net_buffer_length;
  102.   net->buff_end=net->buff+(net->max_packet=net_buffer_length);
  103.   net->vio = vio;
  104.   net->no_send_ok = 0;
  105.   net->error=0; net->return_errno=0; net->return_status=0;
  106.   net->timeout=(uint) net_read_timeout; /* Timeout for read */
  107.   net->pkt_nr=0;
  108.   net->write_pos=net->read_pos = net->buff;
  109.   net->last_error[0]=0;
  110.   net->compress=0; net->reading_or_writing=0;
  111.   net->where_b = net->remain_in_buf=0;
  112.   net->last_errno=0;
  113.   if (vio != 0) /* If real connection */
  114.   {
  115.     net->fd  = vio_fd(vio); /* For perl DBI/DBD */
  116. #if defined(MYSQL_SERVER) && !defined(___WIN__) && !defined(__EMX__)
  117.     if (!(test_flags & TEST_BLOCKING))
  118.       vio_blocking(vio, FALSE);
  119. #endif
  120.     vio_fastsend(vio);
  121.   }
  122.   return 0;
  123. }
  124. void net_end(NET *net)
  125. {
  126.   my_free((gptr) net->buff,MYF(MY_ALLOW_ZERO_PTR));
  127.   net->buff=0;
  128. }
  129. /* Realloc the packet buffer */
  130. static my_bool net_realloc(NET *net, ulong length)
  131. {
  132.   uchar *buff;
  133.   ulong pkt_length;
  134.   if (length >= max_allowed_packet)
  135.   {
  136.     DBUG_PRINT("error",("Packet too large (%lu)", length));
  137.     net->error=1;
  138.     net->last_errno=ER_NET_PACKET_TOO_LARGE;
  139.     return 1;
  140.   }
  141.   pkt_length = (length+IO_SIZE-1) & ~(IO_SIZE-1);
  142.   if (!(buff=(uchar*) my_realloc((char*) net->buff, pkt_length, MYF(MY_WME))))
  143.   {
  144.     net->error=1;
  145. #ifdef MYSQL_SERVER
  146.     net->last_errno=ER_OUT_OF_RESOURCES;
  147. #endif
  148.     return 1;
  149.   }
  150.   net->buff=net->write_pos=buff;
  151.   net->buff_end=buff+(net->max_packet=pkt_length);
  152.   return 0;
  153. }
  154. /* Remove unwanted characters from connection */
  155. void net_clear(NET *net)
  156. {
  157. #ifndef EXTRA_DEBUG
  158.   int count;
  159.   bool is_blocking=vio_is_blocking(net->vio);
  160.   if (is_blocking)
  161.     vio_blocking(net->vio, FALSE);
  162.   if (!vio_is_blocking(net->vio)) /* Safety if SSL */
  163.   {
  164.     while ( (count = vio_read(net->vio, (char*) (net->buff),
  165.       net->max_packet)) > 0)
  166.       DBUG_PRINT("info",("skipped %d bytes from file: %s",
  167.  count,vio_description(net->vio)));
  168.     if (is_blocking)
  169.       vio_blocking(net->vio, TRUE);
  170.   }
  171. #endif /* EXTRA_DEBUG */
  172.   net->pkt_nr=0; /* Ready for new command */
  173.   net->write_pos=net->buff;
  174. }
  175. /* Flush write_buffer if not empty. */
  176. int net_flush(NET *net)
  177. {
  178.   int error=0;
  179.   DBUG_ENTER("net_flush");
  180.   if (net->buff != net->write_pos)
  181.   {
  182.     error=net_real_write(net,(char*) net->buff,
  183.  (uint) (net->write_pos - net->buff));
  184.     net->write_pos=net->buff;
  185.   }
  186.   DBUG_RETURN(error);
  187. }
  188. /*****************************************************************************
  189. ** Write something to server/client buffer
  190. *****************************************************************************/
  191. /*
  192. ** Write a logical packet with packet header
  193. ** Format: Packet length (3 bytes), packet number(1 byte)
  194. **         When compression is used a 3 byte compression length is added
  195. ** NOTE: If compression is used the original package is destroyed!
  196. */
  197. int
  198. my_net_write(NET *net,const char *packet,ulong len)
  199. {
  200.   uchar buff[NET_HEADER_SIZE];
  201.   int3store(buff,len);
  202.   buff[3]= (net->compress) ? 0 : (uchar) (net->pkt_nr++);
  203.   if (net_write_buff(net,(char*) buff,NET_HEADER_SIZE))
  204.     return 1;
  205.   return net_write_buff(net,packet,len);
  206. }
  207. int
  208. net_write_command(NET *net,uchar command,const char *packet,ulong len)
  209. {
  210.   uchar buff[NET_HEADER_SIZE+1];
  211.   uint length=len+1; /* 1 extra byte for command */
  212.   int3store(buff,length);
  213.   buff[3]= (net->compress) ? 0 : (uchar) (net->pkt_nr++);
  214.   buff[4]=command;
  215.   if (net_write_buff(net,(char*) buff,5))
  216.     return 1;
  217.   return test(net_write_buff(net,packet,len) || net_flush(net));
  218. }
  219. static int
  220. net_write_buff(NET *net,const char *packet,uint len)
  221. {
  222.   uint left_length=(uint) (net->buff_end - net->write_pos);
  223.   while (len > left_length)
  224.   {
  225.     memcpy((char*) net->write_pos,packet,left_length);
  226.     if (net_real_write(net,(char*) net->buff,net->max_packet))
  227.       return 1;
  228.     net->write_pos=net->buff;
  229.     packet+=left_length;
  230.     len-=left_length;
  231.     left_length=net->max_packet;
  232.   }
  233.   memcpy((char*) net->write_pos,packet,len);
  234.   net->write_pos+=len;
  235.   return 0;
  236. }
  237. /*  Read and write using timeouts */
  238. int
  239. net_real_write(NET *net,const char *packet,ulong len)
  240. {
  241.   int length;
  242.   char *pos,*end;
  243.   thr_alarm_t alarmed;
  244. #if !defined(__WIN__) && !defined(__EMX__)
  245.   ALARM alarm_buff;
  246. #endif
  247.   uint retry_count=0;
  248.   my_bool net_blocking = vio_is_blocking(net->vio);
  249.   DBUG_ENTER("net_real_write");
  250.   if (net->error == 2)
  251.     DBUG_RETURN(-1); /* socket can't be used */
  252.   net->reading_or_writing=2;
  253. #ifdef HAVE_COMPRESS
  254.   if (net->compress)
  255.   {
  256.     ulong complen;
  257.     uchar *b;
  258.     uint header_length=NET_HEADER_SIZE+COMP_HEADER_SIZE;
  259.     if (!(b=(uchar*) my_malloc(len + NET_HEADER_SIZE + COMP_HEADER_SIZE,
  260.     MYF(MY_WME))))
  261.     {
  262. #ifdef MYSQL_SERVER
  263.       net->last_errno=ER_OUT_OF_RESOURCES;
  264.       net->error=2;
  265. #endif
  266.       net->reading_or_writing=0;
  267.       DBUG_RETURN(1);
  268.     }
  269.     memcpy(b+header_length,packet,len);
  270.     if (my_compress((byte*) b+header_length,&len,&complen))
  271.     {
  272.       DBUG_PRINT("warning",
  273.  ("Compression error; Continuing without compression"));
  274.       complen=0;
  275.     }
  276.     int3store(&b[NET_HEADER_SIZE],complen);
  277.     int3store(b,len);
  278.     b[3]=(uchar) (net->pkt_nr++);
  279.     len+= header_length;
  280.     packet= (char*) b;
  281.   }
  282. #endif /* HAVE_COMPRESS */
  283.   /* DBUG_DUMP("net",packet,len); */
  284. #ifdef MYSQL_SERVER
  285.   thr_alarm_init(&alarmed);
  286.   if (net_blocking)
  287.     thr_alarm(&alarmed,(uint) net_write_timeout,&alarm_buff);
  288. #else
  289.   alarmed=0;
  290. #endif /* MYSQL_SERVER */
  291.   pos=(char*) packet; end=pos+len;
  292.   while (pos != end)
  293.   {
  294.     if ((int) (length=vio_write(net->vio,pos,(int) (end-pos))) <= 0)
  295.     {
  296.       my_bool interrupted = vio_should_retry(net->vio);
  297. #if (!defined(__WIN__) && !defined(__EMX__))
  298.       if ((interrupted || length==0) && !thr_alarm_in_use(&alarmed))
  299.       {
  300.         if (!thr_alarm(&alarmed,(uint) net_write_timeout,&alarm_buff))
  301.         {                                       /* Always true for client */
  302.   if (!vio_is_blocking(net->vio))
  303.   {
  304.     while (vio_blocking(net->vio, TRUE) < 0)
  305.     {
  306.       if (vio_should_retry(net->vio) && retry_count++ < RETRY_COUNT)
  307. continue;
  308. #ifdef EXTRA_DEBUG
  309.       fprintf(stderr,
  310.       "%s: my_net_write: fcntl returned error %d, aborting threadn",
  311.       my_progname,vio_errno(net->vio));
  312. #endif /* EXTRA_DEBUG */
  313.       net->error=2;                     /* Close socket */
  314.       goto end;
  315.     }
  316.   }
  317.   retry_count=0;
  318.   continue;
  319. }
  320.       }
  321.       else
  322. #endif /* (!defined(__WIN__) && !defined(__EMX__)) */
  323. if (thr_alarm_in_use(&alarmed) && !thr_got_alarm(&alarmed) &&
  324.     interrupted)
  325.       {
  326. if (retry_count++ < RETRY_COUNT)
  327.     continue;
  328. #ifdef EXTRA_DEBUG
  329.   fprintf(stderr, "%s: write looped, aborting threadn",
  330.   my_progname);
  331. #endif /* EXTRA_DEBUG */
  332.       }
  333. #if defined(THREAD_SAFE_CLIENT) && !defined(MYSQL_SERVER)
  334.       if (vio_errno(net->vio) == EINTR)
  335.       {
  336. DBUG_PRINT("warning",("Interrupted write. Retrying..."));
  337. continue;
  338.       }
  339. #endif /* defined(THREAD_SAFE_CLIENT) && !defined(MYSQL_SERVER) */
  340.       net->error=2; /* Close socket */
  341. #ifdef MYSQL_SERVER
  342.       net->last_errno= (interrupted ? ER_NET_WRITE_INTERRUPTED :
  343. ER_NET_ERROR_ON_WRITE);
  344. #endif /* MYSQL_SERVER */
  345.       break;
  346.     }
  347.     pos+=length;
  348.     statistic_add(bytes_sent,length,&LOCK_bytes_sent);
  349.   }
  350. #ifndef __WIN__
  351.  end:
  352. #endif
  353. #ifdef HAVE_COMPRESS
  354.   if (net->compress)
  355.     my_free((char*) packet,MYF(0));
  356. #endif
  357.   if (thr_alarm_in_use(&alarmed))
  358.   {
  359.     thr_end_alarm(&alarmed);
  360.     vio_blocking(net->vio, net_blocking);
  361.   }
  362.   net->reading_or_writing=0;
  363.   DBUG_RETURN(((int) (pos != end)));
  364. }
  365. /*****************************************************************************
  366. ** Read something from server/clinet
  367. *****************************************************************************/
  368. #ifdef MYSQL_SERVER
  369. /*
  370.   Help function to clear the commuication buffer when we get a too
  371.   big packet
  372. */
  373. static void my_net_skip_rest(NET *net, ulong remain, thr_alarm_t *alarmed)
  374. {
  375.   ALARM alarm_buff;
  376.   uint retry_count=0;
  377.   if (!thr_alarm_in_use(&alarmed))
  378.   {
  379.     if (!thr_alarm(alarmed,net->timeout,&alarm_buff) ||
  380. (!vio_is_blocking(net->vio) && vio_blocking(net->vio,TRUE) < 0))
  381.       return; /* Can't setup, abort */
  382.   }
  383.   while (remain > 0)
  384.   {
  385.     ulong length;
  386.     if ((int) (length=vio_read(net->vio,(char*) net->buff,remain)) <= 0L)
  387.     {
  388.       my_bool interrupted = vio_should_retry(net->vio);
  389.       if (!thr_got_alarm(&alarmed) && interrupted)
  390.       { /* Probably in MIT threads */
  391. if (retry_count++ < RETRY_COUNT)
  392.   continue;
  393.       }
  394.       return;
  395.     }
  396.     remain -=(ulong) length;
  397.     statistic_add(bytes_received,(ulong) length,&LOCK_bytes_received);
  398.   }
  399. }
  400. #endif /* MYSQL_SERVER */
  401. static uint
  402. my_real_read(NET *net, ulong *complen)
  403. {
  404.   uchar *pos;
  405.   long length;
  406.   uint i,retry_count=0;
  407.   ulong len=packet_error;
  408.   thr_alarm_t alarmed;
  409. #if (!defined(__WIN__) && !defined(__EMX__)) || defined(MYSQL_SERVER)
  410.   ALARM alarm_buff;
  411. #endif
  412.   my_bool net_blocking=vio_is_blocking(net->vio);
  413.   ulong remain= (net->compress ? NET_HEADER_SIZE+COMP_HEADER_SIZE :
  414.  NET_HEADER_SIZE);
  415.   *complen = 0;
  416.   net->reading_or_writing=1;
  417.   thr_alarm_init(&alarmed);
  418. #ifdef MYSQL_SERVER
  419.   if (net_blocking)
  420.     thr_alarm(&alarmed,net->timeout,&alarm_buff);
  421. #endif /* MYSQL_SERVER */
  422.     pos = net->buff + net->where_b; /* net->packet -4 */
  423.     for (i=0 ; i < 2 ; i++)
  424.     {
  425.       while (remain > 0)
  426.       {
  427. /* First read is done with non blocking mode */
  428.         if ((int) (length=vio_read(net->vio,(char*) pos,remain)) <= 0L)
  429.         {
  430.           my_bool interrupted = vio_should_retry(net->vio);
  431.   DBUG_PRINT("info",("vio_read returned %d,  errno: %d",
  432.      length, vio_errno(net->vio)));
  433. #if (!defined(__WIN__) && !defined(__EMX__)) || defined(MYSQL_SERVER)
  434.   /*
  435.     We got an error that there was no data on the socket. We now set up
  436.     an alarm to not 'read forever', change the socket to non blocking
  437.     mode and try again
  438.   */
  439.   if ((interrupted || length == 0) && !thr_alarm_in_use(&alarmed))
  440.   {
  441.     if (!thr_alarm(&alarmed,net->timeout,&alarm_buff)) /* Don't wait too long */
  442.     {
  443.               if (!vio_is_blocking(net->vio))
  444.               {
  445.                 while (vio_blocking(net->vio,TRUE) < 0)
  446.                 {
  447.                   if (vio_should_retry(net->vio) &&
  448.       retry_count++ < RETRY_COUNT)
  449.                     continue;
  450.                   DBUG_PRINT("error",
  451.      ("fcntl returned error %d, aborting thread",
  452.       vio_errno(net->vio)));
  453. #ifdef EXTRA_DEBUG
  454.                   fprintf(stderr,
  455.                           "%s: read: fcntl returned error %d, aborting threadn",
  456.                           my_progname,vio_errno(net->vio));
  457. #endif /* EXTRA_DEBUG */
  458.                   len= packet_error;
  459.                   net->error=2;                 /* Close socket */
  460. #ifdef MYSQL_SERVER
  461.   net->last_errno=ER_NET_FCNTL_ERROR;
  462. #endif
  463.   goto end;
  464.                 }
  465.               }
  466.       retry_count=0;
  467.       continue;
  468.     }
  469.   }
  470. #endif /* (!defined(__WIN__) && !defined(__EMX__)) || defined(MYSQL_SERVER) */
  471.   if (thr_alarm_in_use(&alarmed) && !thr_got_alarm(&alarmed) &&
  472.       interrupted)
  473.   { /* Probably in MIT threads */
  474.     if (retry_count++ < RETRY_COUNT)
  475.       continue;
  476. #ifdef EXTRA_DEBUG
  477.     fprintf(stderr, "%s: read looped with error %d, aborting threadn",
  478.     my_progname,vio_errno(net->vio));
  479. #endif /* EXTRA_DEBUG */
  480.   }
  481. #if defined(THREAD_SAFE_CLIENT) && !defined(MYSQL_SERVER)
  482.   if (vio_should_retry(net->vio))
  483.   {
  484.     DBUG_PRINT("warning",("Interrupted read. Retrying..."));
  485.     continue;
  486.   }
  487. #endif
  488.   DBUG_PRINT("error",("Couldn't read packet: remain: %d  errno: %d  length: %d  alarmed: %d", remain,vio_errno(net->vio),length,alarmed));
  489.   len= packet_error;
  490.   net->error=2; /* Close socket */
  491. #ifdef MYSQL_SERVER
  492.   net->last_errno= (interrupted ? ER_NET_READ_INTERRUPTED :
  493.     ER_NET_READ_ERROR);
  494. #endif
  495.   goto end;
  496. }
  497. remain -= (ulong) length;
  498. pos+= (ulong) length;
  499. statistic_add(bytes_received,(ulong) length,&LOCK_bytes_received);
  500.       }
  501.       if (i == 0)
  502.       { /* First parts is packet length */
  503. ulong helping;
  504. if (net->buff[net->where_b + 3] != (uchar) net->pkt_nr)
  505. {
  506.   if (net->buff[net->where_b] != (uchar) 255)
  507.   {
  508.     DBUG_PRINT("error",
  509.        ("Packets out of order (Found: %d, expected %d)",
  510. (int) net->buff[net->where_b + 3],
  511. (uint) (uchar) net->pkt_nr));
  512. #ifdef EXTRA_DEBUG
  513.     fprintf(stderr,"Packets out of order (Found: %d, expected %d)n",
  514.     (int) net->buff[net->where_b + 3],
  515.     (uint) (uchar) net->pkt_nr);
  516. #endif
  517.   }
  518.   len= packet_error;
  519. #ifdef MYSQL_SERVER
  520.   net->last_errno=ER_NET_PACKETS_OUT_OF_ORDER;
  521. #endif
  522.   goto end;
  523. }
  524. net->pkt_nr++;
  525. #ifdef HAVE_COMPRESS
  526. if (net->compress)
  527. {
  528.   /* complen is > 0 if package is really compressed */
  529.   *complen=uint3korr(&(net->buff[net->where_b + NET_HEADER_SIZE]));
  530. }
  531. #endif
  532. len=uint3korr(net->buff+net->where_b);
  533. helping = max(len,*complen) + net->where_b;
  534. /* The necessary size of net->buff */
  535. if (helping >= net->max_packet)
  536. {
  537.   /* We must allocate one extra byte for the end null */
  538.   if (net_realloc(net,helping+1))
  539.   {
  540. #ifdef MYSQL_SERVER
  541.     if (i == 1)
  542.       my_net_skip_rest(net, len, &alarmed);
  543. #endif
  544.     len= packet_error; /* Return error */
  545.     goto end;
  546.   }
  547. }
  548. pos=net->buff + net->where_b;
  549. remain = len;
  550.       }
  551.     }
  552. end:
  553.   if (thr_alarm_in_use(&alarmed))
  554.   {
  555.     thr_end_alarm(&alarmed);
  556.     vio_blocking(net->vio, net_blocking);
  557.   }
  558.   net->reading_or_writing=0;
  559.   return(len);
  560. }
  561. uint
  562. my_net_read(NET *net)
  563. {
  564.   ulong len,complen;
  565. #ifdef HAVE_COMPRESS
  566.   if (!net->compress)
  567.   {
  568. #endif
  569.     len = my_real_read (net,&complen);
  570.     net->read_pos = net->buff + net->where_b;
  571.     if (len != packet_error)
  572.       net->read_pos[len]=0; /* Safeguard for mysql_use_result */
  573.     return len;
  574. #ifdef HAVE_COMPRESS
  575.   }
  576.   if (net->remain_in_buf)
  577.     net->buff[net->buf_length - net->remain_in_buf]=net->save_char;
  578.   for (;;)
  579.   {
  580.     if (net->remain_in_buf)
  581.     {
  582.       uchar *pos = net->buff + net->buf_length - net->remain_in_buf;
  583.       if (net->remain_in_buf >= 4)
  584.       {
  585. net->length = uint3korr(pos);
  586. if (net->length <= net->remain_in_buf - 4)
  587. {
  588.   /* We have a full packet */
  589.   len=net->length;
  590.   net->remain_in_buf -= net->length + 4;
  591.   net->read_pos=pos + 4;
  592.   break; /* We have a full packet */
  593. }
  594.       }
  595.       /* Move data down to read next data packet after current one */
  596.       if (net->buf_length != net->remain_in_buf)
  597.       {
  598. memmove(net->buff,pos,net->remain_in_buf);
  599. net->buf_length=net->remain_in_buf;
  600.       }
  601.       net->where_b=net->buf_length;
  602.     }
  603.     else
  604.     {
  605.       net->where_b=0;
  606.       net->buf_length=0;
  607.     }
  608.     if ((len = my_real_read(net,&complen)) == packet_error)
  609.       break;
  610.     if (my_uncompress((byte*) net->buff + net->where_b, &len, &complen))
  611.     {
  612.       len= packet_error;
  613.       net->error=2; /* caller will close socket */
  614. #ifdef MYSQL_SERVER
  615.       net->last_errno=ER_NET_UNCOMPRESS_ERROR;
  616. #endif
  617.       break;
  618.     }
  619.     net->buf_length+=len;
  620.     net->remain_in_buf+=len;
  621.   }
  622.   if (len != packet_error)
  623.   {
  624.     net->save_char= net->read_pos[len]; /* Must be saved */
  625.     net->read_pos[len]=0; /* Safeguard for mysql_use_result */
  626.   }
  627.   return len;
  628. #endif
  629. }