ctype-simple.c
上传用户:romrleung
上传日期:2022-05-23
资源大小:18897k
文件大小:27k
源码类别:

MySQL数据库

开发平台:

Visual C++

  1. /* Copyright (C) 2002 MySQL AB
  2.    This program is free software; you can redistribute it and/or modify
  3.    it under the terms of the GNU General Public License as published by
  4.    the Free Software Foundation; either version 2 of the License, or
  5.    (at your option) any later version.
  6.    This program is distributed in the hope that it will be useful,
  7.    but WITHOUT ANY WARRANTY; without even the implied warranty of
  8.    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
  9.    GNU General Public License for more details.
  10.    You should have received a copy of the GNU General Public License
  11.    along with this program; if not, write to the Free Software
  12.    Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA */
  13. #include <my_global.h>
  14. #include "m_string.h"
  15. #include "m_ctype.h"
  16. #include <errno.h>
  17. #include "stdarg.h"
  18. /*
  19.   Converts a string into its sort key.
  20.   
  21.   SYNOPSIS
  22.      my_strnxfrm_xxx()
  23.      
  24.   IMPLEMENTATION
  25.      
  26.      The my_strxfrm_xxx() function transforms a string pointed to by
  27.      'src' with length 'srclen' according to the charset+collation 
  28.      pair 'cs' and copies the result key into 'dest'.
  29.      
  30.      Comparing two strings using memcmp() after my_strnxfrm_xxx()
  31.      is equal to comparing two original strings with my_strnncollsp_xxx().
  32.      
  33.      Not more than 'dstlen' bytes are written into 'dst'.
  34.      To garantee that the whole string is transformed, 'dstlen' must be
  35.      at least srclen*cs->strnxfrm_multiply bytes long. Otherwise,
  36.      consequent memcmp() may return a non-accurate result.
  37.      
  38.      If the source string is too short to fill whole 'dstlen' bytes,
  39.      then the 'dest' string is padded up to 'dstlen', ensuring that:
  40.      
  41.        "a"  == "a "
  42.        "a" < "a"
  43.        "a" < "a "
  44.      
  45.      my_strnxfrm_simple() is implemented for 8bit charsets and
  46.      simple collations with one-to-one string->key transformation.
  47.      
  48.      See also implementations for various charsets/collations in  
  49.      other ctype-xxx.c files.
  50.      
  51.   RETURN
  52.   
  53.     Target len 'dstlen'.
  54.   
  55. */
  56. int my_strnxfrm_simple(CHARSET_INFO * cs, 
  57.                        uchar *dest, uint len,
  58.                        const uchar *src, uint srclen)
  59. {
  60.   uchar *map= cs->sort_order;
  61.   uint dstlen= len;
  62.   set_if_smaller(len, srclen);
  63.   if (dest != src)
  64.   {
  65.     const uchar *end;
  66.     for ( end=src+len; src < end ;  )
  67.       *dest++= map[*src++];
  68.   }
  69.   else
  70.   {
  71.     const uchar *end;
  72.     for ( end=dest+len; dest < end ; dest++)
  73.       *dest= (char) map[(uchar) *dest];
  74.   }
  75.   if (dstlen > len)
  76.     bfill(dest, dstlen - len, ' ');
  77.   return dstlen;
  78. }
  79. int my_strnncoll_simple(CHARSET_INFO * cs, const uchar *s, uint slen, 
  80.                         const uchar *t, uint tlen,
  81.                         my_bool t_is_prefix)
  82. {
  83.   int len = ( slen > tlen ) ? tlen : slen;
  84.   uchar *map= cs->sort_order;
  85.   if (t_is_prefix && slen > tlen)
  86.     slen=tlen;
  87.   while (len--)
  88.   {
  89.     if (map[*s++] != map[*t++])
  90.       return ((int) map[s[-1]] - (int) map[t[-1]]);
  91.   }
  92.   return (int) (slen - tlen);
  93. }
  94. /*
  95.   Compare strings, discarding end space
  96.   SYNOPSIS
  97.     my_strnncollsp_simple()
  98.     cs character set handler
  99.     a First string to compare
  100.     a_length Length of 'a'
  101.     b Second string to compare
  102.     b_length Length of 'b'
  103.   IMPLEMENTATION
  104.     If one string is shorter as the other, then we space extend the other
  105.     so that the strings have equal length.
  106.     This will ensure that the following things hold:
  107.     "a"  == "a "
  108.     "a" < "a"
  109.     "a" < "a "
  110.   RETURN
  111.     < 0  a <  b
  112.     = 0  a == b
  113.     > 0  a > b
  114. */
  115. int my_strnncollsp_simple(CHARSET_INFO * cs, const uchar *a, uint a_length, 
  116.   const uchar *b, uint b_length)
  117. {
  118.   const uchar *map= cs->sort_order, *end;
  119.   uint length;
  120.   end= a + (length= min(a_length, b_length));
  121.   while (a < end)
  122.   {
  123.     if (map[*a++] != map[*b++])
  124.       return ((int) map[a[-1]] - (int) map[b[-1]]);
  125.   }
  126.   if (a_length != b_length)
  127.   {
  128.     int swap= 1;
  129.     /*
  130.       Check the next not space character of the longer key. If it's < ' ',
  131.       then it's smaller than the other key.
  132.     */
  133.     if (a_length < b_length)
  134.     {
  135.       /* put shorter key in s */
  136.       a_length= b_length;
  137.       a= b;
  138.       swap= -1;                                 /* swap sign of result */
  139.     }
  140.     for (end= a + a_length-length; a < end ; a++)
  141.     {
  142.       if (*a != ' ')
  143. return (*a < ' ') ? -swap : swap;
  144.     }
  145.   }
  146.   return 0;
  147. }
  148. void my_caseup_str_8bit(CHARSET_INFO * cs,char *str)
  149. {
  150.   register uchar *map=cs->to_upper;
  151.   while ((*str = (char) map[(uchar) *str]) != 0)
  152.     str++;
  153. }
  154. void my_casedn_str_8bit(CHARSET_INFO * cs,char *str)
  155. {
  156.   register uchar *map=cs->to_lower;
  157.   while ((*str = (char) map[(uchar)*str]) != 0)
  158.     str++;
  159. }
  160. void my_caseup_8bit(CHARSET_INFO * cs, char *str, uint length)
  161. {
  162.   register uchar *map=cs->to_upper;
  163.   for ( ; length>0 ; length--, str++)
  164.     *str= (char) map[(uchar)*str];
  165. }
  166. void my_casedn_8bit(CHARSET_INFO * cs, char *str, uint length)
  167. {
  168.   register uchar *map=cs->to_lower;
  169.   for ( ; length>0 ; length--, str++)
  170.     *str= (char) map[(uchar) *str];
  171. }
  172. int my_strcasecmp_8bit(CHARSET_INFO * cs,const char *s, const char *t)
  173. {
  174.   register uchar *map=cs->to_upper;
  175.   while (map[(uchar) *s] == map[(uchar) *t++])
  176.     if (!*s++) return 0;
  177.   return ((int) map[(uchar) s[0]] - (int) map[(uchar) t[-1]]);
  178. }
  179. int my_mb_wc_8bit(CHARSET_INFO *cs,my_wc_t *wc,
  180.   const unsigned char *str,
  181.   const unsigned char *end __attribute__((unused)))
  182. {
  183.   if (str >= end)
  184.     return MY_CS_TOOFEW(0);
  185.   
  186.   *wc=cs->tab_to_uni[*str];
  187.   return (!wc[0] && str[0]) ? MY_CS_ILSEQ : 1;
  188. }
  189. int my_wc_mb_8bit(CHARSET_INFO *cs,my_wc_t wc,
  190.   unsigned char *str,
  191.   unsigned char *end __attribute__((unused)))
  192. {
  193.   MY_UNI_IDX *idx;
  194.   if (str >= end)
  195.     return MY_CS_TOOSMALL;
  196.   
  197.   for (idx=cs->tab_from_uni; idx->tab ; idx++)
  198.   {
  199.     if (idx->from <= wc && idx->to >= wc)
  200.     {
  201.       str[0]= idx->tab[wc - idx->from];
  202.       return (!str[0] && wc) ? MY_CS_ILUNI : 1;
  203.     }
  204.   }
  205.   return MY_CS_ILUNI;
  206. }
  207. /* 
  208.    We can't use vsprintf here as it's not guaranteed to return
  209.    the length on all operating systems.
  210.    This function is also not called in a safe environment, so the
  211.    end buffer must be checked.
  212. */
  213. int my_snprintf_8bit(CHARSET_INFO *cs  __attribute__((unused)),
  214.      char* to, uint n  __attribute__((unused)),
  215.      const char* fmt, ...)
  216. {
  217.   va_list args;
  218.   int result;
  219.   va_start(args,fmt);
  220.   result= my_vsnprintf(to, n, fmt, args);
  221.   va_end(args);
  222.   return result;
  223. }
  224. void my_hash_sort_simple(CHARSET_INFO *cs,
  225.  const uchar *key, uint len,
  226.  ulong *nr1, ulong *nr2)
  227. {
  228.   register uchar *sort_order=cs->sort_order;
  229.   const uchar *pos = key;
  230.   
  231.   key+= len;
  232.   
  233.   for (; pos < (uchar*) key ; pos++)
  234.   {
  235.     nr1[0]^=(ulong) ((((uint) nr1[0] & 63)+nr2[0]) * 
  236.      ((uint) sort_order[(uint) *pos])) + (nr1[0] << 8);
  237.     nr2[0]+=3;
  238.   }
  239. }
  240. long my_strntol_8bit(CHARSET_INFO *cs,
  241.      const char *nptr, uint l, int base,
  242.      char **endptr, int *err)
  243. {
  244.   int negative;
  245.   register uint32 cutoff;
  246.   register unsigned int cutlim;
  247.   register uint32 i;
  248.   register const char *s;
  249.   register unsigned char c;
  250.   const char *save, *e;
  251.   int overflow;
  252.   *err= 0; /* Initialize error indicator */
  253. #ifdef NOT_USED
  254.   if (base < 0 || base == 1 || base > 36)
  255.     base = 10;
  256. #endif
  257.   s = nptr;
  258.   e = nptr+l;
  259.   
  260.   for ( ; s<e && my_isspace(cs, *s) ; s++);
  261.   
  262.   if (s == e)
  263.   {
  264.     goto noconv;
  265.   }
  266.   
  267.   /* Check for a sign. */
  268.   if (*s == '-')
  269.   {
  270.     negative = 1;
  271.     ++s;
  272.   }
  273.   else if (*s == '+')
  274.   {
  275.     negative = 0;
  276.     ++s;
  277.   }
  278.   else
  279.     negative = 0;
  280. #ifdef NOT_USED
  281.   if (base == 16 && s[0] == '0' && (s[1]=='X' || s[1]=='x'))
  282.     s += 2;
  283. #endif
  284. #ifdef NOT_USED
  285.   if (base == 0)
  286.   {
  287.     if (*s == '0')
  288.     {
  289.       if (s[1]=='X' || s[1]=='x')
  290.       {
  291. s += 2;
  292. base = 16;
  293.       }
  294.       else
  295. base = 8;
  296.     }
  297.     else
  298.       base = 10;
  299.   }
  300. #endif
  301.   save = s;
  302.   cutoff = ((uint32)~0L) / (uint32) base;
  303.   cutlim = (uint) (((uint32)~0L) % (uint32) base);
  304.   overflow = 0;
  305.   i = 0;
  306.   for (c = *s; s != e; c = *++s)
  307.   {
  308.     if (c>='0' && c<='9')
  309.       c -= '0';
  310.     else if (c>='A' && c<='Z')
  311.       c = c - 'A' + 10;
  312.     else if (c>='a' && c<='z')
  313.       c = c - 'a' + 10;
  314.     else
  315.       break;
  316.     if (c >= base)
  317.       break;
  318.     if (i > cutoff || (i == cutoff && c > cutlim))
  319.       overflow = 1;
  320.     else
  321.     {
  322.       i *= (uint32) base;
  323.       i += c;
  324.     }
  325.   }
  326.   
  327.   if (s == save)
  328.     goto noconv;
  329.   
  330.   if (endptr != NULL)
  331.     *endptr = (char *) s;
  332.   
  333.   if (negative)
  334.   {
  335.     if (i  > (uint32) INT_MIN32)
  336.       overflow = 1;
  337.   }
  338.   else if (i > INT_MAX32)
  339.     overflow = 1;
  340.   
  341.   if (overflow)
  342.   {
  343.     err[0]= ERANGE;
  344.     return negative ? INT_MIN32 : INT_MAX32;
  345.   }
  346.   
  347.   return (negative ? -((long) i) : (long) i);
  348. noconv:
  349.   err[0]= EDOM;
  350.   if (endptr != NULL)
  351.     *endptr = (char *) nptr;
  352.   return 0L;
  353. }
  354. ulong my_strntoul_8bit(CHARSET_INFO *cs,
  355.        const char *nptr, uint l, int base,
  356.        char **endptr, int *err)
  357. {
  358.   int negative;
  359.   register uint32 cutoff;
  360.   register unsigned int cutlim;
  361.   register uint32 i;
  362.   register const char *s;
  363.   register unsigned char c;
  364.   const char *save, *e;
  365.   int overflow;
  366.   *err= 0; /* Initialize error indicator */
  367. #ifdef NOT_USED
  368.   if (base < 0 || base == 1 || base > 36)
  369.     base = 10;
  370. #endif
  371.   s = nptr;
  372.   e = nptr+l;
  373.   
  374.   for( ; s<e && my_isspace(cs, *s); s++);
  375.   
  376.   if (s==e)
  377.   {
  378.     goto noconv;
  379.   }
  380.   if (*s == '-')
  381.   {
  382.     negative = 1;
  383.     ++s;
  384.   }
  385.   else if (*s == '+')
  386.   {
  387.     negative = 0;
  388.     ++s;
  389.   }
  390.   else
  391.     negative = 0;
  392. #ifdef NOT_USED
  393.   if (base == 16 && s[0] == '0' && (s[1]=='X' || s[1]=='x'))
  394.     s += 2;
  395. #endif
  396. #ifdef NOT_USED
  397.   if (base == 0)
  398.   {
  399.     if (*s == '0')
  400.     {
  401.       if (s[1]=='X' || s[1]=='x')
  402.       {
  403. s += 2;
  404. base = 16;
  405.       }
  406.       else
  407. base = 8;
  408.     }
  409.     else
  410.       base = 10;
  411.   }
  412. #endif
  413.   save = s;
  414.   cutoff = ((uint32)~0L) / (uint32) base;
  415.   cutlim = (uint) (((uint32)~0L) % (uint32) base);
  416.   overflow = 0;
  417.   i = 0;
  418.   
  419.   for (c = *s; s != e; c = *++s)
  420.   {
  421.     if (c>='0' && c<='9')
  422.       c -= '0';
  423.     else if (c>='A' && c<='Z')
  424.       c = c - 'A' + 10;
  425.     else if (c>='a' && c<='z')
  426.       c = c - 'a' + 10;
  427.     else
  428.       break;
  429.     if (c >= base)
  430.       break;
  431.     if (i > cutoff || (i == cutoff && c > cutlim))
  432.       overflow = 1;
  433.     else
  434.     {
  435.       i *= (uint32) base;
  436.       i += c;
  437.     }
  438.   }
  439.   if (s == save)
  440.     goto noconv;
  441.   if (endptr != NULL)
  442.     *endptr = (char *) s;
  443.   if (overflow)
  444.   {
  445.     err[0]= ERANGE;
  446.     return (~(uint32) 0);
  447.   }
  448.   
  449.   return (negative ? -((long) i) : (long) i);
  450.   
  451. noconv:
  452.   err[0]= EDOM;
  453.   if (endptr != NULL)
  454.     *endptr = (char *) nptr;
  455.   return 0L;
  456. }
  457. longlong my_strntoll_8bit(CHARSET_INFO *cs __attribute__((unused)),
  458.   const char *nptr, uint l, int base,
  459.   char **endptr,int *err)
  460. {
  461.   int negative;
  462.   register ulonglong cutoff;
  463.   register unsigned int cutlim;
  464.   register ulonglong i;
  465.   register const char *s, *e;
  466.   const char *save;
  467.   int overflow;
  468.   *err= 0; /* Initialize error indicator */
  469. #ifdef NOT_USED
  470.   if (base < 0 || base == 1 || base > 36)
  471.     base = 10;
  472. #endif
  473.   s = nptr;
  474.   e = nptr+l;
  475.   for(; s<e && my_isspace(cs,*s); s++);
  476.   if (s == e)
  477.   {
  478.     goto noconv;
  479.   }
  480.   if (*s == '-')
  481.   {
  482.     negative = 1;
  483.     ++s;
  484.   }
  485.   else if (*s == '+')
  486.   {
  487.     negative = 0;
  488.     ++s;
  489.   }
  490.   else
  491.     negative = 0;
  492. #ifdef NOT_USED
  493.   if (base == 16 && s[0] == '0' && (s[1]=='X'|| s[1]=='x'))
  494.     s += 2;
  495. #endif
  496. #ifdef NOT_USED
  497.   if (base == 0)
  498.   {
  499.     if (*s == '0')
  500.     {
  501.       if (s[1]=='X' || s[1]=='x')
  502.       {
  503. s += 2;
  504. base = 16;
  505.       }
  506.       else
  507. base = 8;
  508.     }
  509.     else
  510.       base = 10;
  511.   }
  512. #endif
  513.   save = s;
  514.   cutoff = (~(ulonglong) 0) / (unsigned long int) base;
  515.   cutlim = (uint) ((~(ulonglong) 0) % (unsigned long int) base);
  516.   overflow = 0;
  517.   i = 0;
  518.   for ( ; s != e; s++)
  519.   {
  520.     register unsigned char c= *s;
  521.     if (c>='0' && c<='9')
  522.       c -= '0';
  523.     else if (c>='A' && c<='Z')
  524.       c = c - 'A' + 10;
  525.     else if (c>='a' && c<='z')
  526.       c = c - 'a' + 10;
  527.     else
  528.       break;
  529.     if (c >= base)
  530.       break;
  531.     if (i > cutoff || (i == cutoff && c > cutlim))
  532.       overflow = 1;
  533.     else
  534.     {
  535.       i *= (ulonglong) base;
  536.       i += c;
  537.     }
  538.   }
  539.   if (s == save)
  540.     goto noconv;
  541.   if (endptr != NULL)
  542.     *endptr = (char *) s;
  543.   if (negative)
  544.   {
  545.     if (i  > (ulonglong) LONGLONG_MIN)
  546.       overflow = 1;
  547.   }
  548.   else if (i > (ulonglong) LONGLONG_MAX)
  549.     overflow = 1;
  550.   if (overflow)
  551.   {
  552.     err[0]= ERANGE;
  553.     return negative ? LONGLONG_MIN : LONGLONG_MAX;
  554.   }
  555.   return (negative ? -((longlong) i) : (longlong) i);
  556. noconv:
  557.   err[0]= EDOM;
  558.   if (endptr != NULL)
  559.     *endptr = (char *) nptr;
  560.   return 0L;
  561. }
  562. ulonglong my_strntoull_8bit(CHARSET_INFO *cs,
  563.    const char *nptr, uint l, int base,
  564.    char **endptr, int *err)
  565. {
  566.   int negative;
  567.   register ulonglong cutoff;
  568.   register unsigned int cutlim;
  569.   register ulonglong i;
  570.   register const char *s, *e;
  571.   const char *save;
  572.   int overflow;
  573.   *err= 0; /* Initialize error indicator */
  574. #ifdef NOT_USED
  575.   if (base < 0 || base == 1 || base > 36)
  576.     base = 10;
  577. #endif
  578.   s = nptr;
  579.   e = nptr+l;
  580.   for(; s<e && my_isspace(cs,*s); s++);
  581.   if (s == e)
  582.   {
  583.     goto noconv;
  584.   }
  585.   if (*s == '-')
  586.   {
  587.     negative = 1;
  588.     ++s;
  589.   }
  590.   else if (*s == '+')
  591.   {
  592.     negative = 0;
  593.     ++s;
  594.   }
  595.   else
  596.     negative = 0;
  597. #ifdef NOT_USED
  598.   if (base == 16 && s[0] == '0' && (s[1]=='X' || s[1]=='x'))
  599.     s += 2;
  600. #endif
  601. #ifdef NOT_USED
  602.   if (base == 0)
  603.   {
  604.     if (*s == '0')
  605.     {
  606.       if (s[1]=='X' || s[1]=='x')
  607.       {
  608. s += 2;
  609. base = 16;
  610.       }
  611.       else
  612. base = 8;
  613.     }
  614.     else
  615.       base = 10;
  616.   }
  617. #endif
  618.   save = s;
  619.   cutoff = (~(ulonglong) 0) / (unsigned long int) base;
  620.   cutlim = (uint) ((~(ulonglong) 0) % (unsigned long int) base);
  621.   overflow = 0;
  622.   i = 0;
  623.   for ( ; s != e; s++)
  624.   {
  625.     register unsigned char c= *s;
  626.     if (c>='0' && c<='9')
  627.       c -= '0';
  628.     else if (c>='A' && c<='Z')
  629.       c = c - 'A' + 10;
  630.     else if (c>='a' && c<='z')
  631.       c = c - 'a' + 10;
  632.     else
  633.       break;
  634.     if (c >= base)
  635.       break;
  636.     if (i > cutoff || (i == cutoff && c > cutlim))
  637.       overflow = 1;
  638.     else
  639.     {
  640.       i *= (ulonglong) base;
  641.       i += c;
  642.     }
  643.   }
  644.   if (s == save)
  645.     goto noconv;
  646.   if (endptr != NULL)
  647.     *endptr = (char *) s;
  648.   if (overflow)
  649.   {
  650.     err[0]= ERANGE;
  651.     return (~(ulonglong) 0);
  652.   }
  653.   return (negative ? -((longlong) i) : (longlong) i);
  654. noconv:
  655.   err[0]= EDOM;
  656.   if (endptr != NULL)
  657.     *endptr = (char *) nptr;
  658.   return 0L;
  659. }
  660. /*
  661.   Read double from string
  662.   SYNOPSIS:
  663.     my_strntod_8bit()
  664.     cs Character set information
  665.     str String to convert to double
  666.     length Optional length for string.
  667.     end result pointer to end of converted string
  668.     err Error number if failed conversion
  669.     
  670.   NOTES:
  671.     If length is not INT_MAX32 or str[length] != 0 then the given str must
  672.     be writeable
  673.     If length == INT_MAX32 the str must be  terminated.
  674.     It's implemented this way to save a buffer allocation and a memory copy.
  675.   RETURN
  676.     Value of number in string
  677. */
  678. double my_strntod_8bit(CHARSET_INFO *cs __attribute__((unused)),
  679.        char *str, uint length,
  680.        char **end, int *err)
  681. {
  682.   if (length == INT_MAX32)
  683.     length= 65535;                          /* Should be big enough */
  684.   *end= str + length;
  685.   return my_strtod(str, end, err);
  686. }
  687. /*
  688.   This is a fast version optimized for the case of radix 10 / -10
  689.   Assume len >= 1
  690. */
  691. int my_long10_to_str_8bit(CHARSET_INFO *cs __attribute__((unused)),
  692.      char *dst, uint len, int radix, long int val)
  693. {
  694.   char buffer[66];
  695.   register char *p, *e;
  696.   long int new_val;
  697.   uint sign=0;
  698.   e = p = &buffer[sizeof(buffer)-1];
  699.   *p= 0;
  700.   
  701.   if (radix < 0)
  702.   {
  703.     if (val < 0)
  704.     {
  705.       val= -val;
  706.       *dst++= '-';
  707.       len--;
  708.       sign= 1;
  709.     }
  710.   }
  711.   
  712.   new_val = (long) ((unsigned long int) val / 10);
  713.   *--p    = '0'+ (char) ((unsigned long int) val - (unsigned long) new_val * 10);
  714.   val     = new_val;
  715.   
  716.   while (val != 0)
  717.   {
  718.     new_val=val/10;
  719.     *--p = '0' + (char) (val-new_val*10);
  720.     val= new_val;
  721.   }
  722.   
  723.   len= min(len, (uint) (e-p));
  724.   memcpy(dst, p, len);
  725.   return (int) len+sign;
  726. }
  727. int my_longlong10_to_str_8bit(CHARSET_INFO *cs __attribute__((unused)),
  728.       char *dst, uint len, int radix, longlong val)
  729. {
  730.   char buffer[65];
  731.   register char *p, *e;
  732.   long long_val;
  733.   uint sign= 0;
  734.   
  735.   if (radix < 0)
  736.   {
  737.     if (val < 0)
  738.     {
  739.       val = -val;
  740.       *dst++= '-';
  741.       len--;
  742.       sign= 1;
  743.     }
  744.   }
  745.   
  746.   e = p = &buffer[sizeof(buffer)-1];
  747.   *p= 0;
  748.   
  749.   if (val == 0)
  750.   {
  751.     *--p= '0';
  752.     len= 1;
  753.     goto cnv;
  754.   }
  755.   
  756.   while ((ulonglong) val > (ulonglong) LONG_MAX)
  757.   {
  758.     ulonglong quo=(ulonglong) val/(uint) 10;
  759.     uint rem= (uint) (val- quo* (uint) 10);
  760.     *--p = '0' + rem;
  761.     val= quo;
  762.   }
  763.   
  764.   long_val= (long) val;
  765.   while (long_val != 0)
  766.   {
  767.     long quo= long_val/10;
  768.     *--p = (char) ('0' + (long_val - quo*10));
  769.     long_val= quo;
  770.   }
  771.   
  772.   len= min(len, (uint) (e-p));
  773. cnv:
  774.   memcpy(dst, p, len);
  775.   return len+sign;
  776. }
  777. /*
  778. ** Compare string against string with wildcard
  779. ** 0 if matched
  780. ** -1 if not matched with wildcard
  781. **  1 if matched with wildcard
  782. */
  783. #ifdef LIKE_CMP_TOUPPER
  784. #define likeconv(s,A) (uchar) my_toupper(s,A)
  785. #else
  786. #define likeconv(s,A) (uchar) (s)->sort_order[(uchar) (A)]
  787. #endif
  788. #define INC_PTR(cs,A,B) (A)++
  789. int my_wildcmp_8bit(CHARSET_INFO *cs,
  790.     const char *str,const char *str_end,
  791.     const char *wildstr,const char *wildend,
  792.     int escape, int w_one, int w_many)
  793. {
  794.   int result= -1; /* Not found, using wildcards */
  795.   while (wildstr != wildend)
  796.   {
  797.     while (*wildstr != w_many && *wildstr != w_one)
  798.     {
  799.       if (*wildstr == escape && wildstr+1 != wildend)
  800. wildstr++;
  801.       if (str == str_end || likeconv(cs,*wildstr++) != likeconv(cs,*str++))
  802. return(1); /* No match */
  803.       if (wildstr == wildend)
  804. return(str != str_end); /* Match if both are at end */
  805.       result=1; /* Found an anchor char     */
  806.     }
  807.     if (*wildstr == w_one)
  808.     {
  809.       do
  810.       {
  811. if (str == str_end) /* Skip one char if possible */
  812.   return(result);
  813. INC_PTR(cs,str,str_end);
  814.       } while (++wildstr < wildend && *wildstr == w_one);
  815.       if (wildstr == wildend)
  816. break;
  817.     }
  818.     if (*wildstr == w_many)
  819.     { /* Found w_many */
  820.       uchar cmp;
  821.       
  822.       wildstr++;
  823.       /* Remove any '%' and '_' from the wild search string */
  824.       for (; wildstr != wildend ; wildstr++)
  825.       {
  826. if (*wildstr == w_many)
  827.   continue;
  828. if (*wildstr == w_one)
  829. {
  830.   if (str == str_end)
  831.     return(-1);
  832.   INC_PTR(cs,str,str_end);
  833.   continue;
  834. }
  835. break; /* Not a wild character */
  836.       }
  837.       if (wildstr == wildend)
  838. return(0); /* Ok if w_many is last */
  839.       if (str == str_end)
  840. return(-1);
  841.       
  842.       if ((cmp= *wildstr) == escape && wildstr+1 != wildend)
  843. cmp= *++wildstr;
  844.       INC_PTR(cs,wildstr,wildend); /* This is compared trough cmp */
  845.       cmp=likeconv(cs,cmp);
  846.       do
  847.       {
  848. while (str != str_end && (uchar) likeconv(cs,*str) != cmp)
  849.   str++;
  850. if (str++ == str_end) return(-1);
  851. {
  852.   int tmp=my_wildcmp_8bit(cs,str,str_end,wildstr,wildend,escape,w_one,
  853.   w_many);
  854.   if (tmp <= 0)
  855.     return(tmp);
  856. }
  857.       } while (str != str_end && wildstr[0] != w_many);
  858.       return(-1);
  859.     }
  860.   }
  861.   return(str != str_end ? 1 : 0);
  862. }
  863. /*
  864. ** Calculate min_str and max_str that ranges a LIKE string.
  865. ** Arguments:
  866. ** ptr Pointer to LIKE string.
  867. ** ptr_length Length of LIKE string.
  868. ** escape Escape character in LIKE.  (Normally '').
  869. ** All escape characters should be removed from min_str and max_str
  870. ** res_length Length of min_str and max_str.
  871. ** min_str Smallest case sensitive string that ranges LIKE.
  872. ** Should be space padded to res_length.
  873. ** max_str Largest case sensitive string that ranges LIKE.
  874. ** Normally padded with the biggest character sort value.
  875. **
  876. ** The function should return 0 if ok and 1 if the LIKE string can't be
  877. ** optimized !
  878. */
  879. my_bool my_like_range_simple(CHARSET_INFO *cs,
  880.      const char *ptr,uint ptr_length,
  881.      pbool escape, pbool w_one, pbool w_many,
  882.      uint res_length,
  883.      char *min_str,char *max_str,
  884.      uint *min_length,uint *max_length)
  885. {
  886.   const char *end= ptr + ptr_length;
  887.   char *min_org=min_str;
  888.   char *min_end=min_str+res_length;
  889.   uint charlen= res_length / cs->mbmaxlen;
  890.   for (; ptr != end && min_str != min_end && charlen > 0 ; ptr++, charlen--)
  891.   {
  892.     if (*ptr == escape && ptr+1 != end)
  893.     {
  894.       ptr++; /* Skip escape */
  895.       *min_str++= *max_str++ = *ptr;
  896.       continue;
  897.     }
  898.     if (*ptr == w_one) /* '_' in SQL */
  899.     {
  900.       *min_str++=''; /* This should be min char */
  901.       *max_str++= (char) cs->max_sort_char;
  902.       continue;
  903.     }
  904.     if (*ptr == w_many) /* '%' in SQL */
  905.     {
  906.       *min_length= (uint) (min_str - min_org);
  907.       *max_length=res_length;
  908.       do
  909.       {
  910. *min_str++= 0;
  911. *max_str++= (char) cs->max_sort_char;
  912.       } while (min_str != min_end);
  913.       return 0;
  914.     }
  915.     *min_str++= *max_str++ = *ptr;
  916.   }
  917.   *min_length= *max_length = (uint) (min_str - min_org);
  918.   while (min_str != min_end)
  919.     *min_str++ = *max_str++ = ' '; /* Because if key compression */
  920.   return 0;
  921. }
  922. ulong my_scan_8bit(CHARSET_INFO *cs, const char *str, const char *end, int sq)
  923. {
  924.   const char *str0= str;
  925.   switch (sq)
  926.   {
  927.   case MY_SEQ_INTTAIL:
  928.     if (*str == '.')
  929.     {
  930.       for(str++ ; str != end && *str == '0' ; str++);
  931.       return (ulong) (str - str0);
  932.     }
  933.     return 0;
  934.   case MY_SEQ_SPACES:
  935.     for ( ; str < end ; str++)
  936.     {
  937.       if (!my_isspace(cs,*str))
  938.         break;
  939.     }
  940.     return (ulong) (str - str0);
  941.   default:
  942.     return 0;
  943.   }
  944. }
  945. void my_fill_8bit(CHARSET_INFO *cs __attribute__((unused)),
  946.    char *s, uint l, int fill)
  947. {
  948.   bfill(s,l,fill);
  949. }
  950. uint my_numchars_8bit(CHARSET_INFO *cs __attribute__((unused)),
  951.       const char *b, const char *e)
  952. {
  953.   return (uint) (e - b);
  954. }
  955. uint my_numcells_8bit(CHARSET_INFO *cs __attribute__((unused)),
  956.       const char *b, const char *e)
  957. {
  958.   return (uint) (e - b);
  959. }
  960. uint my_charpos_8bit(CHARSET_INFO *cs __attribute__((unused)),
  961.      const char *b  __attribute__((unused)),
  962.      const char *e  __attribute__((unused)),
  963.      uint pos)
  964. {
  965.   return pos;
  966. }
  967. uint my_well_formed_len_8bit(CHARSET_INFO *cs __attribute__((unused)),
  968.                              const char *start, const char *end,
  969.                              uint nchars, int *error)
  970. {
  971.   uint nbytes= (uint) (end-start);
  972.   *error= 0;
  973.   return min(nbytes, nchars);
  974. }
  975. uint my_lengthsp_8bit(CHARSET_INFO *cs __attribute__((unused)),
  976.       const char *ptr, uint length)
  977. {
  978.   const char *end= ptr+length;
  979.   while (end > ptr && end[-1] == ' ')
  980.     end--;
  981.   return (uint) (end-ptr);
  982. }
  983. uint my_instr_simple(CHARSET_INFO *cs,
  984.                     const char *b, uint b_length, 
  985.     const char *s, uint s_length,
  986.     my_match_t *match, uint nmatch)
  987. {
  988.   register const uchar *str, *search, *end, *search_end;
  989.   
  990.   if (s_length <= b_length)
  991.   {
  992.     if (!s_length)
  993.     {
  994.       if (nmatch)
  995.       {
  996.         match->beg= 0;
  997.         match->end= 0;
  998.         match->mblen= 0;
  999.       }
  1000.       return 1; /* Empty string is always found */
  1001.     }
  1002.     
  1003.     str= (const uchar*) b;
  1004.     search= (const uchar*) s;
  1005.     end= (const uchar*) b+b_length-s_length+1;
  1006.     search_end= (const uchar*) s + s_length;
  1007.     
  1008. skip:
  1009.     while (str != end)
  1010.     {
  1011.       if (cs->sort_order[*str++] == cs->sort_order[*search])
  1012.       {
  1013. register const uchar *i,*j;
  1014. i= str; 
  1015. j= search+1;
  1016. while (j != search_end)
  1017.   if (cs->sort_order[*i++] != cs->sort_order[*j++]) 
  1018.             goto skip;
  1019.         
  1020. if (nmatch > 0)
  1021. {
  1022.   match[0].beg= 0;
  1023.   match[0].end= str- (const uchar*)b-1;
  1024.   match[0].mblen= match[0].end;
  1025.   
  1026.   if (nmatch > 1)
  1027.   {
  1028.     match[1].beg= match[0].end;
  1029.     match[1].end= match[0].end+s_length;
  1030.     match[1].mblen= match[1].end-match[1].beg;
  1031.   }
  1032. }
  1033. return 2;
  1034.       }
  1035.     }
  1036.   }
  1037.   return 0;
  1038. }
  1039. typedef struct
  1040. {
  1041.   int nchars;
  1042.   MY_UNI_IDX uidx;
  1043. } uni_idx;
  1044. #define PLANE_SIZE 0x100
  1045. #define PLANE_NUM 0x100
  1046. #define PLANE_NUMBER(x) (((x)>>8) % PLANE_NUM)
  1047. static int pcmp(const void * f, const void * s)
  1048. {
  1049.   const uni_idx *F= (const uni_idx*) f;
  1050.   const uni_idx *S= (const uni_idx*) s;
  1051.   int res;
  1052.   if (!(res=((S->nchars)-(F->nchars))))
  1053.     res=((F->uidx.from)-(S->uidx.to));
  1054.   return res;
  1055. }
  1056. static my_bool create_fromuni(CHARSET_INFO *cs, void *(*alloc)(uint))
  1057. {
  1058.   uni_idx idx[PLANE_NUM];
  1059.   int i,n;
  1060.   
  1061.   /*
  1062.     Check that Unicode map is loaded.
  1063.     It can be not loaded when the collation is
  1064.     listed in Index.xml but not specified
  1065.     in the character set specific XML file.
  1066.   */
  1067.   if (!cs->tab_to_uni)
  1068.     return TRUE;
  1069.   
  1070.   /* Clear plane statistics */
  1071.   bzero(idx,sizeof(idx));
  1072.   
  1073.   /* Count number of characters in each plane */
  1074.   for (i=0; i< 0x100; i++)
  1075.   {
  1076.     uint16 wc=cs->tab_to_uni[i];
  1077.     int pl= PLANE_NUMBER(wc);
  1078.     
  1079.     if (wc || !i)
  1080.     {
  1081.       if (!idx[pl].nchars)
  1082.       {
  1083.         idx[pl].uidx.from=wc;
  1084.         idx[pl].uidx.to=wc;
  1085.       }else
  1086.       {
  1087.         idx[pl].uidx.from=wc<idx[pl].uidx.from?wc:idx[pl].uidx.from;
  1088.         idx[pl].uidx.to=wc>idx[pl].uidx.to?wc:idx[pl].uidx.to;
  1089.       }
  1090.       idx[pl].nchars++;
  1091.     }
  1092.   }
  1093.   
  1094.   /* Sort planes in descending order */
  1095.   qsort(&idx,PLANE_NUM,sizeof(uni_idx),&pcmp);
  1096.   
  1097.   for (i=0; i < PLANE_NUM; i++)
  1098.   {
  1099.     int ch,numchars;
  1100.     
  1101.     /* Skip empty plane */
  1102.     if (!idx[i].nchars)
  1103.       break;
  1104.     
  1105.     numchars=idx[i].uidx.to-idx[i].uidx.from+1;
  1106.     if (!(idx[i].uidx.tab=(uchar*) alloc(numchars * sizeof(*idx[i].uidx.tab))))
  1107.       return TRUE;
  1108.     
  1109.     bzero(idx[i].uidx.tab,numchars*sizeof(*idx[i].uidx.tab));
  1110.     
  1111.     for (ch=1; ch < PLANE_SIZE; ch++)
  1112.     {
  1113.       uint16 wc=cs->tab_to_uni[ch];
  1114.       if (wc >= idx[i].uidx.from && wc <= idx[i].uidx.to && wc)
  1115.       {
  1116.         int ofs= wc - idx[i].uidx.from;
  1117.         idx[i].uidx.tab[ofs]= ch;
  1118.       }
  1119.     }
  1120.   }
  1121.   
  1122.   /* Allocate and fill reverse table for each plane */
  1123.   n=i;
  1124.   if (!(cs->tab_from_uni= (MY_UNI_IDX*) alloc(sizeof(MY_UNI_IDX)*(n+1))))
  1125.     return TRUE;
  1126.   for (i=0; i< n; i++)
  1127.     cs->tab_from_uni[i]= idx[i].uidx;
  1128.   
  1129.   /* Set end-of-list marker */
  1130.   bzero(&cs->tab_from_uni[i],sizeof(MY_UNI_IDX));
  1131.   return FALSE;
  1132. }
  1133. static my_bool my_cset_init_8bit(CHARSET_INFO *cs, void *(*alloc)(uint))
  1134. {
  1135.   return create_fromuni(cs, alloc);
  1136. }
  1137. static void set_max_sort_char(CHARSET_INFO *cs)
  1138. {
  1139.   uchar max_char;
  1140.   uint  i;
  1141.   
  1142.   if (!cs->sort_order)
  1143.     return;
  1144.   
  1145.   max_char=cs->sort_order[(uchar) cs->max_sort_char];
  1146.   for (i= 0; i < 256; i++)
  1147.   {
  1148.     if ((uchar) cs->sort_order[i] > max_char)
  1149.     {
  1150.       max_char=(uchar) cs->sort_order[i];
  1151.       cs->max_sort_char= i;
  1152.     }
  1153.   }
  1154. }
  1155. static my_bool my_coll_init_simple(CHARSET_INFO *cs,
  1156.                                    void *(*alloc)(uint) __attribute__((unused)))
  1157. {
  1158.   set_max_sort_char(cs);
  1159.   return FALSE;
  1160. }
  1161. longlong my_strtoll10_8bit(CHARSET_INFO *cs __attribute__((unused)),
  1162.                            const char *nptr, char **endptr, int *error)
  1163. {
  1164.   return my_strtoll10(nptr, endptr, error);
  1165. }
  1166. MY_CHARSET_HANDLER my_charset_8bit_handler=
  1167. {
  1168.     my_cset_init_8bit,
  1169.     NULL, /* ismbchar      */
  1170.     my_mbcharlen_8bit, /* mbcharlen     */
  1171.     my_numchars_8bit,
  1172.     my_charpos_8bit,
  1173.     my_well_formed_len_8bit,
  1174.     my_lengthsp_8bit,
  1175.     my_numcells_8bit,
  1176.     my_mb_wc_8bit,
  1177.     my_wc_mb_8bit,
  1178.     my_caseup_str_8bit,
  1179.     my_casedn_str_8bit,
  1180.     my_caseup_8bit,
  1181.     my_casedn_8bit,
  1182.     my_snprintf_8bit,
  1183.     my_long10_to_str_8bit,
  1184.     my_longlong10_to_str_8bit,
  1185.     my_fill_8bit,
  1186.     my_strntol_8bit,
  1187.     my_strntoul_8bit,
  1188.     my_strntoll_8bit,
  1189.     my_strntoull_8bit,
  1190.     my_strntod_8bit,
  1191.     my_strtoll10_8bit,
  1192.     my_scan_8bit
  1193. };
  1194. MY_COLLATION_HANDLER my_collation_8bit_simple_ci_handler =
  1195. {
  1196.     my_coll_init_simple, /* init */
  1197.     my_strnncoll_simple,
  1198.     my_strnncollsp_simple,
  1199.     my_strnxfrm_simple,
  1200.     my_like_range_simple,
  1201.     my_wildcmp_8bit,
  1202.     my_strcasecmp_8bit,
  1203.     my_instr_simple,
  1204.     my_hash_sort_simple
  1205. };