mirror of
				https://github.com/MariaDB/server.git
				synced 2025-10-26 08:28:13 +01:00 
			
		
		
		
	 dbfee9fc2b
			
		
	
	
	dbfee9fc2b
	
	
	
		
			
			Partial commit of the greater MDEV-34348 scope. MDEV-34348: MariaDB is violating clang-16 -Wcast-function-type-strict The functions queue_compare, qsort2_cmp, and qsort_cmp2 all had similar interfaces, and were used interchangable and unsafely cast to one another. This patch consolidates the functions all into the qsort_cmp2 interface. Reviewed By: ============ Marko Mäkelä <marko.makela@mariadb.com>
		
			
				
	
	
		
			1067 lines
		
	
	
	
		
			32 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			1067 lines
		
	
	
	
		
			32 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /* Copyright (c) 2001, 2016, Oracle and/or its affiliates. All rights reserved.
 | |
| 
 | |
|    This program is free software; you can redistribute it and/or modify
 | |
|    it under the terms of the GNU General Public License as published by
 | |
|    the Free Software Foundation; version 2 of the License.
 | |
| 
 | |
|    This program is distributed in the hope that it will be useful,
 | |
|    but WITHOUT ANY WARRANTY; without even the implied warranty of
 | |
|    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 | |
|    GNU General Public License for more details.
 | |
| 
 | |
|    You should have received a copy of the GNU General Public License
 | |
|    along with this program; if not, write to the Free Software
 | |
|    Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1335  USA */
 | |
| 
 | |
| /* Written by Sergei A. Golubchik, who has a shared copyright to this code */
 | |
| 
 | |
| /*  TODO: add caching - pre-read several index entries at once */
 | |
| 
 | |
| /*
 | |
|   Added optimization for full-text queries with plus-words. It was
 | |
|   implemented by sharing maximal document id (max_docid) variable
 | |
|   inside plus subtree. max_docid could be used by any word in plus
 | |
|   subtree, but it could be updated by plus-word only.
 | |
| 
 | |
|   Fulltext "smarter index merge" optimization assumes that rows
 | |
|   it gets are ordered by doc_id. That is not the case when we
 | |
|   search for a word with truncation operator. It may return
 | |
|   rows in random order. Thus we may not use "smarter index merge"
 | |
|   optimization with "trunc-words".
 | |
| 
 | |
|   The idea is: there is no need to search for docid smaller than
 | |
|   biggest docid inside current plus subtree or any upper plus subtree.
 | |
| 
 | |
|   Examples:
 | |
|   +word1 word2
 | |
|     share same max_docid
 | |
|     max_docid updated by word1
 | |
|   +word1 +(word2 word3)
 | |
|     share same max_docid
 | |
|     max_docid updated by word1
 | |
|   +(word1 -word2) +(+word3 word4)
 | |
|     share same max_docid
 | |
|     max_docid updated by word3
 | |
|    +word1 word2 (+word3 word4 (+word5 word6))
 | |
|     three subexpressions (including the top-level one),
 | |
|     every one has its own max_docid, updated by its plus word.
 | |
|     but for the search word6 uses
 | |
|     max(word1.max_docid, word3.max_docid, word5.max_docid),
 | |
|     while word4 uses, accordingly,
 | |
|     max(word1.max_docid, word3.max_docid).
 | |
| */
 | |
| 
 | |
| #define FT_CORE
 | |
| #include "ftdefs.h"
 | |
| 
 | |
| /* search with boolean queries */
 | |
| 
 | |
| static double _wghts[11]=
 | |
| {
 | |
|   0.131687242798354,
 | |
|   0.197530864197531,
 | |
|   0.296296296296296,
 | |
|   0.444444444444444,
 | |
|   0.666666666666667,
 | |
|   1.000000000000000,
 | |
|   1.500000000000000,
 | |
|   2.250000000000000,
 | |
|   3.375000000000000,
 | |
|   5.062500000000000,
 | |
|   7.593750000000000};
 | |
| static double *wghts=_wghts+5; /* wghts[i] = 1.5**i */
 | |
| 
 | |
| static double _nwghts[11]=
 | |
| {
 | |
|  -0.065843621399177,
 | |
|  -0.098765432098766,
 | |
|  -0.148148148148148,
 | |
|  -0.222222222222222,
 | |
|  -0.333333333333334,
 | |
|  -0.500000000000000,
 | |
|  -0.750000000000000,
 | |
|  -1.125000000000000,
 | |
|  -1.687500000000000,
 | |
|  -2.531250000000000,
 | |
|  -3.796875000000000};
 | |
| static double *nwghts=_nwghts+5; /* nwghts[i] = -0.5*1.5**i */
 | |
| 
 | |
| #define FTB_FLAG_TRUNC 1
 | |
| /* At most one of the following flags can be set */
 | |
| #define FTB_FLAG_YES   2
 | |
| #define FTB_FLAG_NO    4
 | |
| #define FTB_FLAG_WONLY 8
 | |
| 
 | |
| typedef struct st_ftb_expr FTB_EXPR;
 | |
| struct st_ftb_expr
 | |
| {
 | |
|   FTB_EXPR *up;
 | |
|   uint      flags;
 | |
| /* ^^^^^^^^^^^^^^^^^^ FTB_{EXPR,WORD} common section */
 | |
|   my_off_t  docid[2];
 | |
|   my_off_t  max_docid;
 | |
|   float     weight;
 | |
|   float     cur_weight;
 | |
|   LIST     *phrase;               /* phrase words */
 | |
|   LIST     *document;             /* for phrase search */
 | |
|   uint      yesses;               /* number of "yes" words matched */
 | |
|   uint      nos;                  /* number of "no"  words matched */
 | |
|   uint      ythresh;              /* number of "yes" words in expr */
 | |
|   uint      yweaks;               /* number of "yes" words for scan only */
 | |
| };
 | |
| 
 | |
| typedef struct st_ftb_word
 | |
| {
 | |
|   FTB_EXPR  *up;
 | |
|   uint       flags;
 | |
| /* ^^^^^^^^^^^^^^^^^^ FTB_{EXPR,WORD} common section */
 | |
|   my_off_t   docid[2];             /* for index search and for scan */
 | |
|   my_off_t   key_root;
 | |
|   FTB_EXPR  *max_docid_expr;
 | |
|   MI_KEYDEF *keyinfo;
 | |
|   struct st_ftb_word *prev;
 | |
|   float      weight;
 | |
|   uint       ndepth;
 | |
|   uint       len;
 | |
|   uchar      off;
 | |
|   uchar      word[1];
 | |
| } FTB_WORD;
 | |
| 
 | |
| typedef struct st_ft_info
 | |
| {
 | |
|   struct _ft_vft *please;
 | |
|   MI_INFO   *info;
 | |
|   CHARSET_INFO *charset;
 | |
|   FTB_EXPR  *root;
 | |
|   FTB_WORD **list;
 | |
|   FTB_WORD  *last_word;
 | |
|   MEM_ROOT   mem_root;
 | |
|   QUEUE      queue;
 | |
|   TREE       no_dupes;
 | |
|   my_off_t   lastpos;
 | |
|   uint       keynr;
 | |
|   uchar      with_scan;
 | |
|   enum { UNINITIALIZED, READY, INDEX_SEARCH, INDEX_DONE } state;
 | |
| } FTB;
 | |
| 
 | |
| static int FTB_WORD_cmp(void *v_, const void *a_, const void *b_)
 | |
| {
 | |
|   int i;
 | |
|   const my_off_t *v= v_;
 | |
|   const FTB_WORD *a= a_;
 | |
|   const FTB_WORD *b= b_;
 | |
| 
 | |
|   /* if a==curdoc, take it as  a < b */
 | |
|   if (v && a->docid[0] == *v)
 | |
|     return -1;
 | |
| 
 | |
|   /* ORDER BY docid, ndepth DESC */
 | |
|   i=CMP_NUM(a->docid[0], b->docid[0]);
 | |
|   if (!i)
 | |
|     i=CMP_NUM(b->ndepth,a->ndepth);
 | |
|   return i;
 | |
| }
 | |
| 
 | |
| static int FTB_WORD_cmp_list(void *cs_, const void *a_, const void *b_)
 | |
| {
 | |
|   CHARSET_INFO *cs= cs_;
 | |
|   const FTB_WORD *const *a= a_;
 | |
|   const FTB_WORD *const *b= b_;
 | |
|   /* ORDER BY word, ndepth */
 | |
|   int i= ha_compare_word(cs, (*a)->word + 1, (*a)->len - 1, (*b)->word + 1,
 | |
|                          (*b)->len - 1);
 | |
|   if (!i)
 | |
|     i= CMP_NUM((*a)->ndepth, (*b)->ndepth);
 | |
|   return i;
 | |
| }
 | |
| 
 | |
| 
 | |
| typedef struct st_my_ftb_param
 | |
| {
 | |
|   FTB *ftb;
 | |
|   FTB_EXPR *ftbe;
 | |
|   uchar *up_quot;
 | |
|   uint depth;
 | |
| } MY_FTB_PARAM;
 | |
| 
 | |
| 
 | |
| static int ftb_query_add_word(MYSQL_FTPARSER_PARAM *param,
 | |
|                               const char *word, int word_len,
 | |
|                               MYSQL_FTPARSER_BOOLEAN_INFO *info)
 | |
| {
 | |
|   MY_FTB_PARAM *ftb_param= param->mysql_ftparam;
 | |
|   FTB_WORD *ftbw;
 | |
|   FTB_EXPR *ftbe, *tmp_expr;
 | |
|   FT_WORD *phrase_word;
 | |
|   LIST *tmp_element;
 | |
|   int r= info->weight_adjust;
 | |
|   float weight= (float)
 | |
|         (info->wasign ? nwghts : wghts)[(r>5)?5:((r<-5)?-5:r)];
 | |
| 
 | |
|   switch (info->type) {
 | |
|     case FT_TOKEN_WORD:
 | |
|       ftbw= (FTB_WORD *)alloc_root(&ftb_param->ftb->mem_root,
 | |
|                                    sizeof(FTB_WORD) + HA_MAX_KEY_BUFF);
 | |
|       ftbw->len= word_len + 1;
 | |
|       ftbw->flags= 0;
 | |
|       ftbw->off= 0;
 | |
|       if (info->yesno > 0) ftbw->flags|= FTB_FLAG_YES;
 | |
|       if (info->yesno < 0) ftbw->flags|= FTB_FLAG_NO;
 | |
|       if (info->trunc) ftbw->flags|= FTB_FLAG_TRUNC;
 | |
|       ftbw->weight= weight;
 | |
|       ftbw->up= ftb_param->ftbe;
 | |
|       ftbw->docid[0]= ftbw->docid[1]= HA_OFFSET_ERROR;
 | |
|       ftbw->ndepth= (info->yesno < 0) + ftb_param->depth;
 | |
|       ftbw->key_root= HA_OFFSET_ERROR;
 | |
|       memcpy(ftbw->word + 1, word, word_len);
 | |
|       ftbw->word[0]= word_len;
 | |
|       if (info->yesno > 0) ftbw->up->ythresh++;
 | |
|       ftb_param->ftb->queue.max_elements++;
 | |
|       ftbw->prev= ftb_param->ftb->last_word;
 | |
|       ftb_param->ftb->last_word= ftbw;
 | |
|       ftb_param->ftb->with_scan|= (info->trunc & FTB_FLAG_TRUNC);
 | |
|       for (tmp_expr= ftb_param->ftbe; tmp_expr->up; tmp_expr= tmp_expr->up)
 | |
|         if (! (tmp_expr->flags & FTB_FLAG_YES))
 | |
|           break;
 | |
|       ftbw->max_docid_expr= tmp_expr;
 | |
|       /* fall through */
 | |
|     case FT_TOKEN_STOPWORD:
 | |
|       if (! ftb_param->up_quot) break;
 | |
|       phrase_word= (FT_WORD *)alloc_root(&ftb_param->ftb->mem_root, sizeof(FT_WORD));
 | |
|       tmp_element= (LIST *)alloc_root(&ftb_param->ftb->mem_root, sizeof(LIST));
 | |
|       phrase_word->pos= (uchar*) word;
 | |
|       phrase_word->len= word_len;
 | |
|       tmp_element->data= (void *)phrase_word;
 | |
|       ftb_param->ftbe->phrase= list_add(ftb_param->ftbe->phrase, tmp_element);
 | |
|       /* Allocate document list at this point.
 | |
|          It allows to avoid huge amount of allocs/frees for each row.*/
 | |
|       tmp_element= (LIST *)alloc_root(&ftb_param->ftb->mem_root, sizeof(LIST));
 | |
|       tmp_element->data= alloc_root(&ftb_param->ftb->mem_root, sizeof(FT_WORD));
 | |
|       ftb_param->ftbe->document=
 | |
|         list_add(ftb_param->ftbe->document, tmp_element);
 | |
|       break;
 | |
|     case FT_TOKEN_LEFT_PAREN:
 | |
|       ftbe=(FTB_EXPR *)alloc_root(&ftb_param->ftb->mem_root, sizeof(FTB_EXPR));
 | |
|       ftbe->flags= 0;
 | |
|       if (info->yesno > 0) ftbe->flags|= FTB_FLAG_YES;
 | |
|       if (info->yesno < 0) ftbe->flags|= FTB_FLAG_NO;
 | |
|       ftbe->weight= weight;
 | |
|       ftbe->up= ftb_param->ftbe;
 | |
|       ftbe->max_docid= ftbe->ythresh= ftbe->yweaks= 0;
 | |
|       ftbe->docid[0]= ftbe->docid[1]= HA_OFFSET_ERROR;
 | |
|       ftbe->phrase= NULL;
 | |
|       ftbe->document= 0;
 | |
|       if (info->quot) ftb_param->ftb->with_scan|= 2;
 | |
|       if (info->yesno > 0) ftbe->up->ythresh++;
 | |
|       ftb_param->ftbe= ftbe;
 | |
|       ftb_param->depth++;
 | |
|       ftb_param->up_quot= (uchar*) info->quot;
 | |
|       break;
 | |
|     case FT_TOKEN_RIGHT_PAREN:
 | |
|       if (ftb_param->ftbe->document)
 | |
|       {
 | |
|         /* Circuit document list */
 | |
|         for (tmp_element= ftb_param->ftbe->document;
 | |
|              tmp_element->next; tmp_element= tmp_element->next) /* no-op */;
 | |
|         tmp_element->next= ftb_param->ftbe->document;
 | |
|         ftb_param->ftbe->document->prev= tmp_element;
 | |
|       }
 | |
|       info->quot= 0;
 | |
|       if (ftb_param->ftbe->up)
 | |
|       {
 | |
|         DBUG_ASSERT(ftb_param->depth);
 | |
|         ftb_param->ftbe= ftb_param->ftbe->up;
 | |
|         ftb_param->depth--;
 | |
|         ftb_param->up_quot= 0;
 | |
|       }
 | |
|       break;
 | |
|     case FT_TOKEN_EOF:
 | |
|     default:
 | |
|       break;
 | |
|   }
 | |
|   return(0);
 | |
| }
 | |
| 
 | |
| 
 | |
| static int ftb_parse_query_internal(MYSQL_FTPARSER_PARAM *param,
 | |
|                                     const char *query, int len)
 | |
| {
 | |
|   MY_FTB_PARAM *ftb_param= param->mysql_ftparam;
 | |
|   MYSQL_FTPARSER_BOOLEAN_INFO info;
 | |
|   CHARSET_INFO *cs= ftb_param->ftb->charset;
 | |
|   const uchar **start= (const uchar**) &query;
 | |
|   uchar *end= (uchar*) query + len;
 | |
|   FT_WORD w;
 | |
| 
 | |
|   w.pos= NULL;
 | |
|   w.len= 0;
 | |
|   info.prev= ' ';
 | |
|   info.quot= 0;
 | |
|   while (ft_get_word(cs, start, end, &w, &info))
 | |
|     param->mysql_add_word(param, (char*) w.pos, (int)w.len, &info);
 | |
|   return(0);
 | |
| }
 | |
| 
 | |
| 
 | |
| static int _ftb_parse_query(FTB *ftb, uchar *query, uint len,
 | |
|                             struct st_mysql_ftparser *parser)
 | |
| {
 | |
|   MYSQL_FTPARSER_PARAM *param;
 | |
|   MY_FTB_PARAM ftb_param;
 | |
|   DBUG_ENTER("_ftb_parse_query");
 | |
|   DBUG_ASSERT(parser);
 | |
| 
 | |
|   if (ftb->state != UNINITIALIZED)
 | |
|     DBUG_RETURN(0);
 | |
|   if (! (param= ftparser_call_initializer(ftb->info, ftb->keynr, 0)))
 | |
|     DBUG_RETURN(1);
 | |
| 
 | |
|   ftb_param.ftb= ftb;
 | |
|   ftb_param.depth= 0;
 | |
|   ftb_param.ftbe= ftb->root;
 | |
|   ftb_param.up_quot= 0;
 | |
| 
 | |
|   param->mysql_parse= ftb_parse_query_internal;
 | |
|   param->mysql_add_word= ftb_query_add_word;
 | |
|   param->mysql_ftparam= (void *)&ftb_param;
 | |
|   param->cs= ftb->charset;
 | |
|   param->doc= (char*) query;
 | |
|   param->length= len;
 | |
|   param->flags= 0;
 | |
|   param->mode= MYSQL_FTPARSER_FULL_BOOLEAN_INFO;
 | |
|   DBUG_RETURN(parser->parse(param));
 | |
| }
 | |
| 
 | |
| 
 | |
| static int _ftb_no_dupes_cmp(void *not_used __attribute__((unused)),
 | |
|                              const void *a, const void *b)
 | |
| {
 | |
|   return CMP_NUM((*((my_off_t*)a)), (*((my_off_t*)b)));
 | |
| }
 | |
| 
 | |
| /*
 | |
|   When performing prefix search (a word with truncation operator), we
 | |
|   must preserve original prefix to ensure that characters which may be
 | |
|   expanded/contracted do not break the prefix. This is done by storing
 | |
|   newly found key immediately after the original word in ftbw->word
 | |
|   buffer.
 | |
| 
 | |
|   ftbw->word= LENGTH WORD [ LENGTH1 WORD1 ] WEIGHT REFERENCE
 | |
|   LENGTH - 1 byte, length of the WORD
 | |
|   WORD - LENGTH bytes, the word itself
 | |
|   LENGTH1 - 1 byte, length of the WORD1, present in case of prefix search
 | |
|   WORD1 - LENGTH bytes, the word itself, present in case of prefix search
 | |
|   WEIGHT - 4 bytes (HA_FT_WLEN), either weight or number of subkeys
 | |
|   REFERENCE - rec_reflength bytes, pointer to the record
 | |
| 
 | |
|   returns 1 if the search was finished (must-word wasn't found)
 | |
| */
 | |
| static int _ft2_search_no_lock(FTB *ftb, FTB_WORD *ftbw, my_bool init_search)
 | |
| {
 | |
|   int r;
 | |
|   int subkeys=1;
 | |
|   my_bool can_go_down;
 | |
|   MI_INFO *info=ftb->info;
 | |
|   uint UNINIT_VAR(off), extra= HA_FT_WLEN + info->s->rec_reflength;
 | |
|   uchar *lastkey_buf=ftbw->word+ftbw->off;
 | |
| 
 | |
|   if (ftbw->flags & FTB_FLAG_TRUNC)
 | |
|     lastkey_buf+=ftbw->len;
 | |
| 
 | |
|   if (init_search)
 | |
|   {
 | |
|     ftbw->key_root=info->s->state.key_root[ftb->keynr];
 | |
|     ftbw->keyinfo=info->s->keyinfo+ftb->keynr;
 | |
| 
 | |
|     r=_mi_search(info, ftbw->keyinfo, (uchar*) ftbw->word, ftbw->len,
 | |
|                  SEARCH_FIND | SEARCH_BIGGER, ftbw->key_root);
 | |
|   }
 | |
|   else
 | |
|   {
 | |
|     uint sflag= SEARCH_BIGGER;
 | |
|     my_off_t max_docid=0;
 | |
|     FTB_EXPR *tmp;
 | |
| 
 | |
|     for (tmp= ftbw->max_docid_expr; tmp; tmp= tmp->up)
 | |
|       set_if_bigger(max_docid, tmp->max_docid);
 | |
| 
 | |
|     if (ftbw->docid[0] < max_docid)
 | |
|     {
 | |
|       sflag|= SEARCH_SAME;
 | |
|       _mi_dpointer(info, (uchar*) (lastkey_buf + HA_FT_WLEN +
 | |
|                                    (ftbw->off ? 0 : lastkey_buf[0] + 1)),
 | |
|                    max_docid);
 | |
|     }
 | |
|     r=_mi_search(info, ftbw->keyinfo, (uchar*) lastkey_buf,
 | |
|                    USE_WHOLE_KEY, sflag, ftbw->key_root);
 | |
|   }
 | |
| 
 | |
|   can_go_down=(!ftbw->off && (init_search || (ftbw->flags & FTB_FLAG_TRUNC)));
 | |
|   /* Skip rows inserted by concurrent insert */
 | |
|   while (!r)
 | |
|   {
 | |
|     if (can_go_down)
 | |
|     {
 | |
|       /* going down ? */
 | |
|       off=info->lastkey_length-extra;
 | |
|       subkeys=ft_sintXkorr(info->lastkey+off);
 | |
|     }
 | |
|     if (subkeys<0 || info->lastpos < info->state->data_file_length)
 | |
|       break;
 | |
|     r= _mi_search_next(info, ftbw->keyinfo, info->lastkey,
 | |
|                        info->lastkey_length,
 | |
| 		       SEARCH_BIGGER, ftbw->key_root);
 | |
|   }
 | |
| 
 | |
|   if (!r && !ftbw->off)
 | |
|   {
 | |
|     r= ha_compare_word_or_prefix(ftb->charset,
 | |
|                                  info->lastkey + 1,
 | |
|                                  info->lastkey_length - extra - 1,
 | |
|                                  (uchar*) ftbw->word + 1,
 | |
|                                  ftbw->len - 1,
 | |
|                                  (my_bool) (ftbw->flags & FTB_FLAG_TRUNC));
 | |
|   }
 | |
| 
 | |
|   if (r) /* not found */
 | |
|   {
 | |
|     if (!ftbw->off || !(ftbw->flags & FTB_FLAG_TRUNC))
 | |
|     {
 | |
|       ftbw->docid[0]=HA_OFFSET_ERROR;
 | |
|       if ((ftbw->flags & FTB_FLAG_YES) && ftbw->up->up==0)
 | |
|       {
 | |
|         /*
 | |
|           This word MUST BE present in every document returned,
 | |
|           so we can stop the search right now
 | |
|         */
 | |
|         ftb->state=INDEX_DONE;
 | |
|         return 1; /* search is done */
 | |
|       }
 | |
|       else
 | |
|         return 0;
 | |
|     }
 | |
| 
 | |
|     /*
 | |
|       Going up to the first-level tree to continue search there.
 | |
|       Only done when performing prefix search.
 | |
| 
 | |
|       Key buffer data pointer as well as docid[0] may be smaller
 | |
|       than values we got while searching first-level tree. Thus
 | |
|       they must be restored to original values to avoid dead-loop,
 | |
|       when subsequent search for a bigger value eventually ends up
 | |
|       in this same second-level tree.
 | |
|     */
 | |
|     _mi_dpointer(info, (uchar*) (lastkey_buf+HA_FT_WLEN), ftbw->key_root);
 | |
|     ftbw->docid[0]= ftbw->key_root;
 | |
|     ftbw->key_root=info->s->state.key_root[ftb->keynr];
 | |
|     ftbw->keyinfo=info->s->keyinfo+ftb->keynr;
 | |
|     ftbw->off=0;
 | |
|     return _ft2_search_no_lock(ftb, ftbw, 0);
 | |
|   }
 | |
| 
 | |
|   /* matching key found */
 | |
|   memcpy(lastkey_buf, info->lastkey, info->lastkey_length);
 | |
|   if (lastkey_buf == ftbw->word)
 | |
|     ftbw->len=info->lastkey_length-extra;
 | |
| 
 | |
|   /* going down ? */
 | |
|   if (subkeys<0)
 | |
|   {
 | |
|     /*
 | |
|       yep, going down, to the second-level tree
 | |
|       TODO here: subkey-based optimization
 | |
|     */
 | |
|     ftbw->off=off;
 | |
|     ftbw->key_root=info->lastpos;
 | |
|     ftbw->keyinfo=& info->s->ft2_keyinfo;
 | |
|     r=_mi_search_first(info, ftbw->keyinfo, ftbw->key_root);
 | |
|     DBUG_ASSERT(r==0);  /* found something */
 | |
|     memcpy(lastkey_buf+off, info->lastkey, info->lastkey_length);
 | |
|   }
 | |
|   ftbw->docid[0]=info->lastpos;
 | |
|   if (ftbw->flags & FTB_FLAG_YES && !(ftbw->flags & FTB_FLAG_TRUNC))
 | |
|     ftbw->max_docid_expr->max_docid= info->lastpos;
 | |
|   return 0;
 | |
| }
 | |
| 
 | |
| static int _ft2_search(FTB *ftb, FTB_WORD *ftbw, my_bool init_search)
 | |
| {
 | |
|   int r;
 | |
|   MYISAM_SHARE *share= ftb->info->s;
 | |
|   if (share->concurrent_insert)
 | |
|     mysql_rwlock_rdlock(&share->key_root_lock[ftb->keynr]);
 | |
|   r= _ft2_search_no_lock(ftb, ftbw, init_search);
 | |
|   if (share->concurrent_insert)
 | |
|     mysql_rwlock_unlock(&share->key_root_lock[ftb->keynr]);
 | |
|   return r;
 | |
| }
 | |
| 
 | |
| static void _ftb_init_index_search(FT_INFO *ftb)
 | |
| {
 | |
|   int i;
 | |
|   FTB_WORD   *ftbw;
 | |
| 
 | |
|   if (ftb->state == UNINITIALIZED || ftb->keynr == NO_SUCH_KEY)
 | |
|     return;
 | |
|   ftb->state=INDEX_SEARCH;
 | |
| 
 | |
|   for (i= queue_last_element(&ftb->queue);
 | |
|        i >= (int) queue_first_element(&ftb->queue);
 | |
|        i--)
 | |
|   {
 | |
|     ftbw=(FTB_WORD *)(queue_element(&ftb->queue, i));
 | |
| 
 | |
|     if (ftbw->flags & FTB_FLAG_TRUNC)
 | |
|     {
 | |
|       /*
 | |
|         special treatment for truncation operator
 | |
|         1. there are some (besides this) +words
 | |
|            | no need to search in the index, it can never ADD new rows
 | |
|            | to the result, and to remove half-matched rows we do scan anyway
 | |
|         2. -trunc*
 | |
|            | same as 1.
 | |
|         3. in 1 and 2, +/- need not be on the same expr. level,
 | |
|            but can be on any upper level, as in +word +(trunc1* trunc2*)
 | |
|         4. otherwise
 | |
|            | We have to index-search for this prefix.
 | |
|            | It may cause duplicates, as in the index (sorted by <word,docid>)
 | |
|            |   <aaaa,row1>
 | |
|            |   <aabb,row2>
 | |
|            |   <aacc,row1>
 | |
|            | Searching for "aa*" will find row1 twice...
 | |
|       */
 | |
|       FTB_EXPR *ftbe;
 | |
|       for (ftbe=(FTB_EXPR*)ftbw;
 | |
|            ftbe->up && !(ftbe->up->flags & FTB_FLAG_TRUNC);
 | |
|            ftbe->up->flags|= FTB_FLAG_TRUNC, ftbe=ftbe->up)
 | |
|       {
 | |
|         if (ftbe->flags & FTB_FLAG_NO ||                     /* 2 */
 | |
|             ftbe->up->ythresh - ftbe->up->yweaks >
 | |
|             (uint) MY_TEST(ftbe->flags & FTB_FLAG_YES))      /* 1 */
 | |
|         {
 | |
|           FTB_EXPR *top_ftbe=ftbe->up;
 | |
|           ftbw->docid[0]=HA_OFFSET_ERROR;
 | |
|           for (ftbe=(FTB_EXPR *)ftbw;
 | |
|                ftbe != top_ftbe && !(ftbe->flags & FTB_FLAG_NO);
 | |
|                ftbe=ftbe->up)
 | |
|               ftbe->up->yweaks++;
 | |
|           ftbe=0;
 | |
|           break;
 | |
|         }
 | |
|       }
 | |
|       if (!ftbe)
 | |
|         continue;
 | |
|       /* 4 */
 | |
|       if (!is_tree_inited(& ftb->no_dupes))
 | |
|         init_tree(& ftb->no_dupes,0,0,sizeof(my_off_t),
 | |
|                   _ftb_no_dupes_cmp,0,0,MYF(0));
 | |
|       else
 | |
|         reset_tree(& ftb->no_dupes);
 | |
|     }
 | |
| 
 | |
|     ftbw->off=0; /* in case of reinit */
 | |
|     if (_ft2_search(ftb, ftbw, 1))
 | |
|       return;
 | |
|   }
 | |
|   queue_fix(& ftb->queue);
 | |
| }
 | |
| 
 | |
| 
 | |
| FT_INFO * ft_init_boolean_search(MI_INFO *info, uint keynr, uchar *query,
 | |
|                                  uint query_len, CHARSET_INFO *cs)
 | |
| {
 | |
|   FTB       *ftb;
 | |
|   FTB_EXPR  *ftbe;
 | |
|   FTB_WORD  *ftbw;
 | |
| 
 | |
|   if (!(ftb=(FTB *)my_malloc(mi_key_memory_FTB, sizeof(FTB), MYF(MY_WME))))
 | |
|     return 0;
 | |
|   ftb->please= (struct _ft_vft *) & _ft_vft_boolean;
 | |
|   ftb->state=UNINITIALIZED;
 | |
|   ftb->info=info;
 | |
|   ftb->keynr=keynr;
 | |
|   ftb->charset=cs;
 | |
|   DBUG_ASSERT(keynr==NO_SUCH_KEY || cs == info->s->keyinfo[keynr].seg->charset);
 | |
|   ftb->with_scan=0;
 | |
|   ftb->lastpos=HA_OFFSET_ERROR;
 | |
|   bzero(& ftb->no_dupes, sizeof(TREE));
 | |
|   ftb->last_word= 0;
 | |
| 
 | |
|   init_alloc_root(mi_key_memory_FTB, &ftb->mem_root, 1024, 1024, MYF(0));
 | |
|   ftb->queue.max_elements= 0;
 | |
|   if (!(ftbe=(FTB_EXPR *)alloc_root(&ftb->mem_root, sizeof(FTB_EXPR))))
 | |
|     goto err;
 | |
|   ftbe->weight=1;
 | |
|   ftbe->flags=FTB_FLAG_YES;
 | |
|   ftbe->nos=1;
 | |
|   ftbe->up=0;
 | |
|   ftbe->max_docid= ftbe->ythresh= ftbe->yweaks= 0;
 | |
|   ftbe->docid[0]=ftbe->docid[1]=HA_OFFSET_ERROR;
 | |
|   ftbe->phrase= NULL;
 | |
|   ftbe->document= 0;
 | |
|   ftb->root=ftbe;
 | |
|   if (unlikely(_ftb_parse_query(ftb, query, query_len,
 | |
|                                 keynr == NO_SUCH_KEY ? &ft_default_parser :
 | |
|                                 info->s->keyinfo[keynr].parser)))
 | |
|     goto err;
 | |
|   /*
 | |
|     Hack: instead of init_queue, we'll use reinit queue to be able
 | |
|     to alloc queue with alloc_root()
 | |
|   */
 | |
|   if (! (ftb->queue.root= (uchar **)alloc_root(&ftb->mem_root,
 | |
|                                               (ftb->queue.max_elements + 1) *
 | |
|                                               sizeof(void *))))
 | |
|     goto err;
 | |
|   reinit_queue(&ftb->queue, ftb->queue.max_elements, 0, 0,
 | |
|                FTB_WORD_cmp, 0, 0, 0);
 | |
|   for (ftbw= ftb->last_word; ftbw; ftbw= ftbw->prev)
 | |
|     queue_insert(&ftb->queue, (uchar *)ftbw);
 | |
|   ftb->list=(FTB_WORD **)alloc_root(&ftb->mem_root,
 | |
|                                      sizeof(FTB_WORD *)*ftb->queue.elements);
 | |
|   memcpy(ftb->list, &queue_top(&ftb->queue), sizeof(FTB_WORD *)*ftb->queue.elements);
 | |
|   my_qsort2(ftb->list, ftb->queue.elements, sizeof(FTB_WORD *),
 | |
|             FTB_WORD_cmp_list, (void*)ftb->charset);
 | |
|   if (ftb->queue.elements<2) ftb->with_scan &= ~FTB_FLAG_TRUNC;
 | |
|   ftb->state=READY;
 | |
|   return ftb;
 | |
| err:
 | |
|   free_root(& ftb->mem_root, MYF(0));
 | |
|   my_free(ftb);
 | |
|   return 0;
 | |
| }
 | |
| 
 | |
| 
 | |
| typedef struct st_my_ftb_phrase_param
 | |
| {
 | |
|   LIST *phrase;
 | |
|   LIST *document;
 | |
|   CHARSET_INFO *cs;
 | |
|   uint phrase_length;
 | |
|   uint document_length;
 | |
|   uint match;
 | |
| } MY_FTB_PHRASE_PARAM;
 | |
| 
 | |
| 
 | |
| static int ftb_phrase_add_word(MYSQL_FTPARSER_PARAM *param,
 | |
|                                const char *word, int word_len,
 | |
|     MYSQL_FTPARSER_BOOLEAN_INFO *boolean_info __attribute__((unused)))
 | |
| {
 | |
|   MY_FTB_PHRASE_PARAM *phrase_param= param->mysql_ftparam;
 | |
|   FT_WORD *w= (FT_WORD *)phrase_param->document->data;
 | |
|   LIST *phrase, *document;
 | |
|   w->pos= (uchar*) word;
 | |
|   w->len= word_len;
 | |
|   phrase_param->document= phrase_param->document->prev;
 | |
|   if (phrase_param->phrase_length > phrase_param->document_length)
 | |
|   {
 | |
|     phrase_param->document_length++;
 | |
|     return 0;
 | |
|   }
 | |
|   /* TODO: rewrite phrase search to avoid
 | |
|      comparing the same word twice. */
 | |
|   for (phrase= phrase_param->phrase, document= phrase_param->document->next;
 | |
|        phrase; phrase= phrase->next, document= document->next)
 | |
|   {
 | |
|     FT_WORD *phrase_word= (FT_WORD *)phrase->data;
 | |
|     FT_WORD *document_word= (FT_WORD *)document->data;
 | |
|     if (my_strnncoll(phrase_param->cs, (uchar*) phrase_word->pos,
 | |
|                      phrase_word->len,
 | |
|                      (uchar*) document_word->pos, document_word->len))
 | |
|       return 0;
 | |
|   }
 | |
|   phrase_param->match++;
 | |
|   return 0;
 | |
| }
 | |
| 
 | |
| 
 | |
| static int ftb_check_phrase_internal(MYSQL_FTPARSER_PARAM *param,
 | |
|                                      const char *document, int len)
 | |
| {
 | |
|   FT_WORD word;
 | |
|   MY_FTB_PHRASE_PARAM *phrase_param= param->mysql_ftparam;
 | |
|   const uchar *docend= (uchar*) document + len;
 | |
|   while (ft_simple_get_word(phrase_param->cs, (uchar**) &document, docend,
 | |
|                             &word, FALSE))
 | |
|   {
 | |
|     param->mysql_add_word(param, (char*) word.pos, (int)word.len, 0);
 | |
|     if (phrase_param->match)
 | |
|       break;
 | |
|   }
 | |
|   return 0;
 | |
| }
 | |
| 
 | |
| 
 | |
| /*
 | |
|   Checks if given buffer matches phrase list.
 | |
| 
 | |
|   SYNOPSIS
 | |
|     _ftb_check_phrase()
 | |
|     s0     start of buffer
 | |
|     e0     end of buffer
 | |
|     phrase broken into list phrase
 | |
|     cs     charset info
 | |
| 
 | |
|   RETURN VALUE
 | |
|     1 is returned if phrase found, 0 else.
 | |
|     -1 is returned if error occurs.
 | |
| */
 | |
| 
 | |
| static int _ftb_check_phrase(FTB *ftb, const uchar *document, uint len,
 | |
|                              FTB_EXPR *ftbe, struct st_mysql_ftparser *parser)
 | |
| {
 | |
|   MY_FTB_PHRASE_PARAM ftb_param;
 | |
|   MYSQL_FTPARSER_PARAM *param;
 | |
|   DBUG_ENTER("_ftb_check_phrase");
 | |
|   DBUG_ASSERT(parser);
 | |
| 
 | |
|   if (! (param= ftparser_call_initializer(ftb->info, ftb->keynr, 1)))
 | |
|     DBUG_RETURN(0);
 | |
| 
 | |
|   ftb_param.phrase= ftbe->phrase;
 | |
|   ftb_param.document= ftbe->document;
 | |
|   ftb_param.cs= ftb->charset;
 | |
|   ftb_param.phrase_length= list_length(ftbe->phrase);
 | |
|   ftb_param.document_length= 1;
 | |
|   ftb_param.match= 0;
 | |
| 
 | |
|   param->mysql_parse= ftb_check_phrase_internal;
 | |
|   param->mysql_add_word= ftb_phrase_add_word;
 | |
|   param->mysql_ftparam= (void *)&ftb_param;
 | |
|   param->cs= ftb->charset;
 | |
|   param->doc= (char *) document;
 | |
|   param->length= len;
 | |
|   param->flags= 0;
 | |
|   param->mode= MYSQL_FTPARSER_WITH_STOPWORDS;
 | |
|   if (unlikely(parser->parse(param)))
 | |
|     return -1;
 | |
|   DBUG_RETURN(ftb_param.match ? 1 : 0);
 | |
| }
 | |
| 
 | |
| 
 | |
| static int _ftb_climb_the_tree(FTB *ftb, FTB_WORD *ftbw, FT_SEG_ITERATOR *ftsi_orig)
 | |
| {
 | |
|   FT_SEG_ITERATOR ftsi;
 | |
|   FTB_EXPR *ftbe;
 | |
|   float weight=ftbw->weight;
 | |
|   int  yn_flag= ftbw->flags, ythresh, mode=(ftsi_orig != 0);
 | |
|   my_off_t curdoc=ftbw->docid[mode];
 | |
|   struct st_mysql_ftparser *parser= ftb->keynr == NO_SUCH_KEY ?
 | |
|                                     &ft_default_parser :
 | |
|                                     ftb->info->s->keyinfo[ftb->keynr].parser;
 | |
| 
 | |
|   for (ftbe=ftbw->up; ftbe; ftbe=ftbe->up)
 | |
|   {
 | |
|     ythresh = ftbe->ythresh - (mode ? 0 : ftbe->yweaks);
 | |
|     if (ftbe->docid[mode] != curdoc)
 | |
|     {
 | |
|       ftbe->cur_weight=0;
 | |
|       ftbe->yesses=ftbe->nos=0;
 | |
|       ftbe->docid[mode]=curdoc;
 | |
|     }
 | |
|     if (ftbe->nos)
 | |
|       break;
 | |
|     if (yn_flag & FTB_FLAG_YES)
 | |
|     {
 | |
|       weight /= ftbe->ythresh;
 | |
|       ftbe->cur_weight += weight;
 | |
|       if ((int) ++ftbe->yesses == ythresh)
 | |
|       {
 | |
|         yn_flag=ftbe->flags;
 | |
|         weight=ftbe->cur_weight*ftbe->weight;
 | |
|         if (mode && ftbe->phrase)
 | |
|         {
 | |
|           int found= 0;
 | |
| 
 | |
|           memcpy(&ftsi, ftsi_orig, sizeof(ftsi));
 | |
|           while (_mi_ft_segiterator(&ftsi) && !found)
 | |
|           {
 | |
|             if (!ftsi.pos)
 | |
|               continue;
 | |
|             found= _ftb_check_phrase(ftb, ftsi.pos, ftsi.len, ftbe, parser);
 | |
|             if (unlikely(found < 0))
 | |
|               return 1;
 | |
|           }
 | |
|           if (!found)
 | |
|             break;
 | |
|         } /* ftbe->quot */
 | |
|       }
 | |
|       else
 | |
|         break;
 | |
|     }
 | |
|     else
 | |
|     if (yn_flag & FTB_FLAG_NO)
 | |
|     {
 | |
|       /*
 | |
|         NOTE: special sort function of queue assures that all
 | |
|         (yn_flag & FTB_FLAG_NO) != 0
 | |
|         events for every particular subexpression will
 | |
|         "auto-magically" happen BEFORE all the
 | |
|         (yn_flag & FTB_FLAG_YES) != 0 events. So no
 | |
|         already matched expression can become not-matched again.
 | |
|       */
 | |
|       ++ftbe->nos;
 | |
|       break;
 | |
|     }
 | |
|     else
 | |
|     {
 | |
|       if (ftbe->ythresh)
 | |
|         weight/=3;
 | |
|       ftbe->cur_weight +=  weight;
 | |
|       if ((int) ftbe->yesses < ythresh)
 | |
|         break;
 | |
|       if (!(yn_flag & FTB_FLAG_WONLY))
 | |
|         yn_flag= ((int) ftbe->yesses++ == ythresh) ? ftbe->flags : FTB_FLAG_WONLY ;
 | |
|       weight*= ftbe->weight;
 | |
|     }
 | |
|   }
 | |
|   return 0;
 | |
| }
 | |
| 
 | |
| 
 | |
| int ft_boolean_read_next(FT_INFO *ftb, char *record)
 | |
| {
 | |
|   FTB_EXPR  *ftbe;
 | |
|   FTB_WORD  *ftbw;
 | |
|   MI_INFO   *info=ftb->info;
 | |
|   my_off_t   curdoc;
 | |
| 
 | |
|   if (ftb->state != INDEX_SEARCH && ftb->state != INDEX_DONE)
 | |
|     return -1;
 | |
| 
 | |
|   /* black magic ON */
 | |
|   if ((int) _mi_check_index(info, ftb->keynr) < 0)
 | |
|     return my_errno;
 | |
|   if (_mi_readinfo(info, F_RDLCK, 1))
 | |
|     return my_errno;
 | |
|   /* black magic OFF */
 | |
| 
 | |
|   if (!ftb->queue.elements)
 | |
|     return my_errno=HA_ERR_END_OF_FILE;
 | |
| 
 | |
|   /* Attention!!! Address of a local variable is used here! See err: label */
 | |
|   ftb->queue.first_cmp_arg=(void *)&curdoc;
 | |
| 
 | |
|   while (ftb->state == INDEX_SEARCH &&
 | |
|          (curdoc=((FTB_WORD *)queue_top(& ftb->queue))->docid[0]) !=
 | |
|          HA_OFFSET_ERROR)
 | |
|   {
 | |
|     while (curdoc == (ftbw=(FTB_WORD *)queue_top(& ftb->queue))->docid[0])
 | |
|     {
 | |
|       if (unlikely(_ftb_climb_the_tree(ftb, ftbw, 0)))
 | |
|       {
 | |
|         my_errno= HA_ERR_OUT_OF_MEM;
 | |
|         goto err;
 | |
|       }
 | |
| 
 | |
|       /* update queue */
 | |
|       _ft2_search(ftb, ftbw, 0);
 | |
|       queue_replace_top(&ftb->queue);
 | |
|     }
 | |
| 
 | |
|     ftbe=ftb->root;
 | |
|     if (ftbe->docid[0]==curdoc && ftbe->cur_weight>0 &&
 | |
|         ftbe->yesses>=(ftbe->ythresh-ftbe->yweaks) && !ftbe->nos)
 | |
|     {
 | |
|       /* curdoc matched ! */
 | |
|       if (is_tree_inited(&ftb->no_dupes) &&
 | |
|           tree_insert(&ftb->no_dupes, &curdoc, 0,
 | |
|                       ftb->no_dupes.custom_arg)->count >1)
 | |
|         /* but it managed already to get past this line once */
 | |
|         continue;
 | |
| 
 | |
|       info->lastpos=curdoc;
 | |
|       /* Clear all states, except that the table was updated */
 | |
|       info->update&= (HA_STATE_CHANGED | HA_STATE_ROW_CHANGED);
 | |
| 
 | |
|       if (!(*info->read_record)(info,curdoc, (uchar*) record))
 | |
|       {
 | |
|         info->update|= HA_STATE_AKTIV;          /* Record is read */
 | |
|         if (ftb->with_scan &&
 | |
|             ft_boolean_find_relevance(ftb,(uchar*) record,0)==0)
 | |
|             continue; /* no match */
 | |
|         my_errno=0;
 | |
|         goto err;
 | |
|       }
 | |
|       goto err;
 | |
|     }
 | |
|   }
 | |
|   ftb->state=INDEX_DONE;
 | |
|   my_errno=HA_ERR_END_OF_FILE;
 | |
| err:
 | |
|   ftb->queue.first_cmp_arg=(void *)0;
 | |
|   return my_errno;
 | |
| }
 | |
| 
 | |
| 
 | |
| typedef struct st_my_ftb_find_param
 | |
| {
 | |
|   FT_INFO *ftb;
 | |
|   FT_SEG_ITERATOR *ftsi;
 | |
| } MY_FTB_FIND_PARAM;
 | |
| 
 | |
| 
 | |
| static int ftb_find_relevance_add_word(MYSQL_FTPARSER_PARAM *param,
 | |
|                                        const char *word, int len,
 | |
|             MYSQL_FTPARSER_BOOLEAN_INFO *boolean_info __attribute__((unused)))
 | |
| {
 | |
|   MY_FTB_FIND_PARAM *ftb_param= param->mysql_ftparam;
 | |
|   FT_INFO *ftb= ftb_param->ftb;
 | |
|   FTB_WORD *ftbw;
 | |
|   int a, b, c;
 | |
|   /*
 | |
|     Find right-most element in the array of query words matching this
 | |
|     word from a document.
 | |
|   */
 | |
|   for (a= 0, b= ftb->queue.elements, c= (a+b)/2; b-a>1; c= (a+b)/2)
 | |
|   {
 | |
|     ftbw= ftb->list[c];
 | |
|     if (ha_compare_word_or_prefix(ftb->charset, (uchar*)word, len,
 | |
|                                   (uchar*)ftbw->word + 1, ftbw->len - 1,
 | |
|                                   (my_bool) (ftbw->flags & FTB_FLAG_TRUNC)) < 0)
 | |
|       b= c;
 | |
|     else
 | |
|       a= c;
 | |
|   }
 | |
|   /*
 | |
|     If there were no words with truncation operator, we iterate to the
 | |
|     beginning of an array until array element is equal to the word from
 | |
|     a document. This is done mainly because the same word may be
 | |
|     mentioned twice (or more) in the query.
 | |
| 
 | |
|     In case query has words with truncation operator we must iterate
 | |
|     to the beginning of the array. There may be non-matching query words
 | |
|     between matching word with truncation operator and the right-most
 | |
|     matching element. E.g., if we're looking for 'aaa15' in an array of
 | |
|     'aaa1* aaa14 aaa15 aaa16'.
 | |
| 
 | |
|     Worse of that there still may be match even if the binary search
 | |
|     above didn't find matching element. E.g., if we're looking for
 | |
|     'aaa15' in an array of 'aaa1* aaa14 aaa16'. The binary search will
 | |
|     stop at 'aaa16'.
 | |
|   */
 | |
|   for (; c >= 0; c--)
 | |
|   {
 | |
|     ftbw= ftb->list[c];
 | |
|     if (ha_compare_word_or_prefix(ftb->charset, (uchar*) word, len,
 | |
|                                   (uchar*) ftbw->word + 1, ftbw->len - 1,
 | |
|                                   (my_bool) (ftbw->flags & FTB_FLAG_TRUNC)))
 | |
|     {
 | |
|       if (ftb->with_scan & FTB_FLAG_TRUNC)
 | |
|         continue;
 | |
|       else
 | |
|         break;
 | |
|     }
 | |
|     if (ftbw->docid[1] == ftb->info->lastpos)
 | |
|       continue;
 | |
|     ftbw->docid[1]= ftb->info->lastpos;
 | |
|     if (unlikely(_ftb_climb_the_tree(ftb, ftbw, ftb_param->ftsi)))
 | |
|       return 1;
 | |
|   }
 | |
|   return(0);
 | |
| }
 | |
| 
 | |
| 
 | |
| static int ftb_find_relevance_parse(MYSQL_FTPARSER_PARAM *param,
 | |
|                                     const char *doc, int len)
 | |
| {
 | |
|   MY_FTB_FIND_PARAM *ftb_param= param->mysql_ftparam;
 | |
|   FT_INFO *ftb= ftb_param->ftb;
 | |
|   uchar *end= (uchar*) doc + len;
 | |
|   FT_WORD w;
 | |
|   while (ft_simple_get_word(ftb->charset, (uchar**) &doc, end, &w, TRUE))
 | |
|     param->mysql_add_word(param, (char*) w.pos, (int)w.len, 0);
 | |
|   return(0);
 | |
| }
 | |
| 
 | |
| 
 | |
| float ft_boolean_find_relevance(FT_INFO *ftb, uchar *record, uint length)
 | |
| {
 | |
|   FTB_EXPR *ftbe;
 | |
|   FT_SEG_ITERATOR ftsi, ftsi2;
 | |
|   my_off_t  docid=ftb->info->lastpos;
 | |
|   MY_FTB_FIND_PARAM ftb_param;
 | |
|   MYSQL_FTPARSER_PARAM *param;
 | |
|   struct st_mysql_ftparser *parser= ftb->keynr == NO_SUCH_KEY ?
 | |
|                                     &ft_default_parser :
 | |
|                                     ftb->info->s->keyinfo[ftb->keynr].parser;
 | |
| 
 | |
|   if (docid == HA_OFFSET_ERROR)
 | |
|     return -2.0;
 | |
|   if (!ftb->queue.elements)
 | |
|     return 0;
 | |
|   if (! (param= ftparser_call_initializer(ftb->info, ftb->keynr, 0)))
 | |
|     return 0;
 | |
| 
 | |
|   if (ftb->state != INDEX_SEARCH && docid <= ftb->lastpos)
 | |
|   {
 | |
|     FTB_EXPR *x;
 | |
|     uint i;
 | |
| 
 | |
|     for (i=0; i < ftb->queue.elements; i++)
 | |
|     {
 | |
|       ftb->list[i]->docid[1]=HA_OFFSET_ERROR;
 | |
|       for (x=ftb->list[i]->up; x; x=x->up)
 | |
|         x->docid[1]=HA_OFFSET_ERROR;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   ftb->lastpos=docid;
 | |
| 
 | |
|   if (ftb->keynr==NO_SUCH_KEY)
 | |
|     _mi_ft_segiterator_dummy_init(record, length, &ftsi);
 | |
|   else
 | |
|     _mi_ft_segiterator_init(ftb->info, ftb->keynr, record, &ftsi);
 | |
|   memcpy(&ftsi2, &ftsi, sizeof(ftsi));
 | |
| 
 | |
|   ftb_param.ftb= ftb;
 | |
|   ftb_param.ftsi= &ftsi2;
 | |
|   param->mysql_parse= ftb_find_relevance_parse;
 | |
|   param->mysql_add_word= ftb_find_relevance_add_word;
 | |
|   param->mysql_ftparam= (void *)&ftb_param;
 | |
|   param->flags= 0;
 | |
|   param->cs= ftb->charset;
 | |
|   param->mode= MYSQL_FTPARSER_SIMPLE_MODE;
 | |
|   while (_mi_ft_segiterator(&ftsi))
 | |
|   {
 | |
|     if (!ftsi.pos)
 | |
|       continue;
 | |
|     param->doc= (char *)ftsi.pos;
 | |
|     param->length= ftsi.len;
 | |
|     if (unlikely(parser->parse(param)))
 | |
|       return 0;
 | |
|   }
 | |
|   ftbe=ftb->root;
 | |
|   if (ftbe->docid[1]==docid && ftbe->cur_weight>0 &&
 | |
|       ftbe->yesses>=ftbe->ythresh && !ftbe->nos)
 | |
|   { /* row matched ! */
 | |
|     return ftbe->cur_weight;
 | |
|   }
 | |
|   else
 | |
|   { /* match failed ! */
 | |
|     return 0.0;
 | |
|   }
 | |
| }
 | |
| 
 | |
| 
 | |
| void ft_boolean_close_search(FT_INFO *ftb)
 | |
| {
 | |
|   if (is_tree_inited(& ftb->no_dupes))
 | |
|   {
 | |
|     delete_tree(&ftb->no_dupes, 0);
 | |
|   }
 | |
|   free_root(& ftb->mem_root, MYF(0));
 | |
|   my_free(ftb);
 | |
| }
 | |
| 
 | |
| 
 | |
| float ft_boolean_get_relevance(FT_INFO *ftb)
 | |
| {
 | |
|   return ftb->root->cur_weight;
 | |
| }
 | |
| 
 | |
| 
 | |
| void ft_boolean_reinit_search(FT_INFO *ftb)
 | |
| {
 | |
|   _ftb_init_index_search(ftb);
 | |
| }
 | |
| 
 |