home *** CD-ROM | disk | FTP | other *** search
/ Chip 2001 January / Chip_2001-01_cd1.bin / tema / mysql / mysql-3.23.28g-win-source.exe / sql / net_serv.cpp < prev    next >
C/C++ Source or Header  |  2000-09-29  |  19KB  |  680 lines

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