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

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.    You should have received a copy of the GNU Library General Public
  13.    License along with this library; if not, write to the Free
  14.    Software Foundation, Inc., 59 Temple Place - Suite 330, Boston,
  15.    MA 02111-1307, USA */
  16. #include "my_handler.h"
  17. int mi_compare_text(CHARSET_INFO *charset_info, uchar *a, uint a_length,
  18.     uchar *b, uint b_length, my_bool part_key,
  19.     my_bool skip_end_space)
  20. {
  21.   if (skip_end_space)
  22.     return charset_info->coll->strnncollsp(charset_info, a, a_length,
  23.    b, b_length);
  24.   return charset_info->coll->strnncoll(charset_info, a, a_length,
  25.        b, b_length, part_key);
  26. }
  27. static int compare_bin(uchar *a, uint a_length, uchar *b, uint b_length,
  28.                        my_bool part_key, my_bool skip_end_space)
  29. {
  30.   uint length= min(a_length,b_length);
  31.   uchar *end= a+ length;
  32.   int flag;
  33.   while (a < end)
  34.     if ((flag= (int) *a++ - (int) *b++))
  35.       return flag;
  36.   if (part_key && b_length < a_length)
  37.     return 0;
  38.   if (skip_end_space && a_length != b_length)
  39.   {
  40.     int swap= 1;
  41.     /*
  42.       We are using space compression. We have to check if longer key
  43.       has next character < ' ', in which case it's less than the shorter
  44.       key that has an implicite space afterwards.
  45.       This code is identical to the one in
  46.       strings/ctype-simple.c:my_strnncollsp_simple
  47.     */
  48.     if (a_length < b_length)
  49.     {
  50.       /* put shorter key in a */
  51.       a_length= b_length;
  52.       a= b;
  53.       swap= -1; /* swap sign of result */
  54.     }
  55.     for (end= a + a_length-length; a < end ; a++)
  56.     {
  57.       if (*a != ' ')
  58. return (*a < ' ') ? -swap : swap;
  59.     }
  60.     return 0;
  61.   }
  62.   return (int) (a_length-b_length);
  63. }
  64. /*
  65.   Compare two keys
  66.   SYNOPSIS
  67.     ha_key_cmp()
  68.     keyseg Array of key segments of key to compare
  69.     a First key to compare, in format from _mi_pack_key()
  70. This is normally key specified by user
  71.     b Second key to compare.  This is always from a row
  72.     key_length Length of key to compare.  This can be shorter than
  73. a to just compare sub keys
  74.     next_flag How keys should be compared
  75. If bit SEARCH_FIND is not set the keys includes the row
  76. position and this should also be compared
  77.     diff_pos    OUT Number of first keypart where values differ, counting 
  78.                 from one.
  79.     diff_pos[1] OUT  (b + diff_pos[1]) points to first value in tuple b
  80.                       that is different from corresponding value in tuple a.
  81.   
  82.   EXAMPLES 
  83.    Example1: if the function is called for tuples
  84.      ('aaa','bbb') and ('eee','fff'), then
  85.      diff_pos[0] = 1 (as 'aaa' != 'eee')
  86.      diff_pos[1] = 0 (offset from beggining of tuple b to 'eee' keypart).
  87.    Example2: if the index function is called for tuples
  88.      ('aaa','bbb') and ('aaa','fff'),
  89.      diff_pos[0] = 2 (as 'aaa' != 'eee')
  90.      diff_pos[1] = 3 (offset from beggining of tuple b to 'fff' keypart,
  91.                       here we assume that first key part is CHAR(3) NOT NULL)
  92.   NOTES
  93.     Number-keys can't be splited
  94.   RETURN VALUES
  95.     <0 If a < b
  96.     0 If a == b
  97.     >0 If a > b
  98. */
  99. #define FCMP(A,B) ((int) (A) - (int) (B))
  100. int ha_key_cmp(register HA_KEYSEG *keyseg, register uchar *a,
  101.        register uchar *b, uint key_length, uint nextflag,
  102.        uint *diff_pos)
  103. {
  104.   int flag;
  105.   int16 s_1,s_2;
  106.   int32 l_1,l_2;
  107.   uint32 u_1,u_2;
  108.   float f_1,f_2;
  109.   double d_1,d_2;
  110.   uint next_key_length;
  111.   uchar *orig_b= b;
  112.   *diff_pos=0;
  113.   for ( ; (int) key_length >0 ; key_length=next_key_length, keyseg++)
  114.   {
  115.     uchar *end;
  116.     uint piks=! (keyseg->flag & HA_NO_SORT);
  117.     (*diff_pos)++;
  118.     diff_pos[1]= (uint)(b - orig_b);
  119.     /* Handle NULL part */
  120.     if (keyseg->null_bit)
  121.     {
  122.       key_length--;
  123.       if (*a != *b && piks)
  124.       {
  125.         flag = (int) *a - (int) *b;
  126.         return ((keyseg->flag & HA_REVERSE_SORT) ? -flag : flag);
  127.       }
  128.       b++;
  129.       if (!*a++)                                /* If key was NULL */
  130.       {
  131.         if (nextflag == (SEARCH_FIND | SEARCH_UPDATE))
  132.           nextflag=SEARCH_SAME;                 /* Allow duplicate keys */
  133.    else if (nextflag & SEARCH_NULL_ARE_NOT_EQUAL)
  134. {
  135.   /*
  136.     This is only used from mi_check() to calculate cardinality.
  137.     It can't be used when searching for a key as this would cause
  138.     compare of (a,b) and (b,a) to return the same value.
  139.   */
  140.   return -1;
  141. }
  142.         next_key_length=key_length;
  143.         continue;                               /* To next key part */
  144.       }
  145.     }
  146.     end= a+ min(keyseg->length,key_length);
  147.     next_key_length=key_length-keyseg->length;
  148.     switch ((enum ha_base_keytype) keyseg->type) {
  149.     case HA_KEYTYPE_TEXT:                       /* Ascii; Key is converted */
  150.       if (keyseg->flag & HA_SPACE_PACK)
  151.       {
  152.         int a_length,b_length,pack_length;
  153.         get_key_length(a_length,a);
  154.         get_key_pack_length(b_length,pack_length,b);
  155.         next_key_length=key_length-b_length-pack_length;
  156.         if (piks &&
  157.             (flag=mi_compare_text(keyseg->charset,a,a_length,b,b_length,
  158.   (my_bool) ((nextflag & SEARCH_PREFIX) &&
  159.      next_key_length <= 0),
  160.   (my_bool)!(nextflag & SEARCH_PREFIX))))
  161.           return ((keyseg->flag & HA_REVERSE_SORT) ? -flag : flag);
  162.         a+=a_length;
  163.         b+=b_length;
  164.         break;
  165.       }
  166.       else
  167.       {
  168. uint length=(uint) (end-a), a_length=length, b_length=length;
  169.         if (piks &&
  170.             (flag= mi_compare_text(keyseg->charset, a, a_length, b, b_length,
  171.    (my_bool) ((nextflag & SEARCH_PREFIX) &&
  172.       next_key_length <= 0),
  173.    (my_bool)!(nextflag & SEARCH_PREFIX))))
  174.           return ((keyseg->flag & HA_REVERSE_SORT) ? -flag : flag);
  175.         a=end;
  176.         b+=length;
  177.       }
  178.       break;
  179.     case HA_KEYTYPE_BINARY:
  180.       if (keyseg->flag & HA_SPACE_PACK)
  181.       {
  182.         int a_length,b_length,pack_length;
  183.         get_key_length(a_length,a);
  184.         get_key_pack_length(b_length,pack_length,b);
  185.         next_key_length=key_length-b_length-pack_length;
  186.         if (piks &&
  187.     (flag=compare_bin(a,a_length,b,b_length,
  188.                               (my_bool) ((nextflag & SEARCH_PREFIX) &&
  189.                                          next_key_length <= 0),1)))
  190.           return ((keyseg->flag & HA_REVERSE_SORT) ? -flag : flag);
  191.         a+=a_length;
  192.         b+=b_length;
  193.         break;
  194.       }
  195.       else
  196.       {
  197.         uint length=keyseg->length;
  198.         if (piks &&
  199.     (flag=compare_bin(a,length,b,length,
  200.                               (my_bool) ((nextflag & SEARCH_PREFIX) &&
  201.                                          next_key_length <= 0),0)))
  202.           return ((keyseg->flag & HA_REVERSE_SORT) ? -flag : flag);
  203.         a+=length;
  204.         b+=length;
  205.       }
  206.       break;
  207.     case HA_KEYTYPE_VARTEXT:
  208.       {
  209.         int a_length,full_a_length,b_length,full_b_length,pack_length;
  210.         get_key_length(a_length,a);
  211.         get_key_pack_length(b_length,pack_length,b);
  212.         full_a_length= a_length;
  213.         full_b_length= b_length;
  214.         next_key_length=key_length-b_length-pack_length;
  215.         if (piks &&
  216.     (flag= mi_compare_text(keyseg->charset,a,a_length,b,b_length,
  217.                                    (my_bool) ((nextflag & SEARCH_PREFIX) &&
  218.                                               next_key_length <= 0),
  219.    (my_bool) ((nextflag & (SEARCH_FIND |
  220.    SEARCH_UPDATE)) ==
  221.       SEARCH_FIND))))
  222.           return ((keyseg->flag & HA_REVERSE_SORT) ? -flag : flag);
  223.         a+= full_a_length;
  224.         b+= full_b_length;
  225.         break;
  226.       }
  227.       break;
  228.     case HA_KEYTYPE_VARBINARY:
  229.       {
  230.         int a_length,b_length,pack_length;
  231.         get_key_length(a_length,a);
  232.         get_key_pack_length(b_length,pack_length,b);
  233.         next_key_length=key_length-b_length-pack_length;
  234.         if (piks &&
  235.     (flag=compare_bin(a,a_length,b,b_length,
  236.                               (my_bool) ((nextflag & SEARCH_PREFIX) &&
  237.                                          next_key_length <= 0), 0)))
  238.           return ((keyseg->flag & HA_REVERSE_SORT) ? -flag : flag);
  239.         a+=a_length;
  240.         b+=b_length;
  241.         break;
  242.       }
  243.       break;
  244.     case HA_KEYTYPE_INT8:
  245.     {
  246.       int i_1= (int) *((signed char*) a);
  247.       int i_2= (int) *((signed char*) b);
  248.       if (piks && (flag = CMP_NUM(i_1,i_2)))
  249.         return ((keyseg->flag & HA_REVERSE_SORT) ? -flag : flag);
  250.       a= end;
  251.       b++;
  252.       break;
  253.     }
  254.     case HA_KEYTYPE_SHORT_INT:
  255.       s_1= mi_sint2korr(a);
  256.       s_2= mi_sint2korr(b);
  257.       if (piks && (flag = CMP_NUM(s_1,s_2)))
  258.         return ((keyseg->flag & HA_REVERSE_SORT) ? -flag : flag);
  259.       a=  end;
  260.       b+= 2; /* sizeof(short int); */
  261.       break;
  262.     case HA_KEYTYPE_USHORT_INT:
  263.       {
  264.         uint16 us_1,us_2;
  265.         us_1= mi_sint2korr(a);
  266.         us_2= mi_sint2korr(b);
  267.         if (piks && (flag = CMP_NUM(us_1,us_2)))
  268.           return ((keyseg->flag & HA_REVERSE_SORT) ? -flag : flag);
  269.         a=  end;
  270.         b+=2; /* sizeof(short int); */
  271.         break;
  272.       }
  273.     case HA_KEYTYPE_LONG_INT:
  274.       l_1= mi_sint4korr(a);
  275.       l_2= mi_sint4korr(b);
  276.       if (piks && (flag = CMP_NUM(l_1,l_2)))
  277.         return ((keyseg->flag & HA_REVERSE_SORT) ? -flag : flag);
  278.       a=  end;
  279.       b+= 4; /* sizeof(long int); */
  280.       break;
  281.     case HA_KEYTYPE_ULONG_INT:
  282.       u_1= mi_sint4korr(a);
  283.       u_2= mi_sint4korr(b);
  284.       if (piks && (flag = CMP_NUM(u_1,u_2)))
  285.         return ((keyseg->flag & HA_REVERSE_SORT) ? -flag : flag);
  286.       a=  end;
  287.       b+= 4; /* sizeof(long int); */
  288.       break;
  289.     case HA_KEYTYPE_INT24:
  290.       l_1=mi_sint3korr(a);
  291.       l_2=mi_sint3korr(b);
  292.       if (piks && (flag = CMP_NUM(l_1,l_2)))
  293.         return ((keyseg->flag & HA_REVERSE_SORT) ? -flag : flag);
  294.       a=  end;
  295.       b+= 3;
  296.       break;
  297.     case HA_KEYTYPE_UINT24:
  298.       l_1=mi_uint3korr(a);
  299.       l_2=mi_uint3korr(b);
  300.       if (piks && (flag = CMP_NUM(l_1,l_2)))
  301.         return ((keyseg->flag & HA_REVERSE_SORT) ? -flag : flag);
  302.       a=  end;
  303.       b+= 3;
  304.       break;
  305.     case HA_KEYTYPE_FLOAT:
  306.       mi_float4get(f_1,a);
  307.       mi_float4get(f_2,b);
  308.       /*
  309.         The following may give a compiler warning about floating point
  310.         comparison not being safe, but this is ok in this context as
  311.         we are bascily doing sorting
  312.       */
  313.       if (piks && (flag = CMP_NUM(f_1,f_2)))
  314.         return ((keyseg->flag & HA_REVERSE_SORT) ? -flag : flag);
  315.       a=  end;
  316.       b+= 4; /* sizeof(float); */
  317.       break;
  318.     case HA_KEYTYPE_DOUBLE:
  319.       mi_float8get(d_1,a);
  320.       mi_float8get(d_2,b);
  321.       /*
  322.         The following may give a compiler warning about floating point
  323.         comparison not being safe, but this is ok in this context as
  324.         we are bascily doing sorting
  325.       */
  326.       if (piks && (flag = CMP_NUM(d_1,d_2)))
  327.         return ((keyseg->flag & HA_REVERSE_SORT) ? -flag : flag);
  328.       a=  end;
  329.       b+= 8;  /* sizeof(double); */
  330.       break;
  331.     case HA_KEYTYPE_NUM:                                /* Numeric key */
  332.     {
  333.       int swap_flag= 0;
  334.       int alength,blength;
  335.       if (keyseg->flag & HA_REVERSE_SORT)
  336.       {
  337.         swap_variables(uchar*, a, b);
  338.         swap_flag=1;                            /* Remember swap of a & b */
  339.         end= a+ (int) (end-b);
  340.       }
  341.       if (keyseg->flag & HA_SPACE_PACK)
  342.       {
  343.         alength= *a++; blength= *b++;
  344.         end=a+alength;
  345.         next_key_length=key_length-blength-1;
  346.       }
  347.       else
  348.       {
  349.         alength= (int) (end-a);
  350.         blength=keyseg->length;
  351.         /* remove pre space from keys */
  352.         for ( ; alength && *a == ' ' ; a++, alength--) ;
  353.         for ( ; blength && *b == ' ' ; b++, blength--) ;
  354.       }
  355.       if (piks)
  356.       {
  357. if (*a == '-')
  358. {
  359.   if (*b != '-')
  360.     return -1;
  361.   a++; b++;
  362.   swap_variables(uchar*, a, b);
  363.   swap_variables(int, alength, blength);
  364.   swap_flag=1-swap_flag;
  365.   alength--; blength--;
  366.   end=a+alength;
  367. }
  368. else if (*b == '-')
  369.   return 1;
  370. while (alength && (*a == '+' || *a == '0'))
  371. {
  372.   a++; alength--;
  373. }
  374. while (blength && (*b == '+' || *b == '0'))
  375. {
  376.   b++; blength--;
  377. }
  378. if (alength != blength)
  379.   return (alength < blength) ? -1 : 1;
  380. while (a < end)
  381.   if (*a++ !=  *b++)
  382.     return ((int) a[-1] - (int) b[-1]);
  383.       }
  384.       else
  385.       {
  386.         b+=(end-a);
  387.         a=end;
  388.       }
  389.       if (swap_flag)                            /* Restore pointers */
  390.         swap_variables(uchar*, a, b);
  391.       break;
  392.     }
  393. #ifdef HAVE_LONG_LONG
  394.     case HA_KEYTYPE_LONGLONG:
  395.     {
  396.       longlong ll_a,ll_b;
  397.       ll_a= mi_sint8korr(a);
  398.       ll_b= mi_sint8korr(b);
  399.       if (piks && (flag = CMP_NUM(ll_a,ll_b)))
  400.         return ((keyseg->flag & HA_REVERSE_SORT) ? -flag : flag);
  401.       a=  end;
  402.       b+= 8;
  403.       break;
  404.     }
  405.     case HA_KEYTYPE_ULONGLONG:
  406.     {
  407.       ulonglong ll_a,ll_b;
  408.       ll_a= mi_uint8korr(a);
  409.       ll_b= mi_uint8korr(b);
  410.       if (piks && (flag = CMP_NUM(ll_a,ll_b)))
  411.         return ((keyseg->flag & HA_REVERSE_SORT) ? -flag : flag);
  412.       a=  end;
  413.       b+= 8;
  414.       break;
  415.     }
  416. #endif
  417.     case HA_KEYTYPE_END:                        /* Ready */
  418.       goto end;                                 /* diff_pos is incremented */
  419.     }
  420.   }
  421.   (*diff_pos)++;
  422. end:
  423.   if (!(nextflag & SEARCH_FIND))
  424.   {
  425.     uint i;
  426.     if (nextflag & (SEARCH_NO_FIND | SEARCH_LAST)) /* Find record after key */
  427.       return (nextflag & (SEARCH_BIGGER | SEARCH_LAST)) ? -1 : 1;
  428.     flag=0;
  429.     for (i=keyseg->length ; i-- > 0 ; )
  430.     {
  431.       if (*a++ != *b++)
  432.       {
  433.         flag= FCMP(a[-1],b[-1]);
  434.         break;
  435.       }
  436.     }
  437.     if (nextflag & SEARCH_SAME)
  438.       return (flag);                            /* read same */
  439.     if (nextflag & SEARCH_BIGGER)
  440.       return (flag <= 0 ? -1 : 1);              /* read next */
  441.     return (flag < 0 ? -1 : 1);                 /* read previous */
  442.   }
  443.   return 0;
  444. } /* ha_key_cmp */
  445. /*
  446.   Find the first NULL value in index-suffix values tuple
  447.   SYNOPSIS
  448.     ha_find_null()
  449.       keyseg     Array of keyparts for key suffix
  450.       a          Key suffix value tuple
  451.   DESCRIPTION
  452.     Find the first NULL value in index-suffix values tuple.
  453.     TODO Consider optimizing this fuction or its use so we don't search for
  454.          NULL values in completely NOT NULL index suffixes.
  455.   RETURN
  456.     First key part that has NULL as value in values tuple, or the last key part 
  457.     (with keyseg->type==HA_TYPE_END) if values tuple doesn't contain NULLs.
  458. */
  459. HA_KEYSEG *ha_find_null(HA_KEYSEG *keyseg, uchar *a)
  460. {
  461.   for (; (enum ha_base_keytype) keyseg->type != HA_KEYTYPE_END; keyseg++)
  462.   {
  463.     uchar *end;
  464.     if (keyseg->null_bit)
  465.     {
  466.       if (!*a++)
  467.         return keyseg;
  468.     }
  469.     end= a+ keyseg->length;
  470.     switch ((enum ha_base_keytype) keyseg->type) {
  471.     case HA_KEYTYPE_TEXT:
  472.     case HA_KEYTYPE_BINARY:
  473.       if (keyseg->flag & HA_SPACE_PACK)
  474.       {
  475.         int a_length;
  476.         get_key_length(a_length, a);
  477.         a += a_length;
  478.         break;
  479.       }
  480.       else
  481.         a= end;
  482.       break;
  483.     case HA_KEYTYPE_VARTEXT:
  484.     case HA_KEYTYPE_VARBINARY:
  485.       {
  486.         int a_length;
  487.         get_key_length(a_length, a);
  488.         a+= a_length;
  489.         break;
  490.       }
  491.     case HA_KEYTYPE_NUM:
  492.       if (keyseg->flag & HA_SPACE_PACK)
  493.       {
  494.         int alength= *a++;
  495.         end= a+alength;
  496.       }
  497.       a= end;
  498.       break;
  499.     case HA_KEYTYPE_INT8:
  500.     case HA_KEYTYPE_SHORT_INT:
  501.     case HA_KEYTYPE_USHORT_INT:
  502.     case HA_KEYTYPE_LONG_INT:
  503.     case HA_KEYTYPE_ULONG_INT:
  504.     case HA_KEYTYPE_INT24:
  505.     case HA_KEYTYPE_UINT24:
  506. #ifdef HAVE_LONG_LONG
  507.     case HA_KEYTYPE_LONGLONG:
  508.     case HA_KEYTYPE_ULONGLONG:
  509. #endif
  510.     case HA_KEYTYPE_FLOAT:
  511.     case HA_KEYTYPE_DOUBLE:
  512.       a= end;
  513.       break;
  514.     }
  515.   }
  516.   return keyseg;
  517. }