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

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. /* Quick & light hash implementation for tab completion purposes
  18.  *
  19.  * by  Andi Gutmans <andi@zend.com>
  20.  * and Zeev Suraski <zeev@zend.com>
  21.  * Small portability changes by Monty. Changed also to use my_malloc/my_free
  22.  */
  23. #include <global.h>
  24. #include <m_string.h>
  25. #undef SAFEMALLOC // Speed things up
  26. #include <my_sys.h>
  27. #include "completion_hash.h"
  28. uint hashpjw(char *arKey, uint nKeyLength)
  29. {
  30.   uint h = 0, g, i;
  31.   for (i = 0; i < nKeyLength; i++) {
  32.     h = (h << 4) + arKey[i];
  33.     if ((g = (h & 0xF0000000))) {
  34.       h = h ^ (g >> 24);
  35.       h = h ^ g;
  36.     }
  37.   }
  38.   return h;
  39. }
  40. int completion_hash_init(HashTable *ht, uint nSize)
  41. {
  42.   ht->arBuckets = (Bucket **) my_malloc(nSize* sizeof(Bucket *),
  43. MYF(MY_ZEROFILL | MY_WME));
  44.   if (!ht->arBuckets) {
  45.     ht->initialized = 0;
  46.     return FAILURE;
  47.   }
  48.   ht->pHashFunction = hashpjw;
  49.   ht->nTableSize = nSize;
  50.   ht->initialized = 1;
  51.   return SUCCESS;
  52. }
  53. int completion_hash_update(HashTable *ht, char *arKey, uint nKeyLength,
  54.    char *str)
  55. {
  56.   uint h, nIndex;
  57.   Bucket *p;
  58.   h = ht->pHashFunction(arKey, nKeyLength);
  59.   nIndex = h % ht->nTableSize;
  60.   if (nKeyLength <= 0) {
  61.     return FAILURE;
  62.   }
  63.   p = ht->arBuckets[nIndex];
  64.   while (p)
  65.   {
  66.     if ((p->h == h) && (p->nKeyLength == nKeyLength)) {
  67.       if (!memcmp(p->arKey, arKey, nKeyLength)) {
  68. entry *n;
  69. n = (entry *) my_malloc(sizeof(entry),
  70. MYF(MY_WME));
  71. n->pNext = p->pData;
  72. n->str = str;
  73. p->pData = n;
  74. p->count++;
  75. return SUCCESS;
  76.       }
  77.     }
  78.     p = p->pNext;
  79.   }
  80.   p = (Bucket *) my_malloc(sizeof(Bucket),MYF(MY_WME));
  81.   if (!p) {
  82.     return FAILURE;
  83.   }
  84.   p->arKey = arKey;
  85.   p->nKeyLength = nKeyLength;
  86.   p->h = h;
  87.   p->pData = (entry*) my_malloc(sizeof(entry),MYF(MY_WME));
  88.   if (!p->pData) {
  89.     my_free((gptr) p,MYF(0));
  90.     return FAILURE;
  91.   }
  92.   p->pData->str = str;
  93.   p->pData->pNext = 0;
  94.   p->count = 1;
  95.   p->pNext = ht->arBuckets[nIndex];
  96.   ht->arBuckets[nIndex] = p;
  97.   return SUCCESS;
  98. }
  99. static Bucket *completion_hash_find(HashTable *ht, char *arKey,
  100.     uint nKeyLength)
  101. {
  102.   uint h, nIndex;
  103.   Bucket *p;
  104.   h = ht->pHashFunction(arKey, nKeyLength);
  105.   nIndex = h % ht->nTableSize;
  106.   p = ht->arBuckets[nIndex];
  107.   while (p)
  108.   {
  109.     if ((p->h == h) && (p->nKeyLength == nKeyLength)) {
  110.       if (!memcmp(p->arKey, arKey, nKeyLength)) {
  111. return p;
  112.       }
  113.     }
  114.     p = p->pNext;
  115.   }
  116.   return (Bucket*) 0;
  117. }
  118. int completion_hash_exists(HashTable *ht, char *arKey, uint nKeyLength)
  119. {
  120.   uint h, nIndex;
  121.   Bucket *p;
  122.   h = ht->pHashFunction(arKey, nKeyLength);
  123.   nIndex = h % ht->nTableSize;
  124.   p = ht->arBuckets[nIndex];
  125.   while (p)
  126.   {
  127.     if ((p->h == h) && (p->nKeyLength == nKeyLength))
  128.     {
  129.       if (!strcmp(p->arKey, arKey)) {
  130. return 1;
  131.       }
  132.     }
  133.     p = p->pNext;
  134.   }
  135.   return 0;
  136. }
  137. Bucket *find_all_matches(HashTable *ht, char *str, uint length,
  138.  uint *res_length)
  139. {
  140.   Bucket *b;
  141.   b = completion_hash_find(ht,str,length);
  142.   if (!b) {
  143.     *res_length = 0;
  144.     return (Bucket*) 0;
  145.   } else {
  146.     *res_length = length;
  147.     return b;
  148.   }
  149. }
  150. Bucket *find_longest_match(HashTable *ht, char *str, uint length,
  151.    uint *res_length)
  152. {
  153.   Bucket *b,*return_b;
  154.   char *s;
  155.   uint count;
  156.   uint lm;
  157.   b = completion_hash_find(ht,str,length);
  158.   if (!b) {
  159.     *res_length = 0;
  160.     return (Bucket*) 0;
  161.   }
  162.   count = b->count;
  163.   lm = length;
  164.   s = b->pData->str;
  165.   return_b = b;
  166.   while (s[lm]!=0 && (b=completion_hash_find(ht,s,lm+1))) {
  167.     if (b->count<count) {
  168.       *res_length=lm;
  169.       return return_b;
  170.     }
  171.     return_b=b;
  172.     lm++;
  173.   }
  174.   *res_length=lm;
  175.   return return_b;
  176. }
  177. void completion_hash_clean(HashTable *ht)
  178. {
  179.   uint i;
  180.   entry *e, *t;
  181.   Bucket *b, *tmp;
  182.   for (i=0; i<ht->nTableSize; i++) {
  183.     b = ht->arBuckets[i];
  184.     while (b) {
  185.       e =  b->pData;
  186.       while (e) {
  187. t = e;
  188. e = e->pNext;
  189. my_free((gptr) t,MYF(0));
  190.       }
  191.       tmp = b;
  192.       b = b->pNext;
  193.       my_free((gptr) tmp,MYF(0));
  194.     }
  195.   }
  196.   bzero((char*) ht->arBuckets,ht->nTableSize*sizeof(Bucket *));
  197. }
  198. void completion_hash_free(HashTable *ht)
  199. {
  200.   completion_hash_clean(ht);
  201.   my_free((gptr) ht->arBuckets,MYF(0));
  202. }
  203. void add_word(HashTable *ht,char *str)
  204. {
  205.   int i;
  206.   int length= (int) strlen(str);
  207.   for (i=1; i<=length; i++) {
  208.     completion_hash_update(ht, str, i, str);
  209.   }
  210. }