mirror of
				https://github.com/jart/cosmopolitan.git
				synced 2025-10-26 03:00:57 +00:00 
			
		
		
		
	Further polish SQLite vendoring
- Now integrated with `make tags` for Emacs IDE features - Delete some old deprecated broken full-text search engines - Rename .h → .inc files that don't meet our definition of header - Make sure every #include line is normal form so tools understand See #162
This commit is contained in:
		
							parent
							
								
									690be544da
								
							
						
					
					
						commit
						221817e537
					
				
					 152 changed files with 255 additions and 14523 deletions
				
			
		
							
								
								
									
										2
									
								
								third_party/sqlite3/alter.c
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								third_party/sqlite3/alter.c
									
										
									
									
										vendored
									
									
								
							|  | @ -12,7 +12,7 @@ | |||
| ** This file contains C code routines that used to generate VDBE code | ||||
| ** that implements the ALTER TABLE command. | ||||
| */ | ||||
| #include "third_party/sqlite3/sqliteInt.h" | ||||
| #include "third_party/sqlite3/sqliteInt.inc" | ||||
| /* clang-format off */ | ||||
| 
 | ||||
| /*
 | ||||
|  |  | |||
							
								
								
									
										2
									
								
								third_party/sqlite3/analyze.c
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								third_party/sqlite3/analyze.c
									
										
									
									
										vendored
									
									
								
							|  | @ -140,7 +140,7 @@ | |||
| ** integer in the equivalent columns in sqlite_stat4. | ||||
| */ | ||||
| #ifndef SQLITE_OMIT_ANALYZE | ||||
| #include "third_party/sqlite3/sqliteInt.h" | ||||
| #include "third_party/sqlite3/sqliteInt.inc" | ||||
| /* clang-format off */ | ||||
| 
 | ||||
| #if defined(SQLITE_ENABLE_STAT4) | ||||
|  |  | |||
							
								
								
									
										2
									
								
								third_party/sqlite3/attach.c
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								third_party/sqlite3/attach.c
									
										
									
									
										vendored
									
									
								
							|  | @ -11,7 +11,7 @@ | |||
| ************************************************************************* | ||||
| ** This file contains code used to implement the ATTACH and DETACH commands. | ||||
| */ | ||||
| #include "third_party/sqlite3/sqliteInt.h" | ||||
| #include "third_party/sqlite3/sqliteInt.inc" | ||||
| /* clang-format off */ | ||||
| 
 | ||||
| #ifndef SQLITE_OMIT_ATTACH | ||||
|  |  | |||
							
								
								
									
										2
									
								
								third_party/sqlite3/auth.c
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								third_party/sqlite3/auth.c
									
										
									
									
										vendored
									
									
								
							|  | @ -14,7 +14,7 @@ | |||
| ** systems that do not need this facility may omit it by recompiling | ||||
| ** the library with -DSQLITE_OMIT_AUTHORIZATION=1 | ||||
| */ | ||||
| #include "third_party/sqlite3/sqliteInt.h" | ||||
| #include "third_party/sqlite3/sqliteInt.inc" | ||||
| /* clang-format off */ | ||||
| 
 | ||||
| /*
 | ||||
|  |  | |||
							
								
								
									
										4
									
								
								third_party/sqlite3/backup.c
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								third_party/sqlite3/backup.c
									
										
									
									
										vendored
									
									
								
							|  | @ -12,8 +12,8 @@ | |||
| ** This file contains the implementation of the sqlite3_backup_XXX() | ||||
| ** API functions and the related features. | ||||
| */ | ||||
| #include "third_party/sqlite3/btreeInt.h" | ||||
| #include "third_party/sqlite3/sqliteInt.h" | ||||
| #include "third_party/sqlite3/btreeInt.inc" | ||||
| #include "third_party/sqlite3/sqliteInt.inc" | ||||
| /* clang-format off */ | ||||
| 
 | ||||
| /*
 | ||||
|  |  | |||
							
								
								
									
										2
									
								
								third_party/sqlite3/bitvec.c
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								third_party/sqlite3/bitvec.c
									
										
									
									
										vendored
									
									
								
							|  | @ -34,7 +34,7 @@ | |||
| ** start of a transaction, and is thus usually less than a few thousand, | ||||
| ** but can be as large as 2 billion for a really big database. | ||||
| */ | ||||
| #include "third_party/sqlite3/sqliteInt.h" | ||||
| #include "third_party/sqlite3/sqliteInt.inc" | ||||
| /* clang-format off */ | ||||
| 
 | ||||
| /* Size of the Bitvec structure in bytes. */ | ||||
|  |  | |||
							
								
								
									
										2
									
								
								third_party/sqlite3/btmutex.c
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								third_party/sqlite3/btmutex.c
									
										
									
									
										vendored
									
									
								
							|  | @ -15,7 +15,7 @@ | |||
| ** big and we want to break it down some.  This packaged seemed like | ||||
| ** a good breakout. | ||||
| */ | ||||
| #include "third_party/sqlite3/btreeInt.h" | ||||
| #include "third_party/sqlite3/btreeInt.inc" | ||||
| 
 | ||||
| /* clang-format off */ | ||||
| #ifndef SQLITE_OMIT_SHARED_CACHE | ||||
|  |  | |||
							
								
								
									
										2
									
								
								third_party/sqlite3/btree.c
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								third_party/sqlite3/btree.c
									
										
									
									
										vendored
									
									
								
							|  | @ -13,7 +13,7 @@ | |||
| ** See the header comment on "btreeInt.h" for additional information. | ||||
| ** Including a description of file format and an overview of operation. | ||||
| */ | ||||
| #include "third_party/sqlite3/btreeInt.h" | ||||
| #include "third_party/sqlite3/btreeInt.inc" | ||||
| 
 | ||||
| /* clang-format off */ | ||||
| 
 | ||||
|  |  | |||
|  | @ -213,7 +213,7 @@ | |||
| **      4     Number of leaf pointers on this page | ||||
| **      *     zero or more pages numbers of leaves | ||||
| */ | ||||
| #include "third_party/sqlite3/sqliteInt.h"
 | ||||
| #include "third_party/sqlite3/sqliteInt.inc"
 | ||||
| 
 | ||||
| /* clang-format off */ | ||||
| 
 | ||||
							
								
								
									
										2
									
								
								third_party/sqlite3/build.c
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								third_party/sqlite3/build.c
									
										
									
									
										vendored
									
									
								
							|  | @ -22,7 +22,7 @@ | |||
| **     COMMIT | ||||
| **     ROLLBACK | ||||
| */ | ||||
| #include "third_party/sqlite3/sqliteInt.h" | ||||
| #include "third_party/sqlite3/sqliteInt.inc" | ||||
| 
 | ||||
| /* clang-format off */ | ||||
| 
 | ||||
|  |  | |||
							
								
								
									
										2
									
								
								third_party/sqlite3/callback.c
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								third_party/sqlite3/callback.c
									
										
									
									
										vendored
									
									
								
							|  | @ -13,7 +13,7 @@ | |||
| ** This file contains functions used to access the internal hash tables | ||||
| ** of user defined functions and collation sequences. | ||||
| */ | ||||
| #include "third_party/sqlite3/sqliteInt.h" | ||||
| #include "third_party/sqlite3/sqliteInt.inc" | ||||
| 
 | ||||
| /* clang-format off */ | ||||
| 
 | ||||
|  |  | |||
							
								
								
									
										2
									
								
								third_party/sqlite3/complete.c
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								third_party/sqlite3/complete.c
									
										
									
									
										vendored
									
									
								
							|  | @ -16,7 +16,7 @@ | |||
| ** separating it out, the code will be automatically omitted from | ||||
| ** static links that do not use it. | ||||
| */ | ||||
| #include "third_party/sqlite3/sqliteInt.h" | ||||
| #include "third_party/sqlite3/sqliteInt.inc" | ||||
| 
 | ||||
| /* clang-format off */ | ||||
| 
 | ||||
|  |  | |||
							
								
								
									
										9
									
								
								third_party/sqlite3/ctime.c
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										9
									
								
								third_party/sqlite3/ctime.c
									
										
									
									
										vendored
									
									
								
							|  | @ -17,15 +17,6 @@ | |||
| 
 | ||||
| #ifndef SQLITE_OMIT_COMPILEOPTION_DIAGS /* IMP: R-16824-07538 */ | ||||
| 
 | ||||
| /*
 | ||||
| ** Include the configuration header output by 'configure' if we're using the | ||||
| ** autoconf-based build | ||||
| */ | ||||
| #if defined(_HAVE_SQLITE_CONFIG_H) && !defined(SQLITECONFIG_H) | ||||
| #include "third_party/sqlite3/config.h" | ||||
| #define SQLITECONFIG_H 1 | ||||
| #endif | ||||
| 
 | ||||
| /* These macros are provided to "stringify" the value of the define
 | ||||
| ** for those options in which the value is meaningful. */ | ||||
| #define CTIMEOPT_VAL_(opt) #opt | ||||
|  |  | |||
							
								
								
									
										2
									
								
								third_party/sqlite3/date.c
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								third_party/sqlite3/date.c
									
										
									
									
										vendored
									
									
								
							|  | @ -48,7 +48,7 @@ | |||
| #include "libc/mem/mem.h" | ||||
| #include "libc/time/struct/tm.h" | ||||
| #include "libc/time/time.h" | ||||
| #include "third_party/sqlite3/sqliteInt.h" | ||||
| #include "third_party/sqlite3/sqliteInt.inc" | ||||
| 
 | ||||
| /* clang-format off */ | ||||
| 
 | ||||
|  |  | |||
							
								
								
									
										2
									
								
								third_party/sqlite3/dbpage.c
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								third_party/sqlite3/dbpage.c
									
										
									
									
										vendored
									
									
								
							|  | @ -30,7 +30,7 @@ | |||
| ** value must be a BLOB which is the correct page size, otherwise the | ||||
| ** update fails.  Rows may not be deleted or inserted. | ||||
| */ | ||||
| #include "third_party/sqlite3/sqliteInt.h" /* Requires access to internal data structures */ | ||||
| #include "third_party/sqlite3/sqliteInt.inc" /* Requires access to internal data inc */ | ||||
| 
 | ||||
| /* clang-format off */ | ||||
| 
 | ||||
|  |  | |||
							
								
								
									
										2
									
								
								third_party/sqlite3/dbstat.c
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								third_party/sqlite3/dbstat.c
									
										
									
									
										vendored
									
									
								
							|  | @ -20,7 +20,7 @@ | |||
| ** Additional information is available on the "dbstat.html" page of the | ||||
| ** official SQLite documentation. | ||||
| */ | ||||
| #include "third_party/sqlite3/sqliteInt.h" /* Requires access to internal data structures */ | ||||
| #include "third_party/sqlite3/sqliteInt.inc" /* Requires access to internal data inc */ | ||||
| 
 | ||||
| /* clang-format off */ | ||||
| 
 | ||||
|  |  | |||
							
								
								
									
										2
									
								
								third_party/sqlite3/delete.c
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								third_party/sqlite3/delete.c
									
										
									
									
										vendored
									
									
								
							|  | @ -12,7 +12,7 @@ | |||
| ** This file contains C code routines that are called by the parser | ||||
| ** in order to generate code for DELETE FROM statements. | ||||
| */ | ||||
| #include "third_party/sqlite3/sqliteInt.h" | ||||
| #include "third_party/sqlite3/sqliteInt.inc" | ||||
| 
 | ||||
| /* clang-format off */ | ||||
| 
 | ||||
|  |  | |||
							
								
								
									
										2
									
								
								third_party/sqlite3/expr.c
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								third_party/sqlite3/expr.c
									
										
									
									
										vendored
									
									
								
							|  | @ -12,7 +12,7 @@ | |||
| ** This file contains routines used for analyzing expressions and | ||||
| ** for generating VDBE code that evaluates expressions in SQLite. | ||||
| */ | ||||
| #include "third_party/sqlite3/sqliteInt.h" | ||||
| #include "third_party/sqlite3/sqliteInt.inc" | ||||
| 
 | ||||
| /* clang-format off */ | ||||
| 
 | ||||
|  |  | |||
							
								
								
									
										2
									
								
								third_party/sqlite3/fault.c
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								third_party/sqlite3/fault.c
									
										
									
									
										vendored
									
									
								
							|  | @ -23,7 +23,7 @@ | |||
| ** hash table will continue to function normally.  So a malloc failure | ||||
| ** during a hash table resize is a benign fault. | ||||
| */ | ||||
| #include "third_party/sqlite3/sqliteInt.h" | ||||
| #include "third_party/sqlite3/sqliteInt.inc" | ||||
| 
 | ||||
| /* clang-format off */ | ||||
| 
 | ||||
|  |  | |||
							
								
								
									
										2
									
								
								third_party/sqlite3/fkey.c
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								third_party/sqlite3/fkey.c
									
										
									
									
										vendored
									
									
								
							|  | @ -11,7 +11,7 @@ | |||
| ** This file contains code used by the compiler to add foreign key | ||||
| ** support to compiled SQL statements. | ||||
| */ | ||||
| #include "third_party/sqlite3/sqliteInt.h" | ||||
| #include "third_party/sqlite3/sqliteInt.inc" | ||||
| 
 | ||||
| /* clang-format off */ | ||||
| 
 | ||||
|  |  | |||
							
								
								
									
										3351
									
								
								third_party/sqlite3/fts1.c
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										3351
									
								
								third_party/sqlite3/fts1.c
									
										
									
									
										vendored
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load diff
											
										
									
								
							
							
								
								
									
										11
									
								
								third_party/sqlite3/fts1.h
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										11
									
								
								third_party/sqlite3/fts1.h
									
										
									
									
										vendored
									
									
								
							|  | @ -1,11 +0,0 @@ | |||
| #include "sqlite3.h" | ||||
| 
 | ||||
| #ifdef __cplusplus | ||||
| extern "C" { | ||||
| #endif  /* __cplusplus */ | ||||
| 
 | ||||
| int sqlite3Fts1Init(sqlite3 *db); | ||||
| 
 | ||||
| #ifdef __cplusplus | ||||
| }  /* extern "C" */ | ||||
| #endif  /* __cplusplus */ | ||||
							
								
								
									
										368
									
								
								third_party/sqlite3/fts1_hash.c
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										368
									
								
								third_party/sqlite3/fts1_hash.c
									
										
									
									
										vendored
									
									
								
							|  | @ -1,368 +0,0 @@ | |||
| /*
 | ||||
| ** 2001 September 22 | ||||
| ** | ||||
| ** The author disclaims copyright to this source code.  In place of | ||||
| ** a legal notice, here is a blessing: | ||||
| ** | ||||
| **    May you do good and not evil. | ||||
| **    May you find forgiveness for yourself and forgive others. | ||||
| **    May you share freely, never taking more than you give. | ||||
| ** | ||||
| ************************************************************************* | ||||
| ** This is the implementation of generic hash-tables used in SQLite. | ||||
| ** We've modified it slightly to serve as a standalone hash table | ||||
| ** implementation for the full-text indexing module. | ||||
| */ | ||||
| 
 | ||||
| /* clang-format off */ | ||||
| 
 | ||||
| /*
 | ||||
| ** The code in this file is only compiled if: | ||||
| ** | ||||
| **     * The FTS1 module is being built as an extension | ||||
| **       (in which case SQLITE_CORE is not defined), or | ||||
| ** | ||||
| **     * The FTS1 module is being built into the core of | ||||
| **       SQLite (in which case SQLITE_ENABLE_FTS1 is defined). | ||||
| */ | ||||
| #if !defined(SQLITE_CORE) || defined(SQLITE_ENABLE_FTS1) | ||||
| 
 | ||||
| 
 | ||||
| #include "third_party/sqlite3/fts1_hash.h" | ||||
| 
 | ||||
| static void *malloc_and_zero(int n){ | ||||
|   void *p = malloc(n); | ||||
|   if( p ){ | ||||
|     memset(p, 0, n); | ||||
|   } | ||||
|   return p; | ||||
| } | ||||
| 
 | ||||
| /* Turn bulk memory into a hash table object by initializing the
 | ||||
| ** fields of the Hash structure. | ||||
| ** | ||||
| ** "pNew" is a pointer to the hash table that is to be initialized. | ||||
| ** keyClass is one of the constants  | ||||
| ** FTS1_HASH_BINARY or FTS1_HASH_STRING.  The value of keyClass  | ||||
| ** determines what kind of key the hash table will use.  "copyKey" is | ||||
| ** true if the hash table should make its own private copy of keys and | ||||
| ** false if it should just use the supplied pointer. | ||||
| */ | ||||
| void sqlite3Fts1HashInit(fts1Hash *pNew, int keyClass, int copyKey){ | ||||
|   assert( pNew!=0 ); | ||||
|   assert( keyClass>=FTS1_HASH_STRING && keyClass<=FTS1_HASH_BINARY ); | ||||
|   pNew->keyClass = keyClass; | ||||
|   pNew->copyKey = copyKey; | ||||
|   pNew->first = 0; | ||||
|   pNew->count = 0; | ||||
|   pNew->htsize = 0; | ||||
|   pNew->ht = 0; | ||||
|   pNew->xMalloc = malloc_and_zero; | ||||
|   pNew->xFree = free; | ||||
| } | ||||
| 
 | ||||
| /* Remove all entries from a hash table.  Reclaim all memory.
 | ||||
| ** Call this routine to delete a hash table or to reset a hash table | ||||
| ** to the empty state. | ||||
| */ | ||||
| void sqlite3Fts1HashClear(fts1Hash *pH){ | ||||
|   fts1HashElem *elem;         /* For looping over all elements of the table */ | ||||
| 
 | ||||
|   assert( pH!=0 ); | ||||
|   elem = pH->first; | ||||
|   pH->first = 0; | ||||
|   if( pH->ht ) pH->xFree(pH->ht); | ||||
|   pH->ht = 0; | ||||
|   pH->htsize = 0; | ||||
|   while( elem ){ | ||||
|     fts1HashElem *next_elem = elem->next; | ||||
|     if( pH->copyKey && elem->pKey ){ | ||||
|       pH->xFree(elem->pKey); | ||||
|     } | ||||
|     pH->xFree(elem); | ||||
|     elem = next_elem; | ||||
|   } | ||||
|   pH->count = 0; | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
| ** Hash and comparison functions when the mode is FTS1_HASH_STRING | ||||
| */ | ||||
| static int strHash(const void *pKey, int nKey){ | ||||
|   const char *z = (const char *)pKey; | ||||
|   int h = 0; | ||||
|   if( nKey<=0 ) nKey = (int) strlen(z); | ||||
|   while( nKey > 0  ){ | ||||
|     h = (h<<3) ^ h ^ *z++; | ||||
|     nKey--; | ||||
|   } | ||||
|   return h & 0x7fffffff; | ||||
| } | ||||
| static int strCompare(const void *pKey1, int n1, const void *pKey2, int n2){ | ||||
|   if( n1!=n2 ) return 1; | ||||
|   return strncmp((const char*)pKey1,(const char*)pKey2,n1); | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
| ** Hash and comparison functions when the mode is FTS1_HASH_BINARY | ||||
| */ | ||||
| static int binHash(const void *pKey, int nKey){ | ||||
|   int h = 0; | ||||
|   const char *z = (const char *)pKey; | ||||
|   while( nKey-- > 0 ){ | ||||
|     h = (h<<3) ^ h ^ *(z++); | ||||
|   } | ||||
|   return h & 0x7fffffff; | ||||
| } | ||||
| static int binCompare(const void *pKey1, int n1, const void *pKey2, int n2){ | ||||
|   if( n1!=n2 ) return 1; | ||||
|   return memcmp(pKey1,pKey2,n1); | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
| ** Return a pointer to the appropriate hash function given the key class. | ||||
| ** | ||||
| ** The C syntax in this function definition may be unfamilar to some  | ||||
| ** programmers, so we provide the following additional explanation: | ||||
| ** | ||||
| ** The name of the function is "hashFunction".  The function takes a | ||||
| ** single parameter "keyClass".  The return value of hashFunction() | ||||
| ** is a pointer to another function.  Specifically, the return value | ||||
| ** of hashFunction() is a pointer to a function that takes two parameters | ||||
| ** with types "const void*" and "int" and returns an "int". | ||||
| */ | ||||
| static int (*hashFunction(int keyClass))(const void*,int){ | ||||
|   if( keyClass==FTS1_HASH_STRING ){ | ||||
|     return &strHash; | ||||
|   }else{ | ||||
|     assert( keyClass==FTS1_HASH_BINARY ); | ||||
|     return &binHash; | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
| ** Return a pointer to the appropriate hash function given the key class. | ||||
| ** | ||||
| ** For help in interpreted the obscure C code in the function definition, | ||||
| ** see the header comment on the previous function. | ||||
| */ | ||||
| static int (*compareFunction(int keyClass))(const void*,int,const void*,int){ | ||||
|   if( keyClass==FTS1_HASH_STRING ){ | ||||
|     return &strCompare; | ||||
|   }else{ | ||||
|     assert( keyClass==FTS1_HASH_BINARY ); | ||||
|     return &binCompare; | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| /* Link an element into the hash table
 | ||||
| */ | ||||
| static void insertElement( | ||||
|   fts1Hash *pH,            /* The complete hash table */ | ||||
|   struct _fts1ht *pEntry,  /* The entry into which pNew is inserted */ | ||||
|   fts1HashElem *pNew       /* The element to be inserted */ | ||||
| ){ | ||||
|   fts1HashElem *pHead;     /* First element already in pEntry */ | ||||
|   pHead = pEntry->chain; | ||||
|   if( pHead ){ | ||||
|     pNew->next = pHead; | ||||
|     pNew->prev = pHead->prev; | ||||
|     if( pHead->prev ){ pHead->prev->next = pNew; } | ||||
|     else             { pH->first = pNew; } | ||||
|     pHead->prev = pNew; | ||||
|   }else{ | ||||
|     pNew->next = pH->first; | ||||
|     if( pH->first ){ pH->first->prev = pNew; } | ||||
|     pNew->prev = 0; | ||||
|     pH->first = pNew; | ||||
|   } | ||||
|   pEntry->count++; | ||||
|   pEntry->chain = pNew; | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| /* Resize the hash table so that it cantains "new_size" buckets.
 | ||||
| ** "new_size" must be a power of 2.  The hash table might fail  | ||||
| ** to resize if sqliteMalloc() fails. | ||||
| */ | ||||
| static void rehash(fts1Hash *pH, int new_size){ | ||||
|   struct _fts1ht *new_ht;          /* The new hash table */ | ||||
|   fts1HashElem *elem, *next_elem;  /* For looping over existing elements */ | ||||
|   int (*xHash)(const void*,int);   /* The hash function */ | ||||
| 
 | ||||
|   assert( (new_size & (new_size-1))==0 ); | ||||
|   new_ht = (struct _fts1ht *)pH->xMalloc( new_size*sizeof(struct _fts1ht) ); | ||||
|   if( new_ht==0 ) return; | ||||
|   if( pH->ht ) pH->xFree(pH->ht); | ||||
|   pH->ht = new_ht; | ||||
|   pH->htsize = new_size; | ||||
|   xHash = hashFunction(pH->keyClass); | ||||
|   for(elem=pH->first, pH->first=0; elem; elem = next_elem){ | ||||
|     int h = (*xHash)(elem->pKey, elem->nKey) & (new_size-1); | ||||
|     next_elem = elem->next; | ||||
|     insertElement(pH, &new_ht[h], elem); | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| /* This function (for internal use only) locates an element in an
 | ||||
| ** hash table that matches the given key.  The hash for this key has | ||||
| ** already been computed and is passed as the 4th parameter. | ||||
| */ | ||||
| static fts1HashElem *findElementGivenHash( | ||||
|   const fts1Hash *pH, /* The pH to be searched */ | ||||
|   const void *pKey,   /* The key we are searching for */ | ||||
|   int nKey, | ||||
|   int h               /* The hash for this key. */ | ||||
| ){ | ||||
|   fts1HashElem *elem;            /* Used to loop thru the element list */ | ||||
|   int count;                     /* Number of elements left to test */ | ||||
|   int (*xCompare)(const void*,int,const void*,int);  /* comparison function */ | ||||
| 
 | ||||
|   if( pH->ht ){ | ||||
|     struct _fts1ht *pEntry = &pH->ht[h]; | ||||
|     elem = pEntry->chain; | ||||
|     count = pEntry->count; | ||||
|     xCompare = compareFunction(pH->keyClass); | ||||
|     while( count-- && elem ){ | ||||
|       if( (*xCompare)(elem->pKey,elem->nKey,pKey,nKey)==0 ){  | ||||
|         return elem; | ||||
|       } | ||||
|       elem = elem->next; | ||||
|     } | ||||
|   } | ||||
|   return 0; | ||||
| } | ||||
| 
 | ||||
| /* Remove a single entry from the hash table given a pointer to that
 | ||||
| ** element and a hash on the element's key. | ||||
| */ | ||||
| static void removeElementGivenHash( | ||||
|   fts1Hash *pH,         /* The pH containing "elem" */ | ||||
|   fts1HashElem* elem,   /* The element to be removed from the pH */ | ||||
|   int h                 /* Hash value for the element */ | ||||
| ){ | ||||
|   struct _fts1ht *pEntry; | ||||
|   if( elem->prev ){ | ||||
|     elem->prev->next = elem->next;  | ||||
|   }else{ | ||||
|     pH->first = elem->next; | ||||
|   } | ||||
|   if( elem->next ){ | ||||
|     elem->next->prev = elem->prev; | ||||
|   } | ||||
|   pEntry = &pH->ht[h]; | ||||
|   if( pEntry->chain==elem ){ | ||||
|     pEntry->chain = elem->next; | ||||
|   } | ||||
|   pEntry->count--; | ||||
|   if( pEntry->count<=0 ){ | ||||
|     pEntry->chain = 0; | ||||
|   } | ||||
|   if( pH->copyKey && elem->pKey ){ | ||||
|     pH->xFree(elem->pKey); | ||||
|   } | ||||
|   pH->xFree( elem ); | ||||
|   pH->count--; | ||||
|   if( pH->count<=0 ){ | ||||
|     assert( pH->first==0 ); | ||||
|     assert( pH->count==0 ); | ||||
|     fts1HashClear(pH); | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| /* Attempt to locate an element of the hash table pH with a key
 | ||||
| ** that matches pKey,nKey.  Return the data for this element if it is | ||||
| ** found, or NULL if there is no match. | ||||
| */ | ||||
| void *sqlite3Fts1HashFind(const fts1Hash *pH, const void *pKey, int nKey){ | ||||
|   int h;                 /* A hash on key */ | ||||
|   fts1HashElem *elem;    /* The element that matches key */ | ||||
|   int (*xHash)(const void*,int);  /* The hash function */ | ||||
| 
 | ||||
|   if( pH==0 || pH->ht==0 ) return 0; | ||||
|   xHash = hashFunction(pH->keyClass); | ||||
|   assert( xHash!=0 ); | ||||
|   h = (*xHash)(pKey,nKey); | ||||
|   assert( (pH->htsize & (pH->htsize-1))==0 ); | ||||
|   elem = findElementGivenHash(pH,pKey,nKey, h & (pH->htsize-1)); | ||||
|   return elem ? elem->data : 0; | ||||
| } | ||||
| 
 | ||||
| /* Insert an element into the hash table pH.  The key is pKey,nKey
 | ||||
| ** and the data is "data". | ||||
| ** | ||||
| ** If no element exists with a matching key, then a new | ||||
| ** element is created.  A copy of the key is made if the copyKey | ||||
| ** flag is set.  NULL is returned. | ||||
| ** | ||||
| ** If another element already exists with the same key, then the | ||||
| ** new data replaces the old data and the old data is returned. | ||||
| ** The key is not copied in this instance.  If a malloc fails, then | ||||
| ** the new data is returned and the hash table is unchanged. | ||||
| ** | ||||
| ** If the "data" parameter to this function is NULL, then the | ||||
| ** element corresponding to "key" is removed from the hash table. | ||||
| */ | ||||
| void *sqlite3Fts1HashInsert( | ||||
|   fts1Hash *pH,        /* The hash table to insert into */ | ||||
|   const void *pKey,    /* The key */ | ||||
|   int nKey,            /* Number of bytes in the key */ | ||||
|   void *data           /* The data */ | ||||
| ){ | ||||
|   int hraw;                 /* Raw hash value of the key */ | ||||
|   int h;                    /* the hash of the key modulo hash table size */ | ||||
|   fts1HashElem *elem;       /* Used to loop thru the element list */ | ||||
|   fts1HashElem *new_elem;   /* New element added to the pH */ | ||||
|   int (*xHash)(const void*,int);  /* The hash function */ | ||||
| 
 | ||||
|   assert( pH!=0 ); | ||||
|   xHash = hashFunction(pH->keyClass); | ||||
|   assert( xHash!=0 ); | ||||
|   hraw = (*xHash)(pKey, nKey); | ||||
|   assert( (pH->htsize & (pH->htsize-1))==0 ); | ||||
|   h = hraw & (pH->htsize-1); | ||||
|   elem = findElementGivenHash(pH,pKey,nKey,h); | ||||
|   if( elem ){ | ||||
|     void *old_data = elem->data; | ||||
|     if( data==0 ){ | ||||
|       removeElementGivenHash(pH,elem,h); | ||||
|     }else{ | ||||
|       elem->data = data; | ||||
|     } | ||||
|     return old_data; | ||||
|   } | ||||
|   if( data==0 ) return 0; | ||||
|   new_elem = (fts1HashElem*)pH->xMalloc( sizeof(fts1HashElem) ); | ||||
|   if( new_elem==0 ) return data; | ||||
|   if( pH->copyKey && pKey!=0 ){ | ||||
|     new_elem->pKey = pH->xMalloc( nKey ); | ||||
|     if( new_elem->pKey==0 ){ | ||||
|       pH->xFree(new_elem); | ||||
|       return data; | ||||
|     } | ||||
|     memcpy((void*)new_elem->pKey, pKey, nKey); | ||||
|   }else{ | ||||
|     new_elem->pKey = (void*)pKey; | ||||
|   } | ||||
|   new_elem->nKey = nKey; | ||||
|   pH->count++; | ||||
|   if( pH->htsize==0 ){ | ||||
|     rehash(pH,8); | ||||
|     if( pH->htsize==0 ){ | ||||
|       pH->count = 0; | ||||
|       pH->xFree(new_elem); | ||||
|       return data; | ||||
|     } | ||||
|   } | ||||
|   if( pH->count > pH->htsize ){ | ||||
|     rehash(pH,pH->htsize*2); | ||||
|   } | ||||
|   assert( pH->htsize>0 ); | ||||
|   assert( (pH->htsize & (pH->htsize-1))==0 ); | ||||
|   h = hraw & (pH->htsize-1); | ||||
|   insertElement(pH, &pH->ht[h], new_elem); | ||||
|   new_elem->data = data; | ||||
|   return 0; | ||||
| } | ||||
| 
 | ||||
| #endif /* !defined(SQLITE_CORE) || defined(SQLITE_ENABLE_FTS1) */ | ||||
							
								
								
									
										114
									
								
								third_party/sqlite3/fts1_hash.h
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										114
									
								
								third_party/sqlite3/fts1_hash.h
									
										
									
									
										vendored
									
									
								
							|  | @ -1,114 +0,0 @@ | |||
| /*
 | ||||
| ** 2001 September 22 | ||||
| ** | ||||
| ** The author disclaims copyright to this source code.  In place of | ||||
| ** a legal notice, here is a blessing: | ||||
| ** | ||||
| **    May you do good and not evil. | ||||
| **    May you find forgiveness for yourself and forgive others. | ||||
| **    May you share freely, never taking more than you give. | ||||
| ** | ||||
| ************************************************************************* | ||||
| ** This is the header file for the generic hash-table implementation | ||||
| ** used in SQLite.  We've modified it slightly to serve as a standalone | ||||
| ** hash table implementation for the full-text indexing module. | ||||
| ** | ||||
| */ | ||||
| #ifndef _FTS1_HASH_H_ | ||||
| #define _FTS1_HASH_H_ | ||||
| 
 | ||||
| /* clang-format off */ | ||||
| 
 | ||||
| /* Forward declarations of structures. */ | ||||
| typedef struct fts1Hash fts1Hash; | ||||
| typedef struct fts1HashElem fts1HashElem; | ||||
| 
 | ||||
| /* A complete hash table is an instance of the following structure.
 | ||||
| ** The internals of this structure are intended to be opaque -- client | ||||
| ** code should not attempt to access or modify the fields of this structure | ||||
| ** directly.  Change this structure only by using the routines below. | ||||
| ** However, many of the "procedures" and "functions" for modifying and | ||||
| ** accessing this structure are really macros, so we can't really make | ||||
| ** this structure opaque. | ||||
| */ | ||||
| struct fts1Hash { | ||||
|   char keyClass;          /* HASH_INT, _POINTER, _STRING, _BINARY */ | ||||
|   char copyKey;           /* True if copy of key made on insert */ | ||||
|   int count;              /* Number of entries in this table */ | ||||
|   fts1HashElem *first;    /* The first element of the array */ | ||||
|   void *(*xMalloc)(int);  /* malloc() function to use */ | ||||
|   void (*xFree)(void *);  /* free() function to use */ | ||||
|   int htsize;             /* Number of buckets in the hash table */ | ||||
|   struct _fts1ht {        /* the hash table */ | ||||
|     int count;               /* Number of entries with this hash */ | ||||
|     fts1HashElem *chain;     /* Pointer to first entry with this hash */ | ||||
|   } *ht; | ||||
| }; | ||||
| 
 | ||||
| /* Each element in the hash table is an instance of the following 
 | ||||
| ** structure.  All elements are stored on a single doubly-linked list. | ||||
| ** | ||||
| ** Again, this structure is intended to be opaque, but it can't really | ||||
| ** be opaque because it is used by macros. | ||||
| */ | ||||
| struct fts1HashElem { | ||||
|   fts1HashElem *next, *prev; /* Next and previous elements in the table */ | ||||
|   void *data;                /* Data associated with this element */ | ||||
|   void *pKey; int nKey;      /* Key associated with this element */ | ||||
| }; | ||||
| 
 | ||||
| /*
 | ||||
| ** There are 2 different modes of operation for a hash table: | ||||
| ** | ||||
| **   FTS1_HASH_STRING        pKey points to a string that is nKey bytes long | ||||
| **                           (including the null-terminator, if any).  Case | ||||
| **                           is respected in comparisons. | ||||
| ** | ||||
| **   FTS1_HASH_BINARY        pKey points to binary data nKey bytes long.  | ||||
| **                           memcmp() is used to compare keys. | ||||
| ** | ||||
| ** A copy of the key is made if the copyKey parameter to fts1HashInit is 1.   | ||||
| */ | ||||
| #define FTS1_HASH_STRING    1 | ||||
| #define FTS1_HASH_BINARY    2 | ||||
| 
 | ||||
| /*
 | ||||
| ** Access routines.  To delete, insert a NULL pointer. | ||||
| */ | ||||
| void sqlite3Fts1HashInit(fts1Hash*, int keytype, int copyKey); | ||||
| void *sqlite3Fts1HashInsert(fts1Hash*, const void *pKey, int nKey, void *pData); | ||||
| void *sqlite3Fts1HashFind(const fts1Hash*, const void *pKey, int nKey); | ||||
| void sqlite3Fts1HashClear(fts1Hash*); | ||||
| 
 | ||||
| /*
 | ||||
| ** Shorthand for the functions above | ||||
| */ | ||||
| #define fts1HashInit   sqlite3Fts1HashInit | ||||
| #define fts1HashInsert sqlite3Fts1HashInsert | ||||
| #define fts1HashFind   sqlite3Fts1HashFind | ||||
| #define fts1HashClear  sqlite3Fts1HashClear | ||||
| 
 | ||||
| /*
 | ||||
| ** Macros for looping over all elements of a hash table.  The idiom is | ||||
| ** like this: | ||||
| ** | ||||
| **   fts1Hash h; | ||||
| **   fts1HashElem *p; | ||||
| **   ... | ||||
| **   for(p=fts1HashFirst(&h); p; p=fts1HashNext(p)){ | ||||
| **     SomeStructure *pData = fts1HashData(p); | ||||
| **     // do something with pData
 | ||||
| **   } | ||||
| */ | ||||
| #define fts1HashFirst(H)  ((H)->first) | ||||
| #define fts1HashNext(E)   ((E)->next) | ||||
| #define fts1HashData(E)   ((E)->data) | ||||
| #define fts1HashKey(E)    ((E)->pKey) | ||||
| #define fts1HashKeysize(E) ((E)->nKey) | ||||
| 
 | ||||
| /*
 | ||||
| ** Number of entries in a hash table | ||||
| */ | ||||
| #define fts1HashCount(H)  ((H)->count) | ||||
| 
 | ||||
| #endif /* _FTS1_HASH_H_ */ | ||||
							
								
								
									
										643
									
								
								third_party/sqlite3/fts1_porter.c
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										643
									
								
								third_party/sqlite3/fts1_porter.c
									
										
									
									
										vendored
									
									
								
							|  | @ -1,643 +0,0 @@ | |||
| /*
 | ||||
| ** 2006 September 30 | ||||
| ** | ||||
| ** The author disclaims copyright to this source code.  In place of | ||||
| ** a legal notice, here is a blessing: | ||||
| ** | ||||
| **    May you do good and not evil. | ||||
| **    May you find forgiveness for yourself and forgive others. | ||||
| **    May you share freely, never taking more than you give. | ||||
| ** | ||||
| ************************************************************************* | ||||
| ** Implementation of the full-text-search tokenizer that implements | ||||
| ** a Porter stemmer. | ||||
| */ | ||||
| /*
 | ||||
| ** The code in this file is only compiled if: | ||||
| ** | ||||
| **     * The FTS1 module is being built as an extension | ||||
| **       (in which case SQLITE_CORE is not defined), or | ||||
| ** | ||||
| **     * The FTS1 module is being built into the core of | ||||
| **       SQLite (in which case SQLITE_ENABLE_FTS1 is defined). | ||||
| */ | ||||
| #if !defined(SQLITE_CORE) || defined(SQLITE_ENABLE_FTS1) | ||||
| 
 | ||||
| /* clang-format off */ | ||||
| 
 | ||||
| #include <assert.h> | ||||
| #include <stdlib.h> | ||||
| #include <stdio.h> | ||||
| #include <string.h> | ||||
| #include <ctype.h> | ||||
| 
 | ||||
| #include "fts1_tokenizer.h" | ||||
| 
 | ||||
| /*
 | ||||
| ** Class derived from sqlite3_tokenizer | ||||
| */ | ||||
| typedef struct porter_tokenizer { | ||||
|   sqlite3_tokenizer base;      /* Base class */ | ||||
| } porter_tokenizer; | ||||
| 
 | ||||
| /*
 | ||||
| ** Class derived from sqlit3_tokenizer_cursor | ||||
| */ | ||||
| typedef struct porter_tokenizer_cursor { | ||||
|   sqlite3_tokenizer_cursor base; | ||||
|   const char *zInput;          /* input we are tokenizing */ | ||||
|   int nInput;                  /* size of the input */ | ||||
|   int iOffset;                 /* current position in zInput */ | ||||
|   int iToken;                  /* index of next token to be returned */ | ||||
|   char *zToken;                /* storage for current token */ | ||||
|   int nAllocated;              /* space allocated to zToken buffer */ | ||||
| } porter_tokenizer_cursor; | ||||
| 
 | ||||
| 
 | ||||
| /* Forward declaration */ | ||||
| static const sqlite3_tokenizer_module porterTokenizerModule; | ||||
| 
 | ||||
| 
 | ||||
| /*
 | ||||
| ** Create a new tokenizer instance. | ||||
| */ | ||||
| static int porterCreate( | ||||
|   int argc, const char * const *argv, | ||||
|   sqlite3_tokenizer **ppTokenizer | ||||
| ){ | ||||
|   porter_tokenizer *t; | ||||
|   t = (porter_tokenizer *) calloc(sizeof(*t), 1); | ||||
|   if( t==NULL ) return SQLITE_NOMEM; | ||||
| 
 | ||||
|   *ppTokenizer = &t->base; | ||||
|   return SQLITE_OK; | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
| ** Destroy a tokenizer | ||||
| */ | ||||
| static int porterDestroy(sqlite3_tokenizer *pTokenizer){ | ||||
|   free(pTokenizer); | ||||
|   return SQLITE_OK; | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
| ** Prepare to begin tokenizing a particular string.  The input | ||||
| ** string to be tokenized is zInput[0..nInput-1].  A cursor | ||||
| ** used to incrementally tokenize this string is returned in  | ||||
| ** *ppCursor. | ||||
| */ | ||||
| static int porterOpen( | ||||
|   sqlite3_tokenizer *pTokenizer,         /* The tokenizer */ | ||||
|   const char *zInput, int nInput,        /* String to be tokenized */ | ||||
|   sqlite3_tokenizer_cursor **ppCursor    /* OUT: Tokenization cursor */ | ||||
| ){ | ||||
|   porter_tokenizer_cursor *c; | ||||
| 
 | ||||
|   c = (porter_tokenizer_cursor *) malloc(sizeof(*c)); | ||||
|   if( c==NULL ) return SQLITE_NOMEM; | ||||
| 
 | ||||
|   c->zInput = zInput; | ||||
|   if( zInput==0 ){ | ||||
|     c->nInput = 0; | ||||
|   }else if( nInput<0 ){ | ||||
|     c->nInput = (int)strlen(zInput); | ||||
|   }else{ | ||||
|     c->nInput = nInput; | ||||
|   } | ||||
|   c->iOffset = 0;                 /* start tokenizing at the beginning */ | ||||
|   c->iToken = 0; | ||||
|   c->zToken = NULL;               /* no space allocated, yet. */ | ||||
|   c->nAllocated = 0; | ||||
| 
 | ||||
|   *ppCursor = &c->base; | ||||
|   return SQLITE_OK; | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
| ** Close a tokenization cursor previously opened by a call to | ||||
| ** porterOpen() above. | ||||
| */ | ||||
| static int porterClose(sqlite3_tokenizer_cursor *pCursor){ | ||||
|   porter_tokenizer_cursor *c = (porter_tokenizer_cursor *) pCursor; | ||||
|   free(c->zToken); | ||||
|   free(c); | ||||
|   return SQLITE_OK; | ||||
| } | ||||
| /*
 | ||||
| ** Vowel or consonant | ||||
| */ | ||||
| static const char cType[] = { | ||||
|    0, 1, 1, 1, 0, 1, 1, 1, 0, 1, 1, 1, 1, 1, 0, 1, 1, 1, 1, 1, 0, | ||||
|    1, 1, 1, 2, 1 | ||||
| }; | ||||
| 
 | ||||
| /*
 | ||||
| ** isConsonant() and isVowel() determine if their first character in | ||||
| ** the string they point to is a consonant or a vowel, according | ||||
| ** to Porter ruls.   | ||||
| ** | ||||
| ** A consonate is any letter other than 'a', 'e', 'i', 'o', or 'u'. | ||||
| ** 'Y' is a consonant unless it follows another consonant, | ||||
| ** in which case it is a vowel. | ||||
| ** | ||||
| ** In these routine, the letters are in reverse order.  So the 'y' rule | ||||
| ** is that 'y' is a consonant unless it is followed by another | ||||
| ** consonent. | ||||
| */ | ||||
| static int isVowel(const char*); | ||||
| static int isConsonant(const char *z){ | ||||
|   int j; | ||||
|   char x = *z; | ||||
|   if( x==0 ) return 0; | ||||
|   assert( x>='a' && x<='z' ); | ||||
|   j = cType[x-'a']; | ||||
|   if( j<2 ) return j; | ||||
|   return z[1]==0 || isVowel(z + 1); | ||||
| } | ||||
| static int isVowel(const char *z){ | ||||
|   int j; | ||||
|   char x = *z; | ||||
|   if( x==0 ) return 0; | ||||
|   assert( x>='a' && x<='z' ); | ||||
|   j = cType[x-'a']; | ||||
|   if( j<2 ) return 1-j; | ||||
|   return isConsonant(z + 1); | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
| ** Let any sequence of one or more vowels be represented by V and let | ||||
| ** C be sequence of one or more consonants.  Then every word can be | ||||
| ** represented as: | ||||
| ** | ||||
| **           [C] (VC){m} [V] | ||||
| ** | ||||
| ** In prose:  A word is an optional consonant followed by zero or | ||||
| ** vowel-consonant pairs followed by an optional vowel.  "m" is the | ||||
| ** number of vowel consonant pairs.  This routine computes the value | ||||
| ** of m for the first i bytes of a word. | ||||
| ** | ||||
| ** Return true if the m-value for z is 1 or more.  In other words, | ||||
| ** return true if z contains at least one vowel that is followed | ||||
| ** by a consonant. | ||||
| ** | ||||
| ** In this routine z[] is in reverse order.  So we are really looking | ||||
| ** for an instance of of a consonant followed by a vowel. | ||||
| */ | ||||
| static int m_gt_0(const char *z){ | ||||
|   while( isVowel(z) ){ z++; } | ||||
|   if( *z==0 ) return 0; | ||||
|   while( isConsonant(z) ){ z++; } | ||||
|   return *z!=0; | ||||
| } | ||||
| 
 | ||||
| /* Like mgt0 above except we are looking for a value of m which is
 | ||||
| ** exactly 1 | ||||
| */ | ||||
| static int m_eq_1(const char *z){ | ||||
|   while( isVowel(z) ){ z++; } | ||||
|   if( *z==0 ) return 0; | ||||
|   while( isConsonant(z) ){ z++; } | ||||
|   if( *z==0 ) return 0; | ||||
|   while( isVowel(z) ){ z++; } | ||||
|   if( *z==0 ) return 1; | ||||
|   while( isConsonant(z) ){ z++; } | ||||
|   return *z==0; | ||||
| } | ||||
| 
 | ||||
| /* Like mgt0 above except we are looking for a value of m>1 instead
 | ||||
| ** or m>0 | ||||
| */ | ||||
| static int m_gt_1(const char *z){ | ||||
|   while( isVowel(z) ){ z++; } | ||||
|   if( *z==0 ) return 0; | ||||
|   while( isConsonant(z) ){ z++; } | ||||
|   if( *z==0 ) return 0; | ||||
|   while( isVowel(z) ){ z++; } | ||||
|   if( *z==0 ) return 0; | ||||
|   while( isConsonant(z) ){ z++; } | ||||
|   return *z!=0; | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
| ** Return TRUE if there is a vowel anywhere within z[0..n-1] | ||||
| */ | ||||
| static int hasVowel(const char *z){ | ||||
|   while( isConsonant(z) ){ z++; } | ||||
|   return *z!=0; | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
| ** Return TRUE if the word ends in a double consonant. | ||||
| ** | ||||
| ** The text is reversed here. So we are really looking at | ||||
| ** the first two characters of z[]. | ||||
| */ | ||||
| static int doubleConsonant(const char *z){ | ||||
|   return isConsonant(z) && z[0]==z[1] && isConsonant(z+1); | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
| ** Return TRUE if the word ends with three letters which | ||||
| ** are consonant-vowel-consonent and where the final consonant | ||||
| ** is not 'w', 'x', or 'y'. | ||||
| ** | ||||
| ** The word is reversed here.  So we are really checking the | ||||
| ** first three letters and the first one cannot be in [wxy]. | ||||
| */ | ||||
| static int star_oh(const char *z){ | ||||
|   return | ||||
|     z[0]!=0 && isConsonant(z) && | ||||
|     z[0]!='w' && z[0]!='x' && z[0]!='y' && | ||||
|     z[1]!=0 && isVowel(z+1) && | ||||
|     z[2]!=0 && isConsonant(z+2); | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
| ** If the word ends with zFrom and xCond() is true for the stem | ||||
| ** of the word that preceeds the zFrom ending, then change the  | ||||
| ** ending to zTo. | ||||
| ** | ||||
| ** The input word *pz and zFrom are both in reverse order.  zTo | ||||
| ** is in normal order.  | ||||
| ** | ||||
| ** Return TRUE if zFrom matches.  Return FALSE if zFrom does not | ||||
| ** match.  Not that TRUE is returned even if xCond() fails and | ||||
| ** no substitution occurs. | ||||
| */ | ||||
| static int stem( | ||||
|   char **pz,             /* The word being stemmed (Reversed) */ | ||||
|   const char *zFrom,     /* If the ending matches this... (Reversed) */ | ||||
|   const char *zTo,       /* ... change the ending to this (not reversed) */ | ||||
|   int (*xCond)(const char*)   /* Condition that must be true */ | ||||
| ){ | ||||
|   char *z = *pz; | ||||
|   while( *zFrom && *zFrom==*z ){ z++; zFrom++; } | ||||
|   if( *zFrom!=0 ) return 0; | ||||
|   if( xCond && !xCond(z) ) return 1; | ||||
|   while( *zTo ){ | ||||
|     *(--z) = *(zTo++); | ||||
|   } | ||||
|   *pz = z; | ||||
|   return 1; | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
| ** This is the fallback stemmer used when the porter stemmer is | ||||
| ** inappropriate.  The input word is copied into the output with | ||||
| ** US-ASCII case folding.  If the input word is too long (more | ||||
| ** than 20 bytes if it contains no digits or more than 6 bytes if | ||||
| ** it contains digits) then word is truncated to 20 or 6 bytes | ||||
| ** by taking 10 or 3 bytes from the beginning and end. | ||||
| */ | ||||
| static void copy_stemmer(const char *zIn, int nIn, char *zOut, int *pnOut){ | ||||
|   int i, mx, j; | ||||
|   int hasDigit = 0; | ||||
|   for(i=0; i<nIn; i++){ | ||||
|     int c = zIn[i]; | ||||
|     if( c>='A' && c<='Z' ){ | ||||
|       zOut[i] = c - 'A' + 'a'; | ||||
|     }else{ | ||||
|       if( c>='0' && c<='9' ) hasDigit = 1; | ||||
|       zOut[i] = c; | ||||
|     } | ||||
|   } | ||||
|   mx = hasDigit ? 3 : 10; | ||||
|   if( nIn>mx*2 ){ | ||||
|     for(j=mx, i=nIn-mx; i<nIn; i++, j++){ | ||||
|       zOut[j] = zOut[i]; | ||||
|     } | ||||
|     i = j; | ||||
|   } | ||||
|   zOut[i] = 0; | ||||
|   *pnOut = i; | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| /*
 | ||||
| ** Stem the input word zIn[0..nIn-1].  Store the output in zOut. | ||||
| ** zOut is at least big enough to hold nIn bytes.  Write the actual | ||||
| ** size of the output word (exclusive of the '\0' terminator) into *pnOut. | ||||
| ** | ||||
| ** Any upper-case characters in the US-ASCII character set ([A-Z]) | ||||
| ** are converted to lower case.  Upper-case UTF characters are | ||||
| ** unchanged. | ||||
| ** | ||||
| ** Words that are longer than about 20 bytes are stemmed by retaining | ||||
| ** a few bytes from the beginning and the end of the word.  If the | ||||
| ** word contains digits, 3 bytes are taken from the beginning and | ||||
| ** 3 bytes from the end.  For long words without digits, 10 bytes | ||||
| ** are taken from each end.  US-ASCII case folding still applies. | ||||
| **  | ||||
| ** If the input word contains not digits but does characters not  | ||||
| ** in [a-zA-Z] then no stemming is attempted and this routine just  | ||||
| ** copies the input into the input into the output with US-ASCII | ||||
| ** case folding. | ||||
| ** | ||||
| ** Stemming never increases the length of the word.  So there is | ||||
| ** no chance of overflowing the zOut buffer. | ||||
| */ | ||||
| static void porter_stemmer(const char *zIn, int nIn, char *zOut, int *pnOut){ | ||||
|   int i, j, c; | ||||
|   char zReverse[28]; | ||||
|   char *z, *z2; | ||||
|   if( nIn<3 || nIn>=sizeof(zReverse)-7 ){ | ||||
|     /* The word is too big or too small for the porter stemmer.
 | ||||
|     ** Fallback to the copy stemmer */ | ||||
|     copy_stemmer(zIn, nIn, zOut, pnOut); | ||||
|     return; | ||||
|   } | ||||
|   for(i=0, j=sizeof(zReverse)-6; i<nIn; i++, j--){ | ||||
|     c = zIn[i]; | ||||
|     if( c>='A' && c<='Z' ){ | ||||
|       zReverse[j] = c + 'a' - 'A'; | ||||
|     }else if( c>='a' && c<='z' ){ | ||||
|       zReverse[j] = c; | ||||
|     }else{ | ||||
|       /* The use of a character not in [a-zA-Z] means that we fallback
 | ||||
|       ** to the copy stemmer */ | ||||
|       copy_stemmer(zIn, nIn, zOut, pnOut); | ||||
|       return; | ||||
|     } | ||||
|   } | ||||
|   memset(&zReverse[sizeof(zReverse)-5], 0, 5); | ||||
|   z = &zReverse[j+1]; | ||||
| 
 | ||||
| 
 | ||||
|   /* Step 1a */ | ||||
|   if( z[0]=='s' ){ | ||||
|     if( | ||||
|      !stem(&z, "sess", "ss", 0) && | ||||
|      !stem(&z, "sei", "i", 0)  && | ||||
|      !stem(&z, "ss", "ss", 0) | ||||
|     ){ | ||||
|       z++; | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   /* Step 1b */   | ||||
|   z2 = z; | ||||
|   if( stem(&z, "dee", "ee", m_gt_0) ){ | ||||
|     /* Do nothing.  The work was all in the test */ | ||||
|   }else if(  | ||||
|      (stem(&z, "gni", "", hasVowel) || stem(&z, "de", "", hasVowel)) | ||||
|       && z!=z2 | ||||
|   ){ | ||||
|      if( stem(&z, "ta", "ate", 0) || | ||||
|          stem(&z, "lb", "ble", 0) || | ||||
|          stem(&z, "zi", "ize", 0) ){ | ||||
|        /* Do nothing.  The work was all in the test */ | ||||
|      }else if( doubleConsonant(z) && (*z!='l' && *z!='s' && *z!='z') ){ | ||||
|        z++; | ||||
|      }else if( m_eq_1(z) && star_oh(z) ){ | ||||
|        *(--z) = 'e'; | ||||
|      } | ||||
|   } | ||||
| 
 | ||||
|   /* Step 1c */ | ||||
|   if( z[0]=='y' && hasVowel(z+1) ){ | ||||
|     z[0] = 'i'; | ||||
|   } | ||||
| 
 | ||||
|   /* Step 2 */ | ||||
|   switch( z[1] ){ | ||||
|    case 'a': | ||||
|      stem(&z, "lanoita", "ate", m_gt_0) || | ||||
|      stem(&z, "lanoit", "tion", m_gt_0); | ||||
|      break; | ||||
|    case 'c': | ||||
|      stem(&z, "icne", "ence", m_gt_0) || | ||||
|      stem(&z, "icna", "ance", m_gt_0); | ||||
|      break; | ||||
|    case 'e': | ||||
|      stem(&z, "rezi", "ize", m_gt_0); | ||||
|      break; | ||||
|    case 'g': | ||||
|      stem(&z, "igol", "log", m_gt_0); | ||||
|      break; | ||||
|    case 'l': | ||||
|      stem(&z, "ilb", "ble", m_gt_0) || | ||||
|      stem(&z, "illa", "al", m_gt_0) || | ||||
|      stem(&z, "iltne", "ent", m_gt_0) || | ||||
|      stem(&z, "ile", "e", m_gt_0) || | ||||
|      stem(&z, "ilsuo", "ous", m_gt_0); | ||||
|      break; | ||||
|    case 'o': | ||||
|      stem(&z, "noitazi", "ize", m_gt_0) || | ||||
|      stem(&z, "noita", "ate", m_gt_0) || | ||||
|      stem(&z, "rota", "ate", m_gt_0); | ||||
|      break; | ||||
|    case 's': | ||||
|      stem(&z, "msila", "al", m_gt_0) || | ||||
|      stem(&z, "ssenevi", "ive", m_gt_0) || | ||||
|      stem(&z, "ssenluf", "ful", m_gt_0) || | ||||
|      stem(&z, "ssensuo", "ous", m_gt_0); | ||||
|      break; | ||||
|    case 't': | ||||
|      stem(&z, "itila", "al", m_gt_0) || | ||||
|      stem(&z, "itivi", "ive", m_gt_0) || | ||||
|      stem(&z, "itilib", "ble", m_gt_0); | ||||
|      break; | ||||
|   } | ||||
| 
 | ||||
|   /* Step 3 */ | ||||
|   switch( z[0] ){ | ||||
|    case 'e': | ||||
|      stem(&z, "etaci", "ic", m_gt_0) || | ||||
|      stem(&z, "evita", "", m_gt_0)   || | ||||
|      stem(&z, "ezila", "al", m_gt_0); | ||||
|      break; | ||||
|    case 'i': | ||||
|      stem(&z, "itici", "ic", m_gt_0); | ||||
|      break; | ||||
|    case 'l': | ||||
|      stem(&z, "laci", "ic", m_gt_0) || | ||||
|      stem(&z, "luf", "", m_gt_0); | ||||
|      break; | ||||
|    case 's': | ||||
|      stem(&z, "ssen", "", m_gt_0); | ||||
|      break; | ||||
|   } | ||||
| 
 | ||||
|   /* Step 4 */ | ||||
|   switch( z[1] ){ | ||||
|    case 'a': | ||||
|      if( z[0]=='l' && m_gt_1(z+2) ){ | ||||
|        z += 2; | ||||
|      } | ||||
|      break; | ||||
|    case 'c': | ||||
|      if( z[0]=='e' && z[2]=='n' && (z[3]=='a' || z[3]=='e')  && m_gt_1(z+4)  ){ | ||||
|        z += 4; | ||||
|      } | ||||
|      break; | ||||
|    case 'e': | ||||
|      if( z[0]=='r' && m_gt_1(z+2) ){ | ||||
|        z += 2; | ||||
|      } | ||||
|      break; | ||||
|    case 'i': | ||||
|      if( z[0]=='c' && m_gt_1(z+2) ){ | ||||
|        z += 2; | ||||
|      } | ||||
|      break; | ||||
|    case 'l': | ||||
|      if( z[0]=='e' && z[2]=='b' && (z[3]=='a' || z[3]=='i') && m_gt_1(z+4) ){ | ||||
|        z += 4; | ||||
|      } | ||||
|      break; | ||||
|    case 'n': | ||||
|      if( z[0]=='t' ){ | ||||
|        if( z[2]=='a' ){ | ||||
|          if( m_gt_1(z+3) ){ | ||||
|            z += 3; | ||||
|          } | ||||
|        }else if( z[2]=='e' ){ | ||||
|          stem(&z, "tneme", "", m_gt_1) || | ||||
|          stem(&z, "tnem", "", m_gt_1) || | ||||
|          stem(&z, "tne", "", m_gt_1); | ||||
|        } | ||||
|      } | ||||
|      break; | ||||
|    case 'o': | ||||
|      if( z[0]=='u' ){ | ||||
|        if( m_gt_1(z+2) ){ | ||||
|          z += 2; | ||||
|        } | ||||
|      }else if( z[3]=='s' || z[3]=='t' ){ | ||||
|        stem(&z, "noi", "", m_gt_1); | ||||
|      } | ||||
|      break; | ||||
|    case 's': | ||||
|      if( z[0]=='m' && z[2]=='i' && m_gt_1(z+3) ){ | ||||
|        z += 3; | ||||
|      } | ||||
|      break; | ||||
|    case 't': | ||||
|      stem(&z, "eta", "", m_gt_1) || | ||||
|      stem(&z, "iti", "", m_gt_1); | ||||
|      break; | ||||
|    case 'u': | ||||
|      if( z[0]=='s' && z[2]=='o' && m_gt_1(z+3) ){ | ||||
|        z += 3; | ||||
|      } | ||||
|      break; | ||||
|    case 'v': | ||||
|    case 'z': | ||||
|      if( z[0]=='e' && z[2]=='i' && m_gt_1(z+3) ){ | ||||
|        z += 3; | ||||
|      } | ||||
|      break; | ||||
|   } | ||||
| 
 | ||||
|   /* Step 5a */ | ||||
|   if( z[0]=='e' ){ | ||||
|     if( m_gt_1(z+1) ){ | ||||
|       z++; | ||||
|     }else if( m_eq_1(z+1) && !star_oh(z+1) ){ | ||||
|       z++; | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   /* Step 5b */ | ||||
|   if( m_gt_1(z) && z[0]=='l' && z[1]=='l' ){ | ||||
|     z++; | ||||
|   } | ||||
| 
 | ||||
|   /* z[] is now the stemmed word in reverse order.  Flip it back
 | ||||
|   ** around into forward order and return. | ||||
|   */ | ||||
|   *pnOut = i = strlen(z); | ||||
|   zOut[i] = 0; | ||||
|   while( *z ){ | ||||
|     zOut[--i] = *(z++); | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
| ** Characters that can be part of a token.  We assume any character | ||||
| ** whose value is greater than 0x80 (any UTF character) can be | ||||
| ** part of a token.  In other words, delimiters all must have | ||||
| ** values of 0x7f or lower. | ||||
| */ | ||||
| static const char isIdChar[] = { | ||||
| /* x0 x1 x2 x3 x4 x5 x6 x7 x8 x9 xA xB xC xD xE xF */ | ||||
|     1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, 0, 0, 0, 0, 0,  /* 3x */ | ||||
|     0, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,  /* 4x */ | ||||
|     1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, 0, 0, 0, 1,  /* 5x */ | ||||
|     0, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,  /* 6x */ | ||||
|     1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, 0, 0, 0, 0,  /* 7x */ | ||||
| }; | ||||
| #define idChar(C)  (((ch=C)&0x80)!=0 || (ch>0x2f && isIdChar[ch-0x30])) | ||||
| #define isDelim(C) (((ch=C)&0x80)==0 && (ch<0x30 || !isIdChar[ch-0x30])) | ||||
| 
 | ||||
| /*
 | ||||
| ** Extract the next token from a tokenization cursor.  The cursor must | ||||
| ** have been opened by a prior call to porterOpen(). | ||||
| */ | ||||
| static int porterNext( | ||||
|   sqlite3_tokenizer_cursor *pCursor,  /* Cursor returned by porterOpen */ | ||||
|   const char **pzToken,               /* OUT: *pzToken is the token text */ | ||||
|   int *pnBytes,                       /* OUT: Number of bytes in token */ | ||||
|   int *piStartOffset,                 /* OUT: Starting offset of token */ | ||||
|   int *piEndOffset,                   /* OUT: Ending offset of token */ | ||||
|   int *piPosition                     /* OUT: Position integer of token */ | ||||
| ){ | ||||
|   porter_tokenizer_cursor *c = (porter_tokenizer_cursor *) pCursor; | ||||
|   const char *z = c->zInput; | ||||
| 
 | ||||
|   while( c->iOffset<c->nInput ){ | ||||
|     int iStartOffset, ch; | ||||
| 
 | ||||
|     /* Scan past delimiter characters */ | ||||
|     while( c->iOffset<c->nInput && isDelim(z[c->iOffset]) ){ | ||||
|       c->iOffset++; | ||||
|     } | ||||
| 
 | ||||
|     /* Count non-delimiter characters. */ | ||||
|     iStartOffset = c->iOffset; | ||||
|     while( c->iOffset<c->nInput && !isDelim(z[c->iOffset]) ){ | ||||
|       c->iOffset++; | ||||
|     } | ||||
| 
 | ||||
|     if( c->iOffset>iStartOffset ){ | ||||
|       int n = c->iOffset-iStartOffset; | ||||
|       if( n>c->nAllocated ){ | ||||
|         c->nAllocated = n+20; | ||||
|         c->zToken = realloc(c->zToken, c->nAllocated); | ||||
|         if( c->zToken==NULL ) return SQLITE_NOMEM; | ||||
|       } | ||||
|       porter_stemmer(&z[iStartOffset], n, c->zToken, pnBytes); | ||||
|       *pzToken = c->zToken; | ||||
|       *piStartOffset = iStartOffset; | ||||
|       *piEndOffset = c->iOffset; | ||||
|       *piPosition = c->iToken++; | ||||
|       return SQLITE_OK; | ||||
|     } | ||||
|   } | ||||
|   return SQLITE_DONE; | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
| ** The set of routines that implement the porter-stemmer tokenizer | ||||
| */ | ||||
| static const sqlite3_tokenizer_module porterTokenizerModule = { | ||||
|   0, | ||||
|   porterCreate, | ||||
|   porterDestroy, | ||||
|   porterOpen, | ||||
|   porterClose, | ||||
|   porterNext, | ||||
| }; | ||||
| 
 | ||||
| /*
 | ||||
| ** Allocate a new porter tokenizer.  Return a pointer to the new | ||||
| ** tokenizer in *ppModule | ||||
| */ | ||||
| void sqlite3Fts1PorterTokenizerModule( | ||||
|   sqlite3_tokenizer_module const**ppModule | ||||
| ){ | ||||
|   *ppModule = &porterTokenizerModule; | ||||
| } | ||||
| 
 | ||||
| #endif /* !defined(SQLITE_CORE) || defined(SQLITE_ENABLE_FTS1) */ | ||||
							
								
								
									
										221
									
								
								third_party/sqlite3/fts1_tokenizer1.c
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										221
									
								
								third_party/sqlite3/fts1_tokenizer1.c
									
										
									
									
										vendored
									
									
								
							|  | @ -1,221 +0,0 @@ | |||
| /*
 | ||||
| ** The author disclaims copyright to this source code. | ||||
| ** | ||||
| ************************************************************************* | ||||
| ** Implementation of the "simple" full-text-search tokenizer. | ||||
| */ | ||||
| /*
 | ||||
| ** The code in this file is only compiled if: | ||||
| ** | ||||
| **     * The FTS1 module is being built as an extension | ||||
| **       (in which case SQLITE_CORE is not defined), or | ||||
| ** | ||||
| **     * The FTS1 module is being built into the core of | ||||
| **       SQLite (in which case SQLITE_ENABLE_FTS1 is defined). | ||||
| */ | ||||
| #if !defined(SQLITE_CORE) || defined(SQLITE_ENABLE_FTS1) | ||||
| 
 | ||||
| /* clang-format off */ | ||||
| 
 | ||||
| #include <assert.h> | ||||
| #include <stdlib.h> | ||||
| #include <stdio.h> | ||||
| #include <string.h> | ||||
| #include <ctype.h> | ||||
| 
 | ||||
| #include "fts1_tokenizer.h" | ||||
| 
 | ||||
| typedef struct simple_tokenizer { | ||||
|   sqlite3_tokenizer base; | ||||
|   char delim[128];             /* flag ASCII delimiters */ | ||||
| } simple_tokenizer; | ||||
| 
 | ||||
| typedef struct simple_tokenizer_cursor { | ||||
|   sqlite3_tokenizer_cursor base; | ||||
|   const char *pInput;          /* input we are tokenizing */ | ||||
|   int nBytes;                  /* size of the input */ | ||||
|   int iOffset;                 /* current position in pInput */ | ||||
|   int iToken;                  /* index of next token to be returned */ | ||||
|   char *pToken;                /* storage for current token */ | ||||
|   int nTokenAllocated;         /* space allocated to zToken buffer */ | ||||
| } simple_tokenizer_cursor; | ||||
| 
 | ||||
| 
 | ||||
| /* Forward declaration */ | ||||
| static const sqlite3_tokenizer_module simpleTokenizerModule; | ||||
| 
 | ||||
| static int isDelim(simple_tokenizer *t, unsigned char c){ | ||||
|   return c<0x80 && t->delim[c]; | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
| ** Create a new tokenizer instance. | ||||
| */ | ||||
| static int simpleCreate( | ||||
|   int argc, const char * const *argv, | ||||
|   sqlite3_tokenizer **ppTokenizer | ||||
| ){ | ||||
|   simple_tokenizer *t; | ||||
| 
 | ||||
|   t = (simple_tokenizer *) calloc(sizeof(*t), 1); | ||||
|   if( t==NULL ) return SQLITE_NOMEM; | ||||
| 
 | ||||
|   /* TODO(shess) Delimiters need to remain the same from run to run,
 | ||||
|   ** else we need to reindex.  One solution would be a meta-table to | ||||
|   ** track such information in the database, then we'd only want this | ||||
|   ** information on the initial create. | ||||
|   */ | ||||
|   if( argc>1 ){ | ||||
|     int i, n = strlen(argv[1]); | ||||
|     for(i=0; i<n; i++){ | ||||
|       unsigned char ch = argv[1][i]; | ||||
|       /* We explicitly don't support UTF-8 delimiters for now. */ | ||||
|       if( ch>=0x80 ){ | ||||
|         free(t); | ||||
|         return SQLITE_ERROR; | ||||
|       } | ||||
|       t->delim[ch] = 1; | ||||
|     } | ||||
|   } else { | ||||
|     /* Mark non-alphanumeric ASCII characters as delimiters */ | ||||
|     int i; | ||||
|     for(i=1; i<0x80; i++){ | ||||
|       t->delim[i] = !isalnum(i); | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   *ppTokenizer = &t->base; | ||||
|   return SQLITE_OK; | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
| ** Destroy a tokenizer | ||||
| */ | ||||
| static int simpleDestroy(sqlite3_tokenizer *pTokenizer){ | ||||
|   free(pTokenizer); | ||||
|   return SQLITE_OK; | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
| ** Prepare to begin tokenizing a particular string.  The input | ||||
| ** string to be tokenized is pInput[0..nBytes-1].  A cursor | ||||
| ** used to incrementally tokenize this string is returned in  | ||||
| ** *ppCursor. | ||||
| */ | ||||
| static int simpleOpen( | ||||
|   sqlite3_tokenizer *pTokenizer,         /* The tokenizer */ | ||||
|   const char *pInput, int nBytes,        /* String to be tokenized */ | ||||
|   sqlite3_tokenizer_cursor **ppCursor    /* OUT: Tokenization cursor */ | ||||
| ){ | ||||
|   simple_tokenizer_cursor *c; | ||||
| 
 | ||||
|   c = (simple_tokenizer_cursor *) malloc(sizeof(*c)); | ||||
|   if( c==NULL ) return SQLITE_NOMEM; | ||||
| 
 | ||||
|   c->pInput = pInput; | ||||
|   if( pInput==0 ){ | ||||
|     c->nBytes = 0; | ||||
|   }else if( nBytes<0 ){ | ||||
|     c->nBytes = (int)strlen(pInput); | ||||
|   }else{ | ||||
|     c->nBytes = nBytes; | ||||
|   } | ||||
|   c->iOffset = 0;                 /* start tokenizing at the beginning */ | ||||
|   c->iToken = 0; | ||||
|   c->pToken = NULL;               /* no space allocated, yet. */ | ||||
|   c->nTokenAllocated = 0; | ||||
| 
 | ||||
|   *ppCursor = &c->base; | ||||
|   return SQLITE_OK; | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
| ** Close a tokenization cursor previously opened by a call to | ||||
| ** simpleOpen() above. | ||||
| */ | ||||
| static int simpleClose(sqlite3_tokenizer_cursor *pCursor){ | ||||
|   simple_tokenizer_cursor *c = (simple_tokenizer_cursor *) pCursor; | ||||
|   free(c->pToken); | ||||
|   free(c); | ||||
|   return SQLITE_OK; | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
| ** Extract the next token from a tokenization cursor.  The cursor must | ||||
| ** have been opened by a prior call to simpleOpen(). | ||||
| */ | ||||
| static int simpleNext( | ||||
|   sqlite3_tokenizer_cursor *pCursor,  /* Cursor returned by simpleOpen */ | ||||
|   const char **ppToken,               /* OUT: *ppToken is the token text */ | ||||
|   int *pnBytes,                       /* OUT: Number of bytes in token */ | ||||
|   int *piStartOffset,                 /* OUT: Starting offset of token */ | ||||
|   int *piEndOffset,                   /* OUT: Ending offset of token */ | ||||
|   int *piPosition                     /* OUT: Position integer of token */ | ||||
| ){ | ||||
|   simple_tokenizer_cursor *c = (simple_tokenizer_cursor *) pCursor; | ||||
|   simple_tokenizer *t = (simple_tokenizer *) pCursor->pTokenizer; | ||||
|   unsigned char *p = (unsigned char *)c->pInput; | ||||
| 
 | ||||
|   while( c->iOffset<c->nBytes ){ | ||||
|     int iStartOffset; | ||||
| 
 | ||||
|     /* Scan past delimiter characters */ | ||||
|     while( c->iOffset<c->nBytes && isDelim(t, p[c->iOffset]) ){ | ||||
|       c->iOffset++; | ||||
|     } | ||||
| 
 | ||||
|     /* Count non-delimiter characters. */ | ||||
|     iStartOffset = c->iOffset; | ||||
|     while( c->iOffset<c->nBytes && !isDelim(t, p[c->iOffset]) ){ | ||||
|       c->iOffset++; | ||||
|     } | ||||
| 
 | ||||
|     if( c->iOffset>iStartOffset ){ | ||||
|       int i, n = c->iOffset-iStartOffset; | ||||
|       if( n>c->nTokenAllocated ){ | ||||
|         c->nTokenAllocated = n+20; | ||||
|         c->pToken = realloc(c->pToken, c->nTokenAllocated); | ||||
|         if( c->pToken==NULL ) return SQLITE_NOMEM; | ||||
|       } | ||||
|       for(i=0; i<n; i++){ | ||||
|         /* TODO(shess) This needs expansion to handle UTF-8
 | ||||
|         ** case-insensitivity. | ||||
|         */ | ||||
|         unsigned char ch = p[iStartOffset+i]; | ||||
|         c->pToken[i] = ch<0x80 ? tolower(ch) : ch; | ||||
|       } | ||||
|       *ppToken = c->pToken; | ||||
|       *pnBytes = n; | ||||
|       *piStartOffset = iStartOffset; | ||||
|       *piEndOffset = c->iOffset; | ||||
|       *piPosition = c->iToken++; | ||||
| 
 | ||||
|       return SQLITE_OK; | ||||
|     } | ||||
|   } | ||||
|   return SQLITE_DONE; | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
| ** The set of routines that implement the simple tokenizer | ||||
| */ | ||||
| static const sqlite3_tokenizer_module simpleTokenizerModule = { | ||||
|   0, | ||||
|   simpleCreate, | ||||
|   simpleDestroy, | ||||
|   simpleOpen, | ||||
|   simpleClose, | ||||
|   simpleNext, | ||||
| }; | ||||
| 
 | ||||
| /*
 | ||||
| ** Allocate a new simple tokenizer.  Return a pointer to the new | ||||
| ** tokenizer in *ppModule | ||||
| */ | ||||
| void sqlite3Fts1SimpleTokenizerModule( | ||||
|   sqlite3_tokenizer_module const**ppModule | ||||
| ){ | ||||
|   *ppModule = &simpleTokenizerModule; | ||||
| } | ||||
| 
 | ||||
| #endif /* !defined(SQLITE_CORE) || defined(SQLITE_ENABLE_FTS1) */ | ||||
							
								
								
									
										6861
									
								
								third_party/sqlite3/fts2.c
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										6861
									
								
								third_party/sqlite3/fts2.c
									
										
									
									
										vendored
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load diff
											
										
									
								
							
							
								
								
									
										26
									
								
								third_party/sqlite3/fts2.h
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										26
									
								
								third_party/sqlite3/fts2.h
									
										
									
									
										vendored
									
									
								
							|  | @ -1,26 +0,0 @@ | |||
| /*
 | ||||
| ** 2006 Oct 10 | ||||
| ** | ||||
| ** The author disclaims copyright to this source code.  In place of | ||||
| ** a legal notice, here is a blessing: | ||||
| ** | ||||
| **    May you do good and not evil. | ||||
| **    May you find forgiveness for yourself and forgive others. | ||||
| **    May you share freely, never taking more than you give. | ||||
| ** | ||||
| ****************************************************************************** | ||||
| ** | ||||
| ** This header file is used by programs that want to link against the | ||||
| ** FTS2 library.  All it does is declare the sqlite3Fts2Init() interface. | ||||
| */ | ||||
| #include "sqlite3.h" | ||||
| 
 | ||||
| #ifdef __cplusplus | ||||
| extern "C" { | ||||
| #endif  /* __cplusplus */ | ||||
| 
 | ||||
| int sqlite3Fts2Init(sqlite3 *db); | ||||
| 
 | ||||
| #ifdef __cplusplus | ||||
| }  /* extern "C" */ | ||||
| #endif  /* __cplusplus */ | ||||
							
								
								
									
										376
									
								
								third_party/sqlite3/fts2_hash.c
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										376
									
								
								third_party/sqlite3/fts2_hash.c
									
										
									
									
										vendored
									
									
								
							|  | @ -1,376 +0,0 @@ | |||
| /*
 | ||||
| ** 2001 September 22 | ||||
| ** | ||||
| ** The author disclaims copyright to this source code.  In place of | ||||
| ** a legal notice, here is a blessing: | ||||
| ** | ||||
| **    May you do good and not evil. | ||||
| **    May you find forgiveness for yourself and forgive others. | ||||
| **    May you share freely, never taking more than you give. | ||||
| ** | ||||
| ************************************************************************* | ||||
| ** This is the implementation of generic hash-tables used in SQLite. | ||||
| ** We've modified it slightly to serve as a standalone hash table | ||||
| ** implementation for the full-text indexing module. | ||||
| */ | ||||
| /*
 | ||||
| ** The code in this file is only compiled if: | ||||
| ** | ||||
| **     * The FTS2 module is being built as an extension | ||||
| **       (in which case SQLITE_CORE is not defined), or | ||||
| ** | ||||
| **     * The FTS2 module is being built into the core of | ||||
| **       SQLite (in which case SQLITE_ENABLE_FTS2 is defined). | ||||
| */ | ||||
| /* clang-format off */ | ||||
| #if !defined(SQLITE_CORE) || defined(SQLITE_ENABLE_FTS2) | ||||
| 
 | ||||
| #include <assert.h> | ||||
| #include <stdlib.h> | ||||
| #include <string.h> | ||||
| 
 | ||||
| #include "sqlite3.h" | ||||
| #include "sqlite3ext.h" | ||||
| SQLITE_EXTENSION_INIT3 | ||||
| #include "fts2_hash.h" | ||||
| 
 | ||||
| /*
 | ||||
| ** Malloc and Free functions | ||||
| */ | ||||
| static void *fts2HashMalloc(int n){ | ||||
|   void *p = sqlite3_malloc(n); | ||||
|   if( p ){ | ||||
|     memset(p, 0, n); | ||||
|   } | ||||
|   return p; | ||||
| } | ||||
| static void fts2HashFree(void *p){ | ||||
|   sqlite3_free(p); | ||||
| } | ||||
| 
 | ||||
| /* Turn bulk memory into a hash table object by initializing the
 | ||||
| ** fields of the Hash structure. | ||||
| ** | ||||
| ** "pNew" is a pointer to the hash table that is to be initialized. | ||||
| ** keyClass is one of the constants  | ||||
| ** FTS2_HASH_BINARY or FTS2_HASH_STRING.  The value of keyClass  | ||||
| ** determines what kind of key the hash table will use.  "copyKey" is | ||||
| ** true if the hash table should make its own private copy of keys and | ||||
| ** false if it should just use the supplied pointer. | ||||
| */ | ||||
| void sqlite3Fts2HashInit(fts2Hash *pNew, int keyClass, int copyKey){ | ||||
|   assert( pNew!=0 ); | ||||
|   assert( keyClass>=FTS2_HASH_STRING && keyClass<=FTS2_HASH_BINARY ); | ||||
|   pNew->keyClass = keyClass; | ||||
|   pNew->copyKey = copyKey; | ||||
|   pNew->first = 0; | ||||
|   pNew->count = 0; | ||||
|   pNew->htsize = 0; | ||||
|   pNew->ht = 0; | ||||
| } | ||||
| 
 | ||||
| /* Remove all entries from a hash table.  Reclaim all memory.
 | ||||
| ** Call this routine to delete a hash table or to reset a hash table | ||||
| ** to the empty state. | ||||
| */ | ||||
| void sqlite3Fts2HashClear(fts2Hash *pH){ | ||||
|   fts2HashElem *elem;         /* For looping over all elements of the table */ | ||||
| 
 | ||||
|   assert( pH!=0 ); | ||||
|   elem = pH->first; | ||||
|   pH->first = 0; | ||||
|   fts2HashFree(pH->ht); | ||||
|   pH->ht = 0; | ||||
|   pH->htsize = 0; | ||||
|   while( elem ){ | ||||
|     fts2HashElem *next_elem = elem->next; | ||||
|     if( pH->copyKey && elem->pKey ){ | ||||
|       fts2HashFree(elem->pKey); | ||||
|     } | ||||
|     fts2HashFree(elem); | ||||
|     elem = next_elem; | ||||
|   } | ||||
|   pH->count = 0; | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
| ** Hash and comparison functions when the mode is FTS2_HASH_STRING | ||||
| */ | ||||
| static int strHash(const void *pKey, int nKey){ | ||||
|   const char *z = (const char *)pKey; | ||||
|   int h = 0; | ||||
|   if( nKey<=0 ) nKey = (int) strlen(z); | ||||
|   while( nKey > 0  ){ | ||||
|     h = (h<<3) ^ h ^ *z++; | ||||
|     nKey--; | ||||
|   } | ||||
|   return h & 0x7fffffff; | ||||
| } | ||||
| static int strCompare(const void *pKey1, int n1, const void *pKey2, int n2){ | ||||
|   if( n1!=n2 ) return 1; | ||||
|   return strncmp((const char*)pKey1,(const char*)pKey2,n1); | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
| ** Hash and comparison functions when the mode is FTS2_HASH_BINARY | ||||
| */ | ||||
| static int binHash(const void *pKey, int nKey){ | ||||
|   int h = 0; | ||||
|   const char *z = (const char *)pKey; | ||||
|   while( nKey-- > 0 ){ | ||||
|     h = (h<<3) ^ h ^ *(z++); | ||||
|   } | ||||
|   return h & 0x7fffffff; | ||||
| } | ||||
| static int binCompare(const void *pKey1, int n1, const void *pKey2, int n2){ | ||||
|   if( n1!=n2 ) return 1; | ||||
|   return memcmp(pKey1,pKey2,n1); | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
| ** Return a pointer to the appropriate hash function given the key class. | ||||
| ** | ||||
| ** The C syntax in this function definition may be unfamilar to some  | ||||
| ** programmers, so we provide the following additional explanation: | ||||
| ** | ||||
| ** The name of the function is "hashFunction".  The function takes a | ||||
| ** single parameter "keyClass".  The return value of hashFunction() | ||||
| ** is a pointer to another function.  Specifically, the return value | ||||
| ** of hashFunction() is a pointer to a function that takes two parameters | ||||
| ** with types "const void*" and "int" and returns an "int". | ||||
| */ | ||||
| static int (*hashFunction(int keyClass))(const void*,int){ | ||||
|   if( keyClass==FTS2_HASH_STRING ){ | ||||
|     return &strHash; | ||||
|   }else{ | ||||
|     assert( keyClass==FTS2_HASH_BINARY ); | ||||
|     return &binHash; | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
| ** Return a pointer to the appropriate hash function given the key class. | ||||
| ** | ||||
| ** For help in interpreted the obscure C code in the function definition, | ||||
| ** see the header comment on the previous function. | ||||
| */ | ||||
| static int (*compareFunction(int keyClass))(const void*,int,const void*,int){ | ||||
|   if( keyClass==FTS2_HASH_STRING ){ | ||||
|     return &strCompare; | ||||
|   }else{ | ||||
|     assert( keyClass==FTS2_HASH_BINARY ); | ||||
|     return &binCompare; | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| /* Link an element into the hash table
 | ||||
| */ | ||||
| static void insertElement( | ||||
|   fts2Hash *pH,            /* The complete hash table */ | ||||
|   struct _fts2ht *pEntry,  /* The entry into which pNew is inserted */ | ||||
|   fts2HashElem *pNew       /* The element to be inserted */ | ||||
| ){ | ||||
|   fts2HashElem *pHead;     /* First element already in pEntry */ | ||||
|   pHead = pEntry->chain; | ||||
|   if( pHead ){ | ||||
|     pNew->next = pHead; | ||||
|     pNew->prev = pHead->prev; | ||||
|     if( pHead->prev ){ pHead->prev->next = pNew; } | ||||
|     else             { pH->first = pNew; } | ||||
|     pHead->prev = pNew; | ||||
|   }else{ | ||||
|     pNew->next = pH->first; | ||||
|     if( pH->first ){ pH->first->prev = pNew; } | ||||
|     pNew->prev = 0; | ||||
|     pH->first = pNew; | ||||
|   } | ||||
|   pEntry->count++; | ||||
|   pEntry->chain = pNew; | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| /* Resize the hash table so that it cantains "new_size" buckets.
 | ||||
| ** "new_size" must be a power of 2.  The hash table might fail  | ||||
| ** to resize if sqliteMalloc() fails. | ||||
| */ | ||||
| static void rehash(fts2Hash *pH, int new_size){ | ||||
|   struct _fts2ht *new_ht;          /* The new hash table */ | ||||
|   fts2HashElem *elem, *next_elem;  /* For looping over existing elements */ | ||||
|   int (*xHash)(const void*,int);   /* The hash function */ | ||||
| 
 | ||||
|   assert( (new_size & (new_size-1))==0 ); | ||||
|   new_ht = (struct _fts2ht *)fts2HashMalloc( new_size*sizeof(struct _fts2ht) ); | ||||
|   if( new_ht==0 ) return; | ||||
|   fts2HashFree(pH->ht); | ||||
|   pH->ht = new_ht; | ||||
|   pH->htsize = new_size; | ||||
|   xHash = hashFunction(pH->keyClass); | ||||
|   for(elem=pH->first, pH->first=0; elem; elem = next_elem){ | ||||
|     int h = (*xHash)(elem->pKey, elem->nKey) & (new_size-1); | ||||
|     next_elem = elem->next; | ||||
|     insertElement(pH, &new_ht[h], elem); | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| /* This function (for internal use only) locates an element in an
 | ||||
| ** hash table that matches the given key.  The hash for this key has | ||||
| ** already been computed and is passed as the 4th parameter. | ||||
| */ | ||||
| static fts2HashElem *findElementGivenHash( | ||||
|   const fts2Hash *pH, /* The pH to be searched */ | ||||
|   const void *pKey,   /* The key we are searching for */ | ||||
|   int nKey, | ||||
|   int h               /* The hash for this key. */ | ||||
| ){ | ||||
|   fts2HashElem *elem;            /* Used to loop thru the element list */ | ||||
|   int count;                     /* Number of elements left to test */ | ||||
|   int (*xCompare)(const void*,int,const void*,int);  /* comparison function */ | ||||
| 
 | ||||
|   if( pH->ht ){ | ||||
|     struct _fts2ht *pEntry = &pH->ht[h]; | ||||
|     elem = pEntry->chain; | ||||
|     count = pEntry->count; | ||||
|     xCompare = compareFunction(pH->keyClass); | ||||
|     while( count-- && elem ){ | ||||
|       if( (*xCompare)(elem->pKey,elem->nKey,pKey,nKey)==0 ){  | ||||
|         return elem; | ||||
|       } | ||||
|       elem = elem->next; | ||||
|     } | ||||
|   } | ||||
|   return 0; | ||||
| } | ||||
| 
 | ||||
| /* Remove a single entry from the hash table given a pointer to that
 | ||||
| ** element and a hash on the element's key. | ||||
| */ | ||||
| static void removeElementGivenHash( | ||||
|   fts2Hash *pH,         /* The pH containing "elem" */ | ||||
|   fts2HashElem* elem,   /* The element to be removed from the pH */ | ||||
|   int h                 /* Hash value for the element */ | ||||
| ){ | ||||
|   struct _fts2ht *pEntry; | ||||
|   if( elem->prev ){ | ||||
|     elem->prev->next = elem->next;  | ||||
|   }else{ | ||||
|     pH->first = elem->next; | ||||
|   } | ||||
|   if( elem->next ){ | ||||
|     elem->next->prev = elem->prev; | ||||
|   } | ||||
|   pEntry = &pH->ht[h]; | ||||
|   if( pEntry->chain==elem ){ | ||||
|     pEntry->chain = elem->next; | ||||
|   } | ||||
|   pEntry->count--; | ||||
|   if( pEntry->count<=0 ){ | ||||
|     pEntry->chain = 0; | ||||
|   } | ||||
|   if( pH->copyKey && elem->pKey ){ | ||||
|     fts2HashFree(elem->pKey); | ||||
|   } | ||||
|   fts2HashFree( elem ); | ||||
|   pH->count--; | ||||
|   if( pH->count<=0 ){ | ||||
|     assert( pH->first==0 ); | ||||
|     assert( pH->count==0 ); | ||||
|     fts2HashClear(pH); | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| /* Attempt to locate an element of the hash table pH with a key
 | ||||
| ** that matches pKey,nKey.  Return the data for this element if it is | ||||
| ** found, or NULL if there is no match. | ||||
| */ | ||||
| void *sqlite3Fts2HashFind(const fts2Hash *pH, const void *pKey, int nKey){ | ||||
|   int h;                 /* A hash on key */ | ||||
|   fts2HashElem *elem;    /* The element that matches key */ | ||||
|   int (*xHash)(const void*,int);  /* The hash function */ | ||||
| 
 | ||||
|   if( pH==0 || pH->ht==0 ) return 0; | ||||
|   xHash = hashFunction(pH->keyClass); | ||||
|   assert( xHash!=0 ); | ||||
|   h = (*xHash)(pKey,nKey); | ||||
|   assert( (pH->htsize & (pH->htsize-1))==0 ); | ||||
|   elem = findElementGivenHash(pH,pKey,nKey, h & (pH->htsize-1)); | ||||
|   return elem ? elem->data : 0; | ||||
| } | ||||
| 
 | ||||
| /* Insert an element into the hash table pH.  The key is pKey,nKey
 | ||||
| ** and the data is "data". | ||||
| ** | ||||
| ** If no element exists with a matching key, then a new | ||||
| ** element is created.  A copy of the key is made if the copyKey | ||||
| ** flag is set.  NULL is returned. | ||||
| ** | ||||
| ** If another element already exists with the same key, then the | ||||
| ** new data replaces the old data and the old data is returned. | ||||
| ** The key is not copied in this instance.  If a malloc fails, then | ||||
| ** the new data is returned and the hash table is unchanged. | ||||
| ** | ||||
| ** If the "data" parameter to this function is NULL, then the | ||||
| ** element corresponding to "key" is removed from the hash table. | ||||
| */ | ||||
| void *sqlite3Fts2HashInsert( | ||||
|   fts2Hash *pH,        /* The hash table to insert into */ | ||||
|   const void *pKey,    /* The key */ | ||||
|   int nKey,            /* Number of bytes in the key */ | ||||
|   void *data           /* The data */ | ||||
| ){ | ||||
|   int hraw;                 /* Raw hash value of the key */ | ||||
|   int h;                    /* the hash of the key modulo hash table size */ | ||||
|   fts2HashElem *elem;       /* Used to loop thru the element list */ | ||||
|   fts2HashElem *new_elem;   /* New element added to the pH */ | ||||
|   int (*xHash)(const void*,int);  /* The hash function */ | ||||
| 
 | ||||
|   assert( pH!=0 ); | ||||
|   xHash = hashFunction(pH->keyClass); | ||||
|   assert( xHash!=0 ); | ||||
|   hraw = (*xHash)(pKey, nKey); | ||||
|   assert( (pH->htsize & (pH->htsize-1))==0 ); | ||||
|   h = hraw & (pH->htsize-1); | ||||
|   elem = findElementGivenHash(pH,pKey,nKey,h); | ||||
|   if( elem ){ | ||||
|     void *old_data = elem->data; | ||||
|     if( data==0 ){ | ||||
|       removeElementGivenHash(pH,elem,h); | ||||
|     }else{ | ||||
|       elem->data = data; | ||||
|     } | ||||
|     return old_data; | ||||
|   } | ||||
|   if( data==0 ) return 0; | ||||
|   new_elem = (fts2HashElem*)fts2HashMalloc( sizeof(fts2HashElem) ); | ||||
|   if( new_elem==0 ) return data; | ||||
|   if( pH->copyKey && pKey!=0 ){ | ||||
|     new_elem->pKey = fts2HashMalloc( nKey ); | ||||
|     if( new_elem->pKey==0 ){ | ||||
|       fts2HashFree(new_elem); | ||||
|       return data; | ||||
|     } | ||||
|     memcpy((void*)new_elem->pKey, pKey, nKey); | ||||
|   }else{ | ||||
|     new_elem->pKey = (void*)pKey; | ||||
|   } | ||||
|   new_elem->nKey = nKey; | ||||
|   pH->count++; | ||||
|   if( pH->htsize==0 ){ | ||||
|     rehash(pH,8); | ||||
|     if( pH->htsize==0 ){ | ||||
|       pH->count = 0; | ||||
|       fts2HashFree(new_elem); | ||||
|       return data; | ||||
|     } | ||||
|   } | ||||
|   if( pH->count > pH->htsize ){ | ||||
|     rehash(pH,pH->htsize*2); | ||||
|   } | ||||
|   assert( pH->htsize>0 ); | ||||
|   assert( (pH->htsize & (pH->htsize-1))==0 ); | ||||
|   h = hraw & (pH->htsize-1); | ||||
|   insertElement(pH, &pH->ht[h], new_elem); | ||||
|   new_elem->data = data; | ||||
|   return 0; | ||||
| } | ||||
| 
 | ||||
| #endif /* !defined(SQLITE_CORE) || defined(SQLITE_ENABLE_FTS2) */ | ||||
							
								
								
									
										112
									
								
								third_party/sqlite3/fts2_hash.h
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										112
									
								
								third_party/sqlite3/fts2_hash.h
									
										
									
									
										vendored
									
									
								
							|  | @ -1,112 +0,0 @@ | |||
| /*
 | ||||
| ** 2001 September 22 | ||||
| ** | ||||
| ** The author disclaims copyright to this source code.  In place of | ||||
| ** a legal notice, here is a blessing: | ||||
| ** | ||||
| **    May you do good and not evil. | ||||
| **    May you find forgiveness for yourself and forgive others. | ||||
| **    May you share freely, never taking more than you give. | ||||
| ** | ||||
| ************************************************************************* | ||||
| ** This is the header file for the generic hash-table implementation | ||||
| ** used in SQLite.  We've modified it slightly to serve as a standalone | ||||
| ** hash table implementation for the full-text indexing module. | ||||
| ** | ||||
| */ | ||||
| #ifndef _FTS2_HASH_H_ | ||||
| #define _FTS2_HASH_H_ | ||||
| 
 | ||||
| /* clang-format off */ | ||||
| 
 | ||||
| /* Forward declarations of structures. */ | ||||
| typedef struct fts2Hash fts2Hash; | ||||
| typedef struct fts2HashElem fts2HashElem; | ||||
| 
 | ||||
| /* A complete hash table is an instance of the following structure.
 | ||||
| ** The internals of this structure are intended to be opaque -- client | ||||
| ** code should not attempt to access or modify the fields of this structure | ||||
| ** directly.  Change this structure only by using the routines below. | ||||
| ** However, many of the "procedures" and "functions" for modifying and | ||||
| ** accessing this structure are really macros, so we can't really make | ||||
| ** this structure opaque. | ||||
| */ | ||||
| struct fts2Hash { | ||||
|   char keyClass;          /* HASH_INT, _POINTER, _STRING, _BINARY */ | ||||
|   char copyKey;           /* True if copy of key made on insert */ | ||||
|   int count;              /* Number of entries in this table */ | ||||
|   fts2HashElem *first;    /* The first element of the array */ | ||||
|   int htsize;             /* Number of buckets in the hash table */ | ||||
|   struct _fts2ht {        /* the hash table */ | ||||
|     int count;               /* Number of entries with this hash */ | ||||
|     fts2HashElem *chain;     /* Pointer to first entry with this hash */ | ||||
|   } *ht; | ||||
| }; | ||||
| 
 | ||||
| /* Each element in the hash table is an instance of the following 
 | ||||
| ** structure.  All elements are stored on a single doubly-linked list. | ||||
| ** | ||||
| ** Again, this structure is intended to be opaque, but it can't really | ||||
| ** be opaque because it is used by macros. | ||||
| */ | ||||
| struct fts2HashElem { | ||||
|   fts2HashElem *next, *prev; /* Next and previous elements in the table */ | ||||
|   void *data;                /* Data associated with this element */ | ||||
|   void *pKey; int nKey;      /* Key associated with this element */ | ||||
| }; | ||||
| 
 | ||||
| /*
 | ||||
| ** There are 2 different modes of operation for a hash table: | ||||
| ** | ||||
| **   FTS2_HASH_STRING        pKey points to a string that is nKey bytes long | ||||
| **                           (including the null-terminator, if any).  Case | ||||
| **                           is respected in comparisons. | ||||
| ** | ||||
| **   FTS2_HASH_BINARY        pKey points to binary data nKey bytes long.  | ||||
| **                           memcmp() is used to compare keys. | ||||
| ** | ||||
| ** A copy of the key is made if the copyKey parameter to fts2HashInit is 1.   | ||||
| */ | ||||
| #define FTS2_HASH_STRING    1 | ||||
| #define FTS2_HASH_BINARY    2 | ||||
| 
 | ||||
| /*
 | ||||
| ** Access routines.  To delete, insert a NULL pointer. | ||||
| */ | ||||
| void sqlite3Fts2HashInit(fts2Hash*, int keytype, int copyKey); | ||||
| void *sqlite3Fts2HashInsert(fts2Hash*, const void *pKey, int nKey, void *pData); | ||||
| void *sqlite3Fts2HashFind(const fts2Hash*, const void *pKey, int nKey); | ||||
| void sqlite3Fts2HashClear(fts2Hash*); | ||||
| 
 | ||||
| /*
 | ||||
| ** Shorthand for the functions above | ||||
| */ | ||||
| #define fts2HashInit   sqlite3Fts2HashInit | ||||
| #define fts2HashInsert sqlite3Fts2HashInsert | ||||
| #define fts2HashFind   sqlite3Fts2HashFind | ||||
| #define fts2HashClear  sqlite3Fts2HashClear | ||||
| 
 | ||||
| /*
 | ||||
| ** Macros for looping over all elements of a hash table.  The idiom is | ||||
| ** like this: | ||||
| ** | ||||
| **   fts2Hash h; | ||||
| **   fts2HashElem *p; | ||||
| **   ... | ||||
| **   for(p=fts2HashFirst(&h); p; p=fts2HashNext(p)){ | ||||
| **     SomeStructure *pData = fts2HashData(p); | ||||
| **     // do something with pData
 | ||||
| **   } | ||||
| */ | ||||
| #define fts2HashFirst(H)  ((H)->first) | ||||
| #define fts2HashNext(E)   ((E)->next) | ||||
| #define fts2HashData(E)   ((E)->data) | ||||
| #define fts2HashKey(E)    ((E)->pKey) | ||||
| #define fts2HashKeysize(E) ((E)->nKey) | ||||
| 
 | ||||
| /*
 | ||||
| ** Number of entries in a hash table | ||||
| */ | ||||
| #define fts2HashCount(H)  ((H)->count) | ||||
| 
 | ||||
| #endif /* _FTS2_HASH_H_ */ | ||||
							
								
								
									
										261
									
								
								third_party/sqlite3/fts2_icu.c
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										261
									
								
								third_party/sqlite3/fts2_icu.c
									
										
									
									
										vendored
									
									
								
							|  | @ -1,261 +0,0 @@ | |||
| /*
 | ||||
| ** 2007 June 22 | ||||
| ** | ||||
| ** The author disclaims copyright to this source code.  In place of | ||||
| ** a legal notice, here is a blessing: | ||||
| ** | ||||
| **    May you do good and not evil. | ||||
| **    May you find forgiveness for yourself and forgive others. | ||||
| **    May you share freely, never taking more than you give. | ||||
| ** | ||||
| ************************************************************************* | ||||
| ** This file implements a tokenizer for fts2 based on the ICU library. | ||||
| ** | ||||
| ** $Id: fts2_icu.c,v 1.3 2008/12/18 05:30:26 danielk1977 Exp $ | ||||
| */ | ||||
| #if !defined(SQLITE_CORE) || defined(SQLITE_ENABLE_FTS2) | ||||
| #ifdef SQLITE_ENABLE_ICU | ||||
| 
 | ||||
| /* clang-format off */ | ||||
| 
 | ||||
| #include <assert.h> | ||||
| #include <string.h> | ||||
| #include "fts2_tokenizer.h" | ||||
| 
 | ||||
| #include <unicode/ubrk.h> | ||||
| #include <unicode/ucol.h> | ||||
| #include <unicode/ustring.h> | ||||
| #include <unicode/utf16.h> | ||||
| 
 | ||||
| typedef struct IcuTokenizer IcuTokenizer; | ||||
| typedef struct IcuCursor IcuCursor; | ||||
| 
 | ||||
| struct IcuTokenizer { | ||||
|   sqlite3_tokenizer base; | ||||
|   char *zLocale; | ||||
| }; | ||||
| 
 | ||||
| struct IcuCursor { | ||||
|   sqlite3_tokenizer_cursor base; | ||||
| 
 | ||||
|   UBreakIterator *pIter;      /* ICU break-iterator object */ | ||||
|   int nChar;                  /* Number of UChar elements in pInput */ | ||||
|   UChar *aChar;               /* Copy of input using utf-16 encoding */ | ||||
|   int *aOffset;               /* Offsets of each character in utf-8 input */ | ||||
| 
 | ||||
|   int nBuffer; | ||||
|   char *zBuffer; | ||||
| 
 | ||||
|   int iToken; | ||||
| }; | ||||
| 
 | ||||
| /*
 | ||||
| ** Create a new tokenizer instance. | ||||
| */ | ||||
| static int icuCreate( | ||||
|   int argc,                            /* Number of entries in argv[] */ | ||||
|   const char * const *argv,            /* Tokenizer creation arguments */ | ||||
|   sqlite3_tokenizer **ppTokenizer      /* OUT: Created tokenizer */ | ||||
| ){ | ||||
|   IcuTokenizer *p; | ||||
|   int n = 0; | ||||
| 
 | ||||
|   if( argc>0 ){ | ||||
|     n = strlen(argv[0])+1; | ||||
|   } | ||||
|   p = (IcuTokenizer *)sqlite3_malloc(sizeof(IcuTokenizer)+n); | ||||
|   if( !p ){ | ||||
|     return SQLITE_NOMEM; | ||||
|   } | ||||
|   memset(p, 0, sizeof(IcuTokenizer)); | ||||
| 
 | ||||
|   if( n ){ | ||||
|     p->zLocale = (char *)&p[1]; | ||||
|     memcpy(p->zLocale, argv[0], n); | ||||
|   } | ||||
| 
 | ||||
|   *ppTokenizer = (sqlite3_tokenizer *)p; | ||||
| 
 | ||||
|   return SQLITE_OK; | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
| ** Destroy a tokenizer | ||||
| */ | ||||
| static int icuDestroy(sqlite3_tokenizer *pTokenizer){ | ||||
|   IcuTokenizer *p = (IcuTokenizer *)pTokenizer; | ||||
|   sqlite3_free(p); | ||||
|   return SQLITE_OK; | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
| ** Prepare to begin tokenizing a particular string.  The input | ||||
| ** string to be tokenized is pInput[0..nBytes-1].  A cursor | ||||
| ** used to incrementally tokenize this string is returned in  | ||||
| ** *ppCursor. | ||||
| */ | ||||
| static int icuOpen( | ||||
|   sqlite3_tokenizer *pTokenizer,         /* The tokenizer */ | ||||
|   const char *zInput,                    /* Input string */ | ||||
|   int nInput,                            /* Length of zInput in bytes */ | ||||
|   sqlite3_tokenizer_cursor **ppCursor    /* OUT: Tokenization cursor */ | ||||
| ){ | ||||
|   IcuTokenizer *p = (IcuTokenizer *)pTokenizer; | ||||
|   IcuCursor *pCsr; | ||||
| 
 | ||||
|   const int32_t opt = U_FOLD_CASE_DEFAULT; | ||||
|   UErrorCode status = U_ZERO_ERROR; | ||||
|   int nChar; | ||||
| 
 | ||||
|   UChar32 c; | ||||
|   int iInput = 0; | ||||
|   int iOut = 0; | ||||
| 
 | ||||
|   *ppCursor = 0; | ||||
| 
 | ||||
|   if( nInput<0 ){ | ||||
|     nInput = strlen(zInput); | ||||
|   } | ||||
|   nChar = nInput+1; | ||||
|   pCsr = (IcuCursor *)sqlite3_malloc( | ||||
|       sizeof(IcuCursor) +                /* IcuCursor */ | ||||
|       ((nChar+3)&~3) * sizeof(UChar) +   /* IcuCursor.aChar[] */ | ||||
|       (nChar+1) * sizeof(int)            /* IcuCursor.aOffset[] */ | ||||
|   ); | ||||
|   if( !pCsr ){ | ||||
|     return SQLITE_NOMEM; | ||||
|   } | ||||
|   memset(pCsr, 0, sizeof(IcuCursor)); | ||||
|   pCsr->aChar = (UChar *)&pCsr[1]; | ||||
|   pCsr->aOffset = (int *)&pCsr->aChar[(nChar+3)&~3]; | ||||
| 
 | ||||
|   pCsr->aOffset[iOut] = iInput; | ||||
|   U8_NEXT(zInput, iInput, nInput, c);  | ||||
|   while( c>0 ){ | ||||
|     int isError = 0; | ||||
|     c = u_foldCase(c, opt); | ||||
|     U16_APPEND(pCsr->aChar, iOut, nChar, c, isError); | ||||
|     if( isError ){ | ||||
|       sqlite3_free(pCsr); | ||||
|       return SQLITE_ERROR; | ||||
|     } | ||||
|     pCsr->aOffset[iOut] = iInput; | ||||
| 
 | ||||
|     if( iInput<nInput ){ | ||||
|       U8_NEXT(zInput, iInput, nInput, c); | ||||
|     }else{ | ||||
|       c = 0; | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   pCsr->pIter = ubrk_open(UBRK_WORD, p->zLocale, pCsr->aChar, iOut, &status); | ||||
|   if( !U_SUCCESS(status) ){ | ||||
|     sqlite3_free(pCsr); | ||||
|     return SQLITE_ERROR; | ||||
|   } | ||||
|   pCsr->nChar = iOut; | ||||
| 
 | ||||
|   ubrk_first(pCsr->pIter); | ||||
|   *ppCursor = (sqlite3_tokenizer_cursor *)pCsr; | ||||
|   return SQLITE_OK; | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
| ** Close a tokenization cursor previously opened by a call to icuOpen(). | ||||
| */ | ||||
| static int icuClose(sqlite3_tokenizer_cursor *pCursor){ | ||||
|   IcuCursor *pCsr = (IcuCursor *)pCursor; | ||||
|   ubrk_close(pCsr->pIter); | ||||
|   sqlite3_free(pCsr->zBuffer); | ||||
|   sqlite3_free(pCsr); | ||||
|   return SQLITE_OK; | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
| ** Extract the next token from a tokenization cursor. | ||||
| */ | ||||
| static int icuNext( | ||||
|   sqlite3_tokenizer_cursor *pCursor,  /* Cursor returned by simpleOpen */ | ||||
|   const char **ppToken,               /* OUT: *ppToken is the token text */ | ||||
|   int *pnBytes,                       /* OUT: Number of bytes in token */ | ||||
|   int *piStartOffset,                 /* OUT: Starting offset of token */ | ||||
|   int *piEndOffset,                   /* OUT: Ending offset of token */ | ||||
|   int *piPosition                     /* OUT: Position integer of token */ | ||||
| ){ | ||||
|   IcuCursor *pCsr = (IcuCursor *)pCursor; | ||||
| 
 | ||||
|   int iStart = 0; | ||||
|   int iEnd = 0; | ||||
|   int nByte = 0; | ||||
| 
 | ||||
|   while( iStart==iEnd ){ | ||||
|     UChar32 c; | ||||
| 
 | ||||
|     iStart = ubrk_current(pCsr->pIter); | ||||
|     iEnd = ubrk_next(pCsr->pIter); | ||||
|     if( iEnd==UBRK_DONE ){ | ||||
|       return SQLITE_DONE; | ||||
|     } | ||||
| 
 | ||||
|     while( iStart<iEnd ){ | ||||
|       int iWhite = iStart; | ||||
|       U8_NEXT(pCsr->aChar, iWhite, pCsr->nChar, c); | ||||
|       if( u_isspace(c) ){ | ||||
|         iStart = iWhite; | ||||
|       }else{ | ||||
|         break; | ||||
|       } | ||||
|     } | ||||
|     assert(iStart<=iEnd); | ||||
|   } | ||||
| 
 | ||||
|   do { | ||||
|     UErrorCode status = U_ZERO_ERROR; | ||||
|     if( nByte ){ | ||||
|       char *zNew = sqlite3_realloc(pCsr->zBuffer, nByte); | ||||
|       if( !zNew ){ | ||||
|         return SQLITE_NOMEM; | ||||
|       } | ||||
|       pCsr->zBuffer = zNew; | ||||
|       pCsr->nBuffer = nByte; | ||||
|     } | ||||
| 
 | ||||
|     u_strToUTF8( | ||||
|         pCsr->zBuffer, pCsr->nBuffer, &nByte,    /* Output vars */ | ||||
|         &pCsr->aChar[iStart], iEnd-iStart,       /* Input vars */ | ||||
|         &status                                  /* Output success/failure */ | ||||
|     ); | ||||
|   } while( nByte>pCsr->nBuffer ); | ||||
| 
 | ||||
|   *ppToken = pCsr->zBuffer; | ||||
|   *pnBytes = nByte; | ||||
|   *piStartOffset = pCsr->aOffset[iStart]; | ||||
|   *piEndOffset = pCsr->aOffset[iEnd]; | ||||
|   *piPosition = pCsr->iToken++; | ||||
| 
 | ||||
|   return SQLITE_OK; | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
| ** The set of routines that implement the simple tokenizer | ||||
| */ | ||||
| static const sqlite3_tokenizer_module icuTokenizerModule = { | ||||
|   0,                           /* iVersion */ | ||||
|   icuCreate,                   /* xCreate  */ | ||||
|   icuDestroy,                  /* xCreate  */ | ||||
|   icuOpen,                     /* xOpen    */ | ||||
|   icuClose,                    /* xClose   */ | ||||
|   icuNext,                     /* xNext    */ | ||||
| }; | ||||
| 
 | ||||
| /*
 | ||||
| ** Set *ppModule to point at the implementation of the ICU tokenizer. | ||||
| */ | ||||
| void sqlite3Fts2IcuTokenizerModule( | ||||
|   sqlite3_tokenizer_module const**ppModule | ||||
| ){ | ||||
|   *ppModule = &icuTokenizerModule; | ||||
| } | ||||
| 
 | ||||
| #endif /* defined(SQLITE_ENABLE_ICU) */ | ||||
| #endif /* !defined(SQLITE_CORE) || defined(SQLITE_ENABLE_FTS2) */ | ||||
							
								
								
									
										644
									
								
								third_party/sqlite3/fts2_porter.c
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										644
									
								
								third_party/sqlite3/fts2_porter.c
									
										
									
									
										vendored
									
									
								
							|  | @ -1,644 +0,0 @@ | |||
| /*
 | ||||
| ** 2006 September 30 | ||||
| ** | ||||
| ** The author disclaims copyright to this source code.  In place of | ||||
| ** a legal notice, here is a blessing: | ||||
| ** | ||||
| **    May you do good and not evil. | ||||
| **    May you find forgiveness for yourself and forgive others. | ||||
| **    May you share freely, never taking more than you give. | ||||
| ** | ||||
| ************************************************************************* | ||||
| ** Implementation of the full-text-search tokenizer that implements | ||||
| ** a Porter stemmer. | ||||
| */ | ||||
| /*
 | ||||
| ** The code in this file is only compiled if: | ||||
| ** | ||||
| **     * The FTS2 module is being built as an extension | ||||
| **       (in which case SQLITE_CORE is not defined), or | ||||
| ** | ||||
| **     * The FTS2 module is being built into the core of | ||||
| **       SQLite (in which case SQLITE_ENABLE_FTS2 is defined). | ||||
| */ | ||||
| #if !defined(SQLITE_CORE) || defined(SQLITE_ENABLE_FTS2) | ||||
| 
 | ||||
| /* clang-format off */ | ||||
| 
 | ||||
| #include <assert.h> | ||||
| #include <stdlib.h> | ||||
| #include <stdio.h> | ||||
| #include <string.h> | ||||
| 
 | ||||
| #include "sqlite3.h" | ||||
| #include "sqlite3ext.h" | ||||
| SQLITE_EXTENSION_INIT3 | ||||
| #include "fts2_tokenizer.h" | ||||
| 
 | ||||
| /*
 | ||||
| ** Class derived from sqlite3_tokenizer | ||||
| */ | ||||
| typedef struct porter_tokenizer { | ||||
|   sqlite3_tokenizer base;      /* Base class */ | ||||
| } porter_tokenizer; | ||||
| 
 | ||||
| /*
 | ||||
| ** Class derived from sqlit3_tokenizer_cursor | ||||
| */ | ||||
| typedef struct porter_tokenizer_cursor { | ||||
|   sqlite3_tokenizer_cursor base; | ||||
|   const char *zInput;          /* input we are tokenizing */ | ||||
|   int nInput;                  /* size of the input */ | ||||
|   int iOffset;                 /* current position in zInput */ | ||||
|   int iToken;                  /* index of next token to be returned */ | ||||
|   char *zToken;                /* storage for current token */ | ||||
|   int nAllocated;              /* space allocated to zToken buffer */ | ||||
| } porter_tokenizer_cursor; | ||||
| 
 | ||||
| 
 | ||||
| /* Forward declaration */ | ||||
| static const sqlite3_tokenizer_module porterTokenizerModule; | ||||
| 
 | ||||
| 
 | ||||
| /*
 | ||||
| ** Create a new tokenizer instance. | ||||
| */ | ||||
| static int porterCreate( | ||||
|   int argc, const char * const *argv, | ||||
|   sqlite3_tokenizer **ppTokenizer | ||||
| ){ | ||||
|   porter_tokenizer *t; | ||||
|   t = (porter_tokenizer *) sqlite3_malloc(sizeof(*t)); | ||||
|   if( t==NULL ) return SQLITE_NOMEM; | ||||
|   memset(t, 0, sizeof(*t)); | ||||
|   *ppTokenizer = &t->base; | ||||
|   return SQLITE_OK; | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
| ** Destroy a tokenizer | ||||
| */ | ||||
| static int porterDestroy(sqlite3_tokenizer *pTokenizer){ | ||||
|   sqlite3_free(pTokenizer); | ||||
|   return SQLITE_OK; | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
| ** Prepare to begin tokenizing a particular string.  The input | ||||
| ** string to be tokenized is zInput[0..nInput-1].  A cursor | ||||
| ** used to incrementally tokenize this string is returned in  | ||||
| ** *ppCursor. | ||||
| */ | ||||
| static int porterOpen( | ||||
|   sqlite3_tokenizer *pTokenizer,         /* The tokenizer */ | ||||
|   const char *zInput, int nInput,        /* String to be tokenized */ | ||||
|   sqlite3_tokenizer_cursor **ppCursor    /* OUT: Tokenization cursor */ | ||||
| ){ | ||||
|   porter_tokenizer_cursor *c; | ||||
| 
 | ||||
|   c = (porter_tokenizer_cursor *) sqlite3_malloc(sizeof(*c)); | ||||
|   if( c==NULL ) return SQLITE_NOMEM; | ||||
| 
 | ||||
|   c->zInput = zInput; | ||||
|   if( zInput==0 ){ | ||||
|     c->nInput = 0; | ||||
|   }else if( nInput<0 ){ | ||||
|     c->nInput = (int)strlen(zInput); | ||||
|   }else{ | ||||
|     c->nInput = nInput; | ||||
|   } | ||||
|   c->iOffset = 0;                 /* start tokenizing at the beginning */ | ||||
|   c->iToken = 0; | ||||
|   c->zToken = NULL;               /* no space allocated, yet. */ | ||||
|   c->nAllocated = 0; | ||||
| 
 | ||||
|   *ppCursor = &c->base; | ||||
|   return SQLITE_OK; | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
| ** Close a tokenization cursor previously opened by a call to | ||||
| ** porterOpen() above. | ||||
| */ | ||||
| static int porterClose(sqlite3_tokenizer_cursor *pCursor){ | ||||
|   porter_tokenizer_cursor *c = (porter_tokenizer_cursor *) pCursor; | ||||
|   sqlite3_free(c->zToken); | ||||
|   sqlite3_free(c); | ||||
|   return SQLITE_OK; | ||||
| } | ||||
| /*
 | ||||
| ** Vowel or consonant | ||||
| */ | ||||
| static const char cType[] = { | ||||
|    0, 1, 1, 1, 0, 1, 1, 1, 0, 1, 1, 1, 1, 1, 0, 1, 1, 1, 1, 1, 0, | ||||
|    1, 1, 1, 2, 1 | ||||
| }; | ||||
| 
 | ||||
| /*
 | ||||
| ** isConsonant() and isVowel() determine if their first character in | ||||
| ** the string they point to is a consonant or a vowel, according | ||||
| ** to Porter ruls.   | ||||
| ** | ||||
| ** A consonate is any letter other than 'a', 'e', 'i', 'o', or 'u'. | ||||
| ** 'Y' is a consonant unless it follows another consonant, | ||||
| ** in which case it is a vowel. | ||||
| ** | ||||
| ** In these routine, the letters are in reverse order.  So the 'y' rule | ||||
| ** is that 'y' is a consonant unless it is followed by another | ||||
| ** consonent. | ||||
| */ | ||||
| static int isVowel(const char*); | ||||
| static int isConsonant(const char *z){ | ||||
|   int j; | ||||
|   char x = *z; | ||||
|   if( x==0 ) return 0; | ||||
|   assert( x>='a' && x<='z' ); | ||||
|   j = cType[x-'a']; | ||||
|   if( j<2 ) return j; | ||||
|   return z[1]==0 || isVowel(z + 1); | ||||
| } | ||||
| static int isVowel(const char *z){ | ||||
|   int j; | ||||
|   char x = *z; | ||||
|   if( x==0 ) return 0; | ||||
|   assert( x>='a' && x<='z' ); | ||||
|   j = cType[x-'a']; | ||||
|   if( j<2 ) return 1-j; | ||||
|   return isConsonant(z + 1); | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
| ** Let any sequence of one or more vowels be represented by V and let | ||||
| ** C be sequence of one or more consonants.  Then every word can be | ||||
| ** represented as: | ||||
| ** | ||||
| **           [C] (VC){m} [V] | ||||
| ** | ||||
| ** In prose:  A word is an optional consonant followed by zero or | ||||
| ** vowel-consonant pairs followed by an optional vowel.  "m" is the | ||||
| ** number of vowel consonant pairs.  This routine computes the value | ||||
| ** of m for the first i bytes of a word. | ||||
| ** | ||||
| ** Return true if the m-value for z is 1 or more.  In other words, | ||||
| ** return true if z contains at least one vowel that is followed | ||||
| ** by a consonant. | ||||
| ** | ||||
| ** In this routine z[] is in reverse order.  So we are really looking | ||||
| ** for an instance of of a consonant followed by a vowel. | ||||
| */ | ||||
| static int m_gt_0(const char *z){ | ||||
|   while( isVowel(z) ){ z++; } | ||||
|   if( *z==0 ) return 0; | ||||
|   while( isConsonant(z) ){ z++; } | ||||
|   return *z!=0; | ||||
| } | ||||
| 
 | ||||
| /* Like mgt0 above except we are looking for a value of m which is
 | ||||
| ** exactly 1 | ||||
| */ | ||||
| static int m_eq_1(const char *z){ | ||||
|   while( isVowel(z) ){ z++; } | ||||
|   if( *z==0 ) return 0; | ||||
|   while( isConsonant(z) ){ z++; } | ||||
|   if( *z==0 ) return 0; | ||||
|   while( isVowel(z) ){ z++; } | ||||
|   if( *z==0 ) return 1; | ||||
|   while( isConsonant(z) ){ z++; } | ||||
|   return *z==0; | ||||
| } | ||||
| 
 | ||||
| /* Like mgt0 above except we are looking for a value of m>1 instead
 | ||||
| ** or m>0 | ||||
| */ | ||||
| static int m_gt_1(const char *z){ | ||||
|   while( isVowel(z) ){ z++; } | ||||
|   if( *z==0 ) return 0; | ||||
|   while( isConsonant(z) ){ z++; } | ||||
|   if( *z==0 ) return 0; | ||||
|   while( isVowel(z) ){ z++; } | ||||
|   if( *z==0 ) return 0; | ||||
|   while( isConsonant(z) ){ z++; } | ||||
|   return *z!=0; | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
| ** Return TRUE if there is a vowel anywhere within z[0..n-1] | ||||
| */ | ||||
| static int hasVowel(const char *z){ | ||||
|   while( isConsonant(z) ){ z++; } | ||||
|   return *z!=0; | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
| ** Return TRUE if the word ends in a double consonant. | ||||
| ** | ||||
| ** The text is reversed here. So we are really looking at | ||||
| ** the first two characters of z[]. | ||||
| */ | ||||
| static int doubleConsonant(const char *z){ | ||||
|   return isConsonant(z) && z[0]==z[1] && isConsonant(z+1); | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
| ** Return TRUE if the word ends with three letters which | ||||
| ** are consonant-vowel-consonent and where the final consonant | ||||
| ** is not 'w', 'x', or 'y'. | ||||
| ** | ||||
| ** The word is reversed here.  So we are really checking the | ||||
| ** first three letters and the first one cannot be in [wxy]. | ||||
| */ | ||||
| static int star_oh(const char *z){ | ||||
|   return | ||||
|     z[0]!=0 && isConsonant(z) && | ||||
|     z[0]!='w' && z[0]!='x' && z[0]!='y' && | ||||
|     z[1]!=0 && isVowel(z+1) && | ||||
|     z[2]!=0 && isConsonant(z+2); | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
| ** If the word ends with zFrom and xCond() is true for the stem | ||||
| ** of the word that preceeds the zFrom ending, then change the  | ||||
| ** ending to zTo. | ||||
| ** | ||||
| ** The input word *pz and zFrom are both in reverse order.  zTo | ||||
| ** is in normal order.  | ||||
| ** | ||||
| ** Return TRUE if zFrom matches.  Return FALSE if zFrom does not | ||||
| ** match.  Not that TRUE is returned even if xCond() fails and | ||||
| ** no substitution occurs. | ||||
| */ | ||||
| static int stem( | ||||
|   char **pz,             /* The word being stemmed (Reversed) */ | ||||
|   const char *zFrom,     /* If the ending matches this... (Reversed) */ | ||||
|   const char *zTo,       /* ... change the ending to this (not reversed) */ | ||||
|   int (*xCond)(const char*)   /* Condition that must be true */ | ||||
| ){ | ||||
|   char *z = *pz; | ||||
|   while( *zFrom && *zFrom==*z ){ z++; zFrom++; } | ||||
|   if( *zFrom!=0 ) return 0; | ||||
|   if( xCond && !xCond(z) ) return 1; | ||||
|   while( *zTo ){ | ||||
|     *(--z) = *(zTo++); | ||||
|   } | ||||
|   *pz = z; | ||||
|   return 1; | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
| ** This is the fallback stemmer used when the porter stemmer is | ||||
| ** inappropriate.  The input word is copied into the output with | ||||
| ** US-ASCII case folding.  If the input word is too long (more | ||||
| ** than 20 bytes if it contains no digits or more than 6 bytes if | ||||
| ** it contains digits) then word is truncated to 20 or 6 bytes | ||||
| ** by taking 10 or 3 bytes from the beginning and end. | ||||
| */ | ||||
| static void copy_stemmer(const char *zIn, int nIn, char *zOut, int *pnOut){ | ||||
|   int i, mx, j; | ||||
|   int hasDigit = 0; | ||||
|   for(i=0; i<nIn; i++){ | ||||
|     int c = zIn[i]; | ||||
|     if( c>='A' && c<='Z' ){ | ||||
|       zOut[i] = c - 'A' + 'a'; | ||||
|     }else{ | ||||
|       if( c>='0' && c<='9' ) hasDigit = 1; | ||||
|       zOut[i] = c; | ||||
|     } | ||||
|   } | ||||
|   mx = hasDigit ? 3 : 10; | ||||
|   if( nIn>mx*2 ){ | ||||
|     for(j=mx, i=nIn-mx; i<nIn; i++, j++){ | ||||
|       zOut[j] = zOut[i]; | ||||
|     } | ||||
|     i = j; | ||||
|   } | ||||
|   zOut[i] = 0; | ||||
|   *pnOut = i; | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| /*
 | ||||
| ** Stem the input word zIn[0..nIn-1].  Store the output in zOut. | ||||
| ** zOut is at least big enough to hold nIn bytes.  Write the actual | ||||
| ** size of the output word (exclusive of the '\0' terminator) into *pnOut. | ||||
| ** | ||||
| ** Any upper-case characters in the US-ASCII character set ([A-Z]) | ||||
| ** are converted to lower case.  Upper-case UTF characters are | ||||
| ** unchanged. | ||||
| ** | ||||
| ** Words that are longer than about 20 bytes are stemmed by retaining | ||||
| ** a few bytes from the beginning and the end of the word.  If the | ||||
| ** word contains digits, 3 bytes are taken from the beginning and | ||||
| ** 3 bytes from the end.  For long words without digits, 10 bytes | ||||
| ** are taken from each end.  US-ASCII case folding still applies. | ||||
| **  | ||||
| ** If the input word contains not digits but does characters not  | ||||
| ** in [a-zA-Z] then no stemming is attempted and this routine just  | ||||
| ** copies the input into the input into the output with US-ASCII | ||||
| ** case folding. | ||||
| ** | ||||
| ** Stemming never increases the length of the word.  So there is | ||||
| ** no chance of overflowing the zOut buffer. | ||||
| */ | ||||
| static void porter_stemmer(const char *zIn, int nIn, char *zOut, int *pnOut){ | ||||
|   int i, j, c; | ||||
|   char zReverse[28]; | ||||
|   char *z, *z2; | ||||
|   if( nIn<3 || nIn>=sizeof(zReverse)-7 ){ | ||||
|     /* The word is too big or too small for the porter stemmer.
 | ||||
|     ** Fallback to the copy stemmer */ | ||||
|     copy_stemmer(zIn, nIn, zOut, pnOut); | ||||
|     return; | ||||
|   } | ||||
|   for(i=0, j=sizeof(zReverse)-6; i<nIn; i++, j--){ | ||||
|     c = zIn[i]; | ||||
|     if( c>='A' && c<='Z' ){ | ||||
|       zReverse[j] = c + 'a' - 'A'; | ||||
|     }else if( c>='a' && c<='z' ){ | ||||
|       zReverse[j] = c; | ||||
|     }else{ | ||||
|       /* The use of a character not in [a-zA-Z] means that we fallback
 | ||||
|       ** to the copy stemmer */ | ||||
|       copy_stemmer(zIn, nIn, zOut, pnOut); | ||||
|       return; | ||||
|     } | ||||
|   } | ||||
|   memset(&zReverse[sizeof(zReverse)-5], 0, 5); | ||||
|   z = &zReverse[j+1]; | ||||
| 
 | ||||
| 
 | ||||
|   /* Step 1a */ | ||||
|   if( z[0]=='s' ){ | ||||
|     if( | ||||
|      !stem(&z, "sess", "ss", 0) && | ||||
|      !stem(&z, "sei", "i", 0)  && | ||||
|      !stem(&z, "ss", "ss", 0) | ||||
|     ){ | ||||
|       z++; | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   /* Step 1b */   | ||||
|   z2 = z; | ||||
|   if( stem(&z, "dee", "ee", m_gt_0) ){ | ||||
|     /* Do nothing.  The work was all in the test */ | ||||
|   }else if(  | ||||
|      (stem(&z, "gni", "", hasVowel) || stem(&z, "de", "", hasVowel)) | ||||
|       && z!=z2 | ||||
|   ){ | ||||
|      if( stem(&z, "ta", "ate", 0) || | ||||
|          stem(&z, "lb", "ble", 0) || | ||||
|          stem(&z, "zi", "ize", 0) ){ | ||||
|        /* Do nothing.  The work was all in the test */ | ||||
|      }else if( doubleConsonant(z) && (*z!='l' && *z!='s' && *z!='z') ){ | ||||
|        z++; | ||||
|      }else if( m_eq_1(z) && star_oh(z) ){ | ||||
|        *(--z) = 'e'; | ||||
|      } | ||||
|   } | ||||
| 
 | ||||
|   /* Step 1c */ | ||||
|   if( z[0]=='y' && hasVowel(z+1) ){ | ||||
|     z[0] = 'i'; | ||||
|   } | ||||
| 
 | ||||
|   /* Step 2 */ | ||||
|   switch( z[1] ){ | ||||
|    case 'a': | ||||
|      stem(&z, "lanoita", "ate", m_gt_0) || | ||||
|      stem(&z, "lanoit", "tion", m_gt_0); | ||||
|      break; | ||||
|    case 'c': | ||||
|      stem(&z, "icne", "ence", m_gt_0) || | ||||
|      stem(&z, "icna", "ance", m_gt_0); | ||||
|      break; | ||||
|    case 'e': | ||||
|      stem(&z, "rezi", "ize", m_gt_0); | ||||
|      break; | ||||
|    case 'g': | ||||
|      stem(&z, "igol", "log", m_gt_0); | ||||
|      break; | ||||
|    case 'l': | ||||
|      stem(&z, "ilb", "ble", m_gt_0) || | ||||
|      stem(&z, "illa", "al", m_gt_0) || | ||||
|      stem(&z, "iltne", "ent", m_gt_0) || | ||||
|      stem(&z, "ile", "e", m_gt_0) || | ||||
|      stem(&z, "ilsuo", "ous", m_gt_0); | ||||
|      break; | ||||
|    case 'o': | ||||
|      stem(&z, "noitazi", "ize", m_gt_0) || | ||||
|      stem(&z, "noita", "ate", m_gt_0) || | ||||
|      stem(&z, "rota", "ate", m_gt_0); | ||||
|      break; | ||||
|    case 's': | ||||
|      stem(&z, "msila", "al", m_gt_0) || | ||||
|      stem(&z, "ssenevi", "ive", m_gt_0) || | ||||
|      stem(&z, "ssenluf", "ful", m_gt_0) || | ||||
|      stem(&z, "ssensuo", "ous", m_gt_0); | ||||
|      break; | ||||
|    case 't': | ||||
|      stem(&z, "itila", "al", m_gt_0) || | ||||
|      stem(&z, "itivi", "ive", m_gt_0) || | ||||
|      stem(&z, "itilib", "ble", m_gt_0); | ||||
|      break; | ||||
|   } | ||||
| 
 | ||||
|   /* Step 3 */ | ||||
|   switch( z[0] ){ | ||||
|    case 'e': | ||||
|      stem(&z, "etaci", "ic", m_gt_0) || | ||||
|      stem(&z, "evita", "", m_gt_0)   || | ||||
|      stem(&z, "ezila", "al", m_gt_0); | ||||
|      break; | ||||
|    case 'i': | ||||
|      stem(&z, "itici", "ic", m_gt_0); | ||||
|      break; | ||||
|    case 'l': | ||||
|      stem(&z, "laci", "ic", m_gt_0) || | ||||
|      stem(&z, "luf", "", m_gt_0); | ||||
|      break; | ||||
|    case 's': | ||||
|      stem(&z, "ssen", "", m_gt_0); | ||||
|      break; | ||||
|   } | ||||
| 
 | ||||
|   /* Step 4 */ | ||||
|   switch( z[1] ){ | ||||
|    case 'a': | ||||
|      if( z[0]=='l' && m_gt_1(z+2) ){ | ||||
|        z += 2; | ||||
|      } | ||||
|      break; | ||||
|    case 'c': | ||||
|      if( z[0]=='e' && z[2]=='n' && (z[3]=='a' || z[3]=='e')  && m_gt_1(z+4)  ){ | ||||
|        z += 4; | ||||
|      } | ||||
|      break; | ||||
|    case 'e': | ||||
|      if( z[0]=='r' && m_gt_1(z+2) ){ | ||||
|        z += 2; | ||||
|      } | ||||
|      break; | ||||
|    case 'i': | ||||
|      if( z[0]=='c' && m_gt_1(z+2) ){ | ||||
|        z += 2; | ||||
|      } | ||||
|      break; | ||||
|    case 'l': | ||||
|      if( z[0]=='e' && z[2]=='b' && (z[3]=='a' || z[3]=='i') && m_gt_1(z+4) ){ | ||||
|        z += 4; | ||||
|      } | ||||
|      break; | ||||
|    case 'n': | ||||
|      if( z[0]=='t' ){ | ||||
|        if( z[2]=='a' ){ | ||||
|          if( m_gt_1(z+3) ){ | ||||
|            z += 3; | ||||
|          } | ||||
|        }else if( z[2]=='e' ){ | ||||
|          stem(&z, "tneme", "", m_gt_1) || | ||||
|          stem(&z, "tnem", "", m_gt_1) || | ||||
|          stem(&z, "tne", "", m_gt_1); | ||||
|        } | ||||
|      } | ||||
|      break; | ||||
|    case 'o': | ||||
|      if( z[0]=='u' ){ | ||||
|        if( m_gt_1(z+2) ){ | ||||
|          z += 2; | ||||
|        } | ||||
|      }else if( z[3]=='s' || z[3]=='t' ){ | ||||
|        stem(&z, "noi", "", m_gt_1); | ||||
|      } | ||||
|      break; | ||||
|    case 's': | ||||
|      if( z[0]=='m' && z[2]=='i' && m_gt_1(z+3) ){ | ||||
|        z += 3; | ||||
|      } | ||||
|      break; | ||||
|    case 't': | ||||
|      stem(&z, "eta", "", m_gt_1) || | ||||
|      stem(&z, "iti", "", m_gt_1); | ||||
|      break; | ||||
|    case 'u': | ||||
|      if( z[0]=='s' && z[2]=='o' && m_gt_1(z+3) ){ | ||||
|        z += 3; | ||||
|      } | ||||
|      break; | ||||
|    case 'v': | ||||
|    case 'z': | ||||
|      if( z[0]=='e' && z[2]=='i' && m_gt_1(z+3) ){ | ||||
|        z += 3; | ||||
|      } | ||||
|      break; | ||||
|   } | ||||
| 
 | ||||
|   /* Step 5a */ | ||||
|   if( z[0]=='e' ){ | ||||
|     if( m_gt_1(z+1) ){ | ||||
|       z++; | ||||
|     }else if( m_eq_1(z+1) && !star_oh(z+1) ){ | ||||
|       z++; | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   /* Step 5b */ | ||||
|   if( m_gt_1(z) && z[0]=='l' && z[1]=='l' ){ | ||||
|     z++; | ||||
|   } | ||||
| 
 | ||||
|   /* z[] is now the stemmed word in reverse order.  Flip it back
 | ||||
|   ** around into forward order and return. | ||||
|   */ | ||||
|   *pnOut = i = strlen(z); | ||||
|   zOut[i] = 0; | ||||
|   while( *z ){ | ||||
|     zOut[--i] = *(z++); | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
| ** Characters that can be part of a token.  We assume any character | ||||
| ** whose value is greater than 0x80 (any UTF character) can be | ||||
| ** part of a token.  In other words, delimiters all must have | ||||
| ** values of 0x7f or lower. | ||||
| */ | ||||
| static const char porterIdChar[] = { | ||||
| /* x0 x1 x2 x3 x4 x5 x6 x7 x8 x9 xA xB xC xD xE xF */ | ||||
|     1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, 0, 0, 0, 0, 0,  /* 3x */ | ||||
|     0, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,  /* 4x */ | ||||
|     1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, 0, 0, 0, 1,  /* 5x */ | ||||
|     0, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,  /* 6x */ | ||||
|     1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, 0, 0, 0, 0,  /* 7x */ | ||||
| }; | ||||
| #define isDelim(C) (((ch=C)&0x80)==0 && (ch<0x30 || !porterIdChar[ch-0x30])) | ||||
| 
 | ||||
| /*
 | ||||
| ** Extract the next token from a tokenization cursor.  The cursor must | ||||
| ** have been opened by a prior call to porterOpen(). | ||||
| */ | ||||
| static int porterNext( | ||||
|   sqlite3_tokenizer_cursor *pCursor,  /* Cursor returned by porterOpen */ | ||||
|   const char **pzToken,               /* OUT: *pzToken is the token text */ | ||||
|   int *pnBytes,                       /* OUT: Number of bytes in token */ | ||||
|   int *piStartOffset,                 /* OUT: Starting offset of token */ | ||||
|   int *piEndOffset,                   /* OUT: Ending offset of token */ | ||||
|   int *piPosition                     /* OUT: Position integer of token */ | ||||
| ){ | ||||
|   porter_tokenizer_cursor *c = (porter_tokenizer_cursor *) pCursor; | ||||
|   const char *z = c->zInput; | ||||
| 
 | ||||
|   while( c->iOffset<c->nInput ){ | ||||
|     int iStartOffset, ch; | ||||
| 
 | ||||
|     /* Scan past delimiter characters */ | ||||
|     while( c->iOffset<c->nInput && isDelim(z[c->iOffset]) ){ | ||||
|       c->iOffset++; | ||||
|     } | ||||
| 
 | ||||
|     /* Count non-delimiter characters. */ | ||||
|     iStartOffset = c->iOffset; | ||||
|     while( c->iOffset<c->nInput && !isDelim(z[c->iOffset]) ){ | ||||
|       c->iOffset++; | ||||
|     } | ||||
| 
 | ||||
|     if( c->iOffset>iStartOffset ){ | ||||
|       int n = c->iOffset-iStartOffset; | ||||
|       if( n>c->nAllocated ){ | ||||
|         c->nAllocated = n+20; | ||||
|         c->zToken = sqlite3_realloc(c->zToken, c->nAllocated); | ||||
|         if( c->zToken==NULL ) return SQLITE_NOMEM; | ||||
|       } | ||||
|       porter_stemmer(&z[iStartOffset], n, c->zToken, pnBytes); | ||||
|       *pzToken = c->zToken; | ||||
|       *piStartOffset = iStartOffset; | ||||
|       *piEndOffset = c->iOffset; | ||||
|       *piPosition = c->iToken++; | ||||
|       return SQLITE_OK; | ||||
|     } | ||||
|   } | ||||
|   return SQLITE_DONE; | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
| ** The set of routines that implement the porter-stemmer tokenizer | ||||
| */ | ||||
| static const sqlite3_tokenizer_module porterTokenizerModule = { | ||||
|   0, | ||||
|   porterCreate, | ||||
|   porterDestroy, | ||||
|   porterOpen, | ||||
|   porterClose, | ||||
|   porterNext, | ||||
| }; | ||||
| 
 | ||||
| /*
 | ||||
| ** Allocate a new porter tokenizer.  Return a pointer to the new | ||||
| ** tokenizer in *ppModule | ||||
| */ | ||||
| void sqlite3Fts2PorterTokenizerModule( | ||||
|   sqlite3_tokenizer_module const**ppModule | ||||
| ){ | ||||
|   *ppModule = &porterTokenizerModule; | ||||
| } | ||||
| 
 | ||||
| #endif /* !defined(SQLITE_CORE) || defined(SQLITE_ENABLE_FTS2) */ | ||||
							
								
								
									
										375
									
								
								third_party/sqlite3/fts2_tokenizer.c
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										375
									
								
								third_party/sqlite3/fts2_tokenizer.c
									
										
									
									
										vendored
									
									
								
							|  | @ -1,375 +0,0 @@ | |||
| /*
 | ||||
| ** 2007 June 22 | ||||
| ** | ||||
| ** The author disclaims copyright to this source code.  In place of | ||||
| ** a legal notice, here is a blessing: | ||||
| ** | ||||
| **    May you do good and not evil. | ||||
| **    May you find forgiveness for yourself and forgive others. | ||||
| **    May you share freely, never taking more than you give. | ||||
| ** | ||||
| ****************************************************************************** | ||||
| ** | ||||
| ** This is part of an SQLite module implementing full-text search. | ||||
| ** This particular file implements the generic tokenizer interface. | ||||
| */ | ||||
| /*
 | ||||
| ** The code in this file is only compiled if: | ||||
| ** | ||||
| **     * The FTS2 module is being built as an extension | ||||
| **       (in which case SQLITE_CORE is not defined), or | ||||
| ** | ||||
| **     * The FTS2 module is being built into the core of | ||||
| **       SQLite (in which case SQLITE_ENABLE_FTS2 is defined). | ||||
| */ | ||||
| #if !defined(SQLITE_CORE) || defined(SQLITE_ENABLE_FTS2) | ||||
| 
 | ||||
| /* clang-format off */ | ||||
| 
 | ||||
| #include "sqlite3.h" | ||||
| #include "sqlite3ext.h" | ||||
| SQLITE_EXTENSION_INIT3 | ||||
| 
 | ||||
| #include "fts2_hash.h" | ||||
| #include "fts2_tokenizer.h" | ||||
| #include <assert.h> | ||||
| 
 | ||||
| /*
 | ||||
| ** Implementation of the SQL scalar function for accessing the underlying  | ||||
| ** hash table. This function may be called as follows: | ||||
| ** | ||||
| **   SELECT <function-name>(<key-name>); | ||||
| **   SELECT <function-name>(<key-name>, <pointer>); | ||||
| ** | ||||
| ** where <function-name> is the name passed as the second argument | ||||
| ** to the sqlite3Fts2InitHashTable() function (e.g. 'fts2_tokenizer'). | ||||
| ** | ||||
| ** If the <pointer> argument is specified, it must be a blob value | ||||
| ** containing a pointer to be stored as the hash data corresponding | ||||
| ** to the string <key-name>. If <pointer> is not specified, then | ||||
| ** the string <key-name> must already exist in the has table. Otherwise, | ||||
| ** an error is returned. | ||||
| ** | ||||
| ** Whether or not the <pointer> argument is specified, the value returned | ||||
| ** is a blob containing the pointer stored as the hash data corresponding | ||||
| ** to string <key-name> (after the hash-table is updated, if applicable). | ||||
| */ | ||||
| static void scalarFunc( | ||||
|   sqlite3_context *context, | ||||
|   int argc, | ||||
|   sqlite3_value **argv | ||||
| ){ | ||||
|   fts2Hash *pHash; | ||||
|   void *pPtr = 0; | ||||
|   const unsigned char *zName; | ||||
|   int nName; | ||||
| 
 | ||||
|   assert( argc==1 || argc==2 ); | ||||
| 
 | ||||
|   pHash = (fts2Hash *)sqlite3_user_data(context); | ||||
| 
 | ||||
|   zName = sqlite3_value_text(argv[0]); | ||||
|   nName = sqlite3_value_bytes(argv[0])+1; | ||||
| 
 | ||||
|   if( argc==2 ){ | ||||
|     void *pOld; | ||||
|     int n = sqlite3_value_bytes(argv[1]); | ||||
|     if( n!=sizeof(pPtr) ){ | ||||
|       sqlite3_result_error(context, "argument type mismatch", -1); | ||||
|       return; | ||||
|     } | ||||
|     pPtr = *(void **)sqlite3_value_blob(argv[1]); | ||||
|     pOld = sqlite3Fts2HashInsert(pHash, (void *)zName, nName, pPtr); | ||||
|     if( pOld==pPtr ){ | ||||
|       sqlite3_result_error(context, "out of memory", -1); | ||||
|       return; | ||||
|     } | ||||
|   }else{ | ||||
|     pPtr = sqlite3Fts2HashFind(pHash, zName, nName); | ||||
|     if( !pPtr ){ | ||||
|       char *zErr = sqlite3_mprintf("unknown tokenizer: %s", zName); | ||||
|       sqlite3_result_error(context, zErr, -1); | ||||
|       sqlite3_free(zErr); | ||||
|       return; | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   sqlite3_result_blob(context, (void *)&pPtr, sizeof(pPtr), SQLITE_TRANSIENT); | ||||
| } | ||||
| 
 | ||||
| #ifdef SQLITE_TEST | ||||
| 
 | ||||
| #if defined(INCLUDE_SQLITE_TCL_H) | ||||
| #  include "sqlite_tcl.h" | ||||
| #else | ||||
| #  include "tcl.h" | ||||
| #endif | ||||
| #include <string.h> | ||||
| 
 | ||||
| /*
 | ||||
| ** Implementation of a special SQL scalar function for testing tokenizers  | ||||
| ** designed to be used in concert with the Tcl testing framework. This | ||||
| ** function must be called with two arguments: | ||||
| ** | ||||
| **   SELECT <function-name>(<key-name>, <input-string>); | ||||
| **   SELECT <function-name>(<key-name>, <pointer>); | ||||
| ** | ||||
| ** where <function-name> is the name passed as the second argument | ||||
| ** to the sqlite3Fts2InitHashTable() function (e.g. 'fts2_tokenizer') | ||||
| ** concatenated with the string '_test' (e.g. 'fts2_tokenizer_test'). | ||||
| ** | ||||
| ** The return value is a string that may be interpreted as a Tcl | ||||
| ** list. For each token in the <input-string>, three elements are | ||||
| ** added to the returned list. The first is the token position, the  | ||||
| ** second is the token text (folded, stemmed, etc.) and the third is the | ||||
| ** substring of <input-string> associated with the token. For example,  | ||||
| ** using the built-in "simple" tokenizer: | ||||
| ** | ||||
| **   SELECT fts_tokenizer_test('simple', 'I don't see how'); | ||||
| ** | ||||
| ** will return the string: | ||||
| ** | ||||
| **   "{0 i I 1 dont don't 2 see see 3 how how}" | ||||
| **    | ||||
| */ | ||||
| static void testFunc( | ||||
|   sqlite3_context *context, | ||||
|   int argc, | ||||
|   sqlite3_value **argv | ||||
| ){ | ||||
|   fts2Hash *pHash; | ||||
|   sqlite3_tokenizer_module *p; | ||||
|   sqlite3_tokenizer *pTokenizer = 0; | ||||
|   sqlite3_tokenizer_cursor *pCsr = 0; | ||||
| 
 | ||||
|   const char *zErr = 0; | ||||
| 
 | ||||
|   const char *zName; | ||||
|   int nName; | ||||
|   const char *zInput; | ||||
|   int nInput; | ||||
| 
 | ||||
|   const char *zArg = 0; | ||||
| 
 | ||||
|   const char *zToken; | ||||
|   int nToken; | ||||
|   int iStart; | ||||
|   int iEnd; | ||||
|   int iPos; | ||||
| 
 | ||||
|   Tcl_Obj *pRet; | ||||
| 
 | ||||
|   assert( argc==2 || argc==3 ); | ||||
| 
 | ||||
|   nName = sqlite3_value_bytes(argv[0]); | ||||
|   zName = (const char *)sqlite3_value_text(argv[0]); | ||||
|   nInput = sqlite3_value_bytes(argv[argc-1]); | ||||
|   zInput = (const char *)sqlite3_value_text(argv[argc-1]); | ||||
| 
 | ||||
|   if( argc==3 ){ | ||||
|     zArg = (const char *)sqlite3_value_text(argv[1]); | ||||
|   } | ||||
| 
 | ||||
|   pHash = (fts2Hash *)sqlite3_user_data(context); | ||||
|   p = (sqlite3_tokenizer_module *)sqlite3Fts2HashFind(pHash, zName, nName+1); | ||||
| 
 | ||||
|   if( !p ){ | ||||
|     char *zErr = sqlite3_mprintf("unknown tokenizer: %s", zName); | ||||
|     sqlite3_result_error(context, zErr, -1); | ||||
|     sqlite3_free(zErr); | ||||
|     return; | ||||
|   } | ||||
| 
 | ||||
|   pRet = Tcl_NewObj(); | ||||
|   Tcl_IncrRefCount(pRet); | ||||
| 
 | ||||
|   if( SQLITE_OK!=p->xCreate(zArg ? 1 : 0, &zArg, &pTokenizer) ){ | ||||
|     zErr = "error in xCreate()"; | ||||
|     goto finish; | ||||
|   } | ||||
|   pTokenizer->pModule = p; | ||||
|   if( SQLITE_OK!=p->xOpen(pTokenizer, zInput, nInput, &pCsr) ){ | ||||
|     zErr = "error in xOpen()"; | ||||
|     goto finish; | ||||
|   } | ||||
|   pCsr->pTokenizer = pTokenizer; | ||||
| 
 | ||||
|   while( SQLITE_OK==p->xNext(pCsr, &zToken, &nToken, &iStart, &iEnd, &iPos) ){ | ||||
|     Tcl_ListObjAppendElement(0, pRet, Tcl_NewIntObj(iPos)); | ||||
|     Tcl_ListObjAppendElement(0, pRet, Tcl_NewStringObj(zToken, nToken)); | ||||
|     zToken = &zInput[iStart]; | ||||
|     nToken = iEnd-iStart; | ||||
|     Tcl_ListObjAppendElement(0, pRet, Tcl_NewStringObj(zToken, nToken)); | ||||
|   } | ||||
| 
 | ||||
|   if( SQLITE_OK!=p->xClose(pCsr) ){ | ||||
|     zErr = "error in xClose()"; | ||||
|     goto finish; | ||||
|   } | ||||
|   if( SQLITE_OK!=p->xDestroy(pTokenizer) ){ | ||||
|     zErr = "error in xDestroy()"; | ||||
|     goto finish; | ||||
|   } | ||||
| 
 | ||||
| finish: | ||||
|   if( zErr ){ | ||||
|     sqlite3_result_error(context, zErr, -1); | ||||
|   }else{ | ||||
|     sqlite3_result_text(context, Tcl_GetString(pRet), -1, SQLITE_TRANSIENT); | ||||
|   } | ||||
|   Tcl_DecrRefCount(pRet); | ||||
| } | ||||
| 
 | ||||
| static | ||||
| int registerTokenizer( | ||||
|   sqlite3 *db,  | ||||
|   char *zName,  | ||||
|   const sqlite3_tokenizer_module *p | ||||
| ){ | ||||
|   int rc; | ||||
|   sqlite3_stmt *pStmt; | ||||
|   const char zSql[] = "SELECT fts2_tokenizer(?, ?)"; | ||||
| 
 | ||||
|   rc = sqlite3_prepare_v2(db, zSql, -1, &pStmt, 0); | ||||
|   if( rc!=SQLITE_OK ){ | ||||
|     return rc; | ||||
|   } | ||||
| 
 | ||||
|   sqlite3_bind_text(pStmt, 1, zName, -1, SQLITE_STATIC); | ||||
|   sqlite3_bind_blob(pStmt, 2, &p, sizeof(p), SQLITE_STATIC); | ||||
|   sqlite3_step(pStmt); | ||||
| 
 | ||||
|   return sqlite3_finalize(pStmt); | ||||
| } | ||||
| 
 | ||||
| static | ||||
| int queryFts2Tokenizer( | ||||
|   sqlite3 *db,  | ||||
|   char *zName,   | ||||
|   const sqlite3_tokenizer_module **pp | ||||
| ){ | ||||
|   int rc; | ||||
|   sqlite3_stmt *pStmt; | ||||
|   const char zSql[] = "SELECT fts2_tokenizer(?)"; | ||||
| 
 | ||||
|   *pp = 0; | ||||
|   rc = sqlite3_prepare_v2(db, zSql, -1, &pStmt, 0); | ||||
|   if( rc!=SQLITE_OK ){ | ||||
|     return rc; | ||||
|   } | ||||
| 
 | ||||
|   sqlite3_bind_text(pStmt, 1, zName, -1, SQLITE_STATIC); | ||||
|   if( SQLITE_ROW==sqlite3_step(pStmt) ){ | ||||
|     if( sqlite3_column_type(pStmt, 0)==SQLITE_BLOB ){ | ||||
|       memcpy(pp, sqlite3_column_blob(pStmt, 0), sizeof(*pp)); | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   return sqlite3_finalize(pStmt); | ||||
| } | ||||
| 
 | ||||
| void sqlite3Fts2SimpleTokenizerModule(sqlite3_tokenizer_module const**ppModule); | ||||
| 
 | ||||
| /*
 | ||||
| ** Implementation of the scalar function fts2_tokenizer_internal_test(). | ||||
| ** This function is used for testing only, it is not included in the | ||||
| ** build unless SQLITE_TEST is defined. | ||||
| ** | ||||
| ** The purpose of this is to test that the fts2_tokenizer() function | ||||
| ** can be used as designed by the C-code in the queryFts2Tokenizer and | ||||
| ** registerTokenizer() functions above. These two functions are repeated | ||||
| ** in the README.tokenizer file as an example, so it is important to | ||||
| ** test them. | ||||
| ** | ||||
| ** To run the tests, evaluate the fts2_tokenizer_internal_test() scalar | ||||
| ** function with no arguments. An assert() will fail if a problem is | ||||
| ** detected. i.e.: | ||||
| ** | ||||
| **     SELECT fts2_tokenizer_internal_test(); | ||||
| ** | ||||
| */ | ||||
| static void intTestFunc( | ||||
|   sqlite3_context *context, | ||||
|   int argc, | ||||
|   sqlite3_value **argv | ||||
| ){ | ||||
|   int rc; | ||||
|   const sqlite3_tokenizer_module *p1; | ||||
|   const sqlite3_tokenizer_module *p2; | ||||
|   sqlite3 *db = (sqlite3 *)sqlite3_user_data(context); | ||||
| 
 | ||||
|   /* Test the query function */ | ||||
|   sqlite3Fts2SimpleTokenizerModule(&p1); | ||||
|   rc = queryFts2Tokenizer(db, "simple", &p2); | ||||
|   assert( rc==SQLITE_OK ); | ||||
|   assert( p1==p2 ); | ||||
|   rc = queryFts2Tokenizer(db, "nosuchtokenizer", &p2); | ||||
|   assert( rc==SQLITE_ERROR ); | ||||
|   assert( p2==0 ); | ||||
|   assert( 0==strcmp(sqlite3_errmsg(db), "unknown tokenizer: nosuchtokenizer") ); | ||||
| 
 | ||||
|   /* Test the storage function */ | ||||
|   rc = registerTokenizer(db, "nosuchtokenizer", p1); | ||||
|   assert( rc==SQLITE_OK ); | ||||
|   rc = queryFts2Tokenizer(db, "nosuchtokenizer", &p2); | ||||
|   assert( rc==SQLITE_OK ); | ||||
|   assert( p2==p1 ); | ||||
| 
 | ||||
|   sqlite3_result_text(context, "ok", -1, SQLITE_STATIC); | ||||
| } | ||||
| 
 | ||||
| #endif | ||||
| 
 | ||||
| /*
 | ||||
| ** Set up SQL objects in database db used to access the contents of | ||||
| ** the hash table pointed to by argument pHash. The hash table must | ||||
| ** been initialized to use string keys, and to take a private copy  | ||||
| ** of the key when a value is inserted. i.e. by a call similar to: | ||||
| ** | ||||
| **    sqlite3Fts2HashInit(pHash, FTS2_HASH_STRING, 1); | ||||
| ** | ||||
| ** This function adds a scalar function (see header comment above | ||||
| ** scalarFunc() in this file for details) and, if ENABLE_TABLE is | ||||
| ** defined at compilation time, a temporary virtual table (see header  | ||||
| ** comment above struct HashTableVtab) to the database schema. Both  | ||||
| ** provide read/write access to the contents of *pHash. | ||||
| ** | ||||
| ** The third argument to this function, zName, is used as the name | ||||
| ** of both the scalar and, if created, the virtual table. | ||||
| */ | ||||
| int sqlite3Fts2InitHashTable( | ||||
|   sqlite3 *db,  | ||||
|   fts2Hash *pHash,  | ||||
|   const char *zName | ||||
| ){ | ||||
|   int rc = SQLITE_OK; | ||||
|   void *p = (void *)pHash; | ||||
|   const int any = SQLITE_ANY; | ||||
|   char *zTest = 0; | ||||
|   char *zTest2 = 0; | ||||
| 
 | ||||
| #ifdef SQLITE_TEST | ||||
|   void *pdb = (void *)db; | ||||
|   zTest = sqlite3_mprintf("%s_test", zName); | ||||
|   zTest2 = sqlite3_mprintf("%s_internal_test", zName); | ||||
|   if( !zTest || !zTest2 ){ | ||||
|     rc = SQLITE_NOMEM; | ||||
|   } | ||||
| #endif | ||||
| 
 | ||||
|   if( rc!=SQLITE_OK | ||||
|    || (rc = sqlite3_create_function(db, zName, 1, any, p, scalarFunc, 0, 0)) | ||||
|    || (rc = sqlite3_create_function(db, zName, 2, any, p, scalarFunc, 0, 0)) | ||||
| #ifdef SQLITE_TEST | ||||
|    || (rc = sqlite3_create_function(db, zTest, 2, any, p, testFunc, 0, 0)) | ||||
|    || (rc = sqlite3_create_function(db, zTest, 3, any, p, testFunc, 0, 0)) | ||||
|    || (rc = sqlite3_create_function(db, zTest2, 0, any, pdb, intTestFunc, 0, 0)) | ||||
| #endif | ||||
|   ); | ||||
| 
 | ||||
|   sqlite3_free(zTest); | ||||
|   sqlite3_free(zTest2); | ||||
|   return rc; | ||||
| } | ||||
| 
 | ||||
| #endif /* !defined(SQLITE_CORE) || defined(SQLITE_ENABLE_FTS2) */ | ||||
							
								
								
									
										147
									
								
								third_party/sqlite3/fts2_tokenizer.h
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										147
									
								
								third_party/sqlite3/fts2_tokenizer.h
									
										
									
									
										vendored
									
									
								
							|  | @ -1,147 +0,0 @@ | |||
| /*
 | ||||
| ** 2006 July 10 | ||||
| ** | ||||
| ** The author disclaims copyright to this source code. | ||||
| ** | ||||
| ************************************************************************* | ||||
| ** Defines the interface to tokenizers used by fulltext-search.  There | ||||
| ** are three basic components: | ||||
| ** | ||||
| ** sqlite3_tokenizer_module is a singleton defining the tokenizer | ||||
| ** interface functions.  This is essentially the class structure for | ||||
| ** tokenizers. | ||||
| ** | ||||
| ** sqlite3_tokenizer is used to define a particular tokenizer, perhaps | ||||
| ** including customization information defined at creation time. | ||||
| ** | ||||
| ** sqlite3_tokenizer_cursor is generated by a tokenizer to generate | ||||
| ** tokens from a particular input. | ||||
| */ | ||||
| #ifndef _FTS2_TOKENIZER_H_ | ||||
| #define _FTS2_TOKENIZER_H_ | ||||
| 
 | ||||
| /* clang-format off */ | ||||
| 
 | ||||
| /* TODO(shess) Only used for SQLITE_OK and SQLITE_DONE at this time.
 | ||||
| ** If tokenizers are to be allowed to call sqlite3_*() functions, then | ||||
| ** we will need a way to register the API consistently. | ||||
| */ | ||||
| #include "sqlite3.h" | ||||
| 
 | ||||
| /*
 | ||||
| ** Structures used by the tokenizer interface. When a new tokenizer | ||||
| ** implementation is registered, the caller provides a pointer to | ||||
| ** an sqlite3_tokenizer_module containing pointers to the callback | ||||
| ** functions that make up an implementation. | ||||
| ** | ||||
| ** When an fts2 table is created, it passes any arguments passed to | ||||
| ** the tokenizer clause of the CREATE VIRTUAL TABLE statement to the | ||||
| ** sqlite3_tokenizer_module.xCreate() function of the requested tokenizer | ||||
| ** implementation. The xCreate() function in turn returns an  | ||||
| ** sqlite3_tokenizer structure representing the specific tokenizer to | ||||
| ** be used for the fts2 table (customized by the tokenizer clause arguments). | ||||
| ** | ||||
| ** To tokenize an input buffer, the sqlite3_tokenizer_module.xOpen() | ||||
| ** method is called. It returns an sqlite3_tokenizer_cursor object | ||||
| ** that may be used to tokenize a specific input buffer based on | ||||
| ** the tokenization rules supplied by a specific sqlite3_tokenizer | ||||
| ** object. | ||||
| */ | ||||
| typedef struct sqlite3_tokenizer_module sqlite3_tokenizer_module; | ||||
| typedef struct sqlite3_tokenizer sqlite3_tokenizer; | ||||
| typedef struct sqlite3_tokenizer_cursor sqlite3_tokenizer_cursor; | ||||
| 
 | ||||
| struct sqlite3_tokenizer_module { | ||||
| 
 | ||||
|   /*
 | ||||
|   ** Structure version. Should always be set to 0. | ||||
|   */ | ||||
|   int iVersion; | ||||
| 
 | ||||
|   /*
 | ||||
|   ** Create a new tokenizer. The values in the argv[] array are the | ||||
|   ** arguments passed to the "tokenizer" clause of the CREATE VIRTUAL | ||||
|   ** TABLE statement that created the fts2 table. For example, if | ||||
|   ** the following SQL is executed: | ||||
|   ** | ||||
|   **   CREATE .. USING fts2( ... , tokenizer <tokenizer-name> arg1 arg2) | ||||
|   ** | ||||
|   ** then argc is set to 2, and the argv[] array contains pointers | ||||
|   ** to the strings "arg1" and "arg2". | ||||
|   ** | ||||
|   ** This method should return either SQLITE_OK (0), or an SQLite error  | ||||
|   ** code. If SQLITE_OK is returned, then *ppTokenizer should be set | ||||
|   ** to point at the newly created tokenizer structure. The generic | ||||
|   ** sqlite3_tokenizer.pModule variable should not be initialized by | ||||
|   ** this callback. The caller will do so. | ||||
|   */ | ||||
|   int (*xCreate)( | ||||
|     int argc,                           /* Size of argv array */ | ||||
|     const char *const*argv,             /* Tokenizer argument strings */ | ||||
|     sqlite3_tokenizer **ppTokenizer     /* OUT: Created tokenizer */ | ||||
|   ); | ||||
| 
 | ||||
|   /*
 | ||||
|   ** Destroy an existing tokenizer. The fts2 module calls this method | ||||
|   ** exactly once for each successful call to xCreate(). | ||||
|   */ | ||||
|   int (*xDestroy)(sqlite3_tokenizer *pTokenizer); | ||||
| 
 | ||||
|   /*
 | ||||
|   ** Create a tokenizer cursor to tokenize an input buffer. The caller | ||||
|   ** is responsible for ensuring that the input buffer remains valid | ||||
|   ** until the cursor is closed (using the xClose() method).  | ||||
|   */ | ||||
|   int (*xOpen)( | ||||
|     sqlite3_tokenizer *pTokenizer,       /* Tokenizer object */ | ||||
|     const char *pInput, int nBytes,      /* Input buffer */ | ||||
|     sqlite3_tokenizer_cursor **ppCursor  /* OUT: Created tokenizer cursor */ | ||||
|   ); | ||||
| 
 | ||||
|   /*
 | ||||
|   ** Destroy an existing tokenizer cursor. The fts2 module calls this  | ||||
|   ** method exactly once for each successful call to xOpen(). | ||||
|   */ | ||||
|   int (*xClose)(sqlite3_tokenizer_cursor *pCursor); | ||||
| 
 | ||||
|   /*
 | ||||
|   ** Retrieve the next token from the tokenizer cursor pCursor. This | ||||
|   ** method should either return SQLITE_OK and set the values of the | ||||
|   ** "OUT" variables identified below, or SQLITE_DONE to indicate that | ||||
|   ** the end of the buffer has been reached, or an SQLite error code. | ||||
|   ** | ||||
|   ** *ppToken should be set to point at a buffer containing the  | ||||
|   ** normalized version of the token (i.e. after any case-folding and/or | ||||
|   ** stemming has been performed). *pnBytes should be set to the length | ||||
|   ** of this buffer in bytes. The input text that generated the token is | ||||
|   ** identified by the byte offsets returned in *piStartOffset and | ||||
|   ** *piEndOffset. | ||||
|   ** | ||||
|   ** The buffer *ppToken is set to point at is managed by the tokenizer | ||||
|   ** implementation. It is only required to be valid until the next call | ||||
|   ** to xNext() or xClose().  | ||||
|   */ | ||||
|   /* TODO(shess) current implementation requires pInput to be
 | ||||
|   ** nul-terminated.  This should either be fixed, or pInput/nBytes | ||||
|   ** should be converted to zInput. | ||||
|   */ | ||||
|   int (*xNext)( | ||||
|     sqlite3_tokenizer_cursor *pCursor,   /* Tokenizer cursor */ | ||||
|     const char **ppToken, int *pnBytes,  /* OUT: Normalized text for token */ | ||||
|     int *piStartOffset,  /* OUT: Byte offset of token in input buffer */ | ||||
|     int *piEndOffset,    /* OUT: Byte offset of end of token in input buffer */ | ||||
|     int *piPosition      /* OUT: Number of tokens returned before this one */ | ||||
|   ); | ||||
| }; | ||||
| 
 | ||||
| struct sqlite3_tokenizer { | ||||
|   const sqlite3_tokenizer_module *pModule;  /* The module for this tokenizer */ | ||||
|   /* Tokenizer implementations will typically add additional fields */ | ||||
| }; | ||||
| 
 | ||||
| struct sqlite3_tokenizer_cursor { | ||||
|   sqlite3_tokenizer *pTokenizer;       /* Tokenizer for this cursor. */ | ||||
|   /* Tokenizer implementations will typically add additional fields */ | ||||
| }; | ||||
| 
 | ||||
| #endif /* _FTS2_TOKENIZER_H_ */ | ||||
							
								
								
									
										233
									
								
								third_party/sqlite3/fts2_tokenizer1.c
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										233
									
								
								third_party/sqlite3/fts2_tokenizer1.c
									
										
									
									
										vendored
									
									
								
							|  | @ -1,233 +0,0 @@ | |||
| /*
 | ||||
| ** 2006 Oct 10 | ||||
| ** | ||||
| ** The author disclaims copyright to this source code.  In place of | ||||
| ** a legal notice, here is a blessing: | ||||
| ** | ||||
| **    May you do good and not evil. | ||||
| **    May you find forgiveness for yourself and forgive others. | ||||
| **    May you share freely, never taking more than you give. | ||||
| ** | ||||
| ****************************************************************************** | ||||
| ** | ||||
| ** Implementation of the "simple" full-text-search tokenizer. | ||||
| */ | ||||
| /*
 | ||||
| ** The code in this file is only compiled if: | ||||
| ** | ||||
| **     * The FTS2 module is being built as an extension | ||||
| **       (in which case SQLITE_CORE is not defined), or | ||||
| ** | ||||
| **     * The FTS2 module is being built into the core of | ||||
| **       SQLite (in which case SQLITE_ENABLE_FTS2 is defined). | ||||
| */ | ||||
| #if !defined(SQLITE_CORE) || defined(SQLITE_ENABLE_FTS2) | ||||
| 
 | ||||
| /* clang-format off */ | ||||
| 
 | ||||
| #include <assert.h> | ||||
| #include <stdlib.h> | ||||
| #include <stdio.h> | ||||
| #include <string.h> | ||||
| 
 | ||||
| #include "sqlite3.h" | ||||
| #include "sqlite3ext.h" | ||||
| SQLITE_EXTENSION_INIT3 | ||||
| #include "fts2_tokenizer.h" | ||||
| 
 | ||||
| typedef struct simple_tokenizer { | ||||
|   sqlite3_tokenizer base; | ||||
|   char delim[128];             /* flag ASCII delimiters */ | ||||
| } simple_tokenizer; | ||||
| 
 | ||||
| typedef struct simple_tokenizer_cursor { | ||||
|   sqlite3_tokenizer_cursor base; | ||||
|   const char *pInput;          /* input we are tokenizing */ | ||||
|   int nBytes;                  /* size of the input */ | ||||
|   int iOffset;                 /* current position in pInput */ | ||||
|   int iToken;                  /* index of next token to be returned */ | ||||
|   char *pToken;                /* storage for current token */ | ||||
|   int nTokenAllocated;         /* space allocated to zToken buffer */ | ||||
| } simple_tokenizer_cursor; | ||||
| 
 | ||||
| 
 | ||||
| /* Forward declaration */ | ||||
| static const sqlite3_tokenizer_module simpleTokenizerModule; | ||||
| 
 | ||||
| static int simpleDelim(simple_tokenizer *t, unsigned char c){ | ||||
|   return c<0x80 && t->delim[c]; | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
| ** Create a new tokenizer instance. | ||||
| */ | ||||
| static int simpleCreate( | ||||
|   int argc, const char * const *argv, | ||||
|   sqlite3_tokenizer **ppTokenizer | ||||
| ){ | ||||
|   simple_tokenizer *t; | ||||
| 
 | ||||
|   t = (simple_tokenizer *) sqlite3_malloc(sizeof(*t)); | ||||
|   if( t==NULL ) return SQLITE_NOMEM; | ||||
|   memset(t, 0, sizeof(*t)); | ||||
| 
 | ||||
|   /* TODO(shess) Delimiters need to remain the same from run to run,
 | ||||
|   ** else we need to reindex.  One solution would be a meta-table to | ||||
|   ** track such information in the database, then we'd only want this | ||||
|   ** information on the initial create. | ||||
|   */ | ||||
|   if( argc>1 ){ | ||||
|     int i, n = strlen(argv[1]); | ||||
|     for(i=0; i<n; i++){ | ||||
|       unsigned char ch = argv[1][i]; | ||||
|       /* We explicitly don't support UTF-8 delimiters for now. */ | ||||
|       if( ch>=0x80 ){ | ||||
|         sqlite3_free(t); | ||||
|         return SQLITE_ERROR; | ||||
|       } | ||||
|       t->delim[ch] = 1; | ||||
|     } | ||||
|   } else { | ||||
|     /* Mark non-alphanumeric ASCII characters as delimiters */ | ||||
|     int i; | ||||
|     for(i=1; i<0x80; i++){ | ||||
|       t->delim[i] = !((i>='0' && i<='9') || (i>='A' && i<='Z') || | ||||
|                       (i>='a' && i<='z')); | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   *ppTokenizer = &t->base; | ||||
|   return SQLITE_OK; | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
| ** Destroy a tokenizer | ||||
| */ | ||||
| static int simpleDestroy(sqlite3_tokenizer *pTokenizer){ | ||||
|   sqlite3_free(pTokenizer); | ||||
|   return SQLITE_OK; | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
| ** Prepare to begin tokenizing a particular string.  The input | ||||
| ** string to be tokenized is pInput[0..nBytes-1].  A cursor | ||||
| ** used to incrementally tokenize this string is returned in  | ||||
| ** *ppCursor. | ||||
| */ | ||||
| static int simpleOpen( | ||||
|   sqlite3_tokenizer *pTokenizer,         /* The tokenizer */ | ||||
|   const char *pInput, int nBytes,        /* String to be tokenized */ | ||||
|   sqlite3_tokenizer_cursor **ppCursor    /* OUT: Tokenization cursor */ | ||||
| ){ | ||||
|   simple_tokenizer_cursor *c; | ||||
| 
 | ||||
|   c = (simple_tokenizer_cursor *) sqlite3_malloc(sizeof(*c)); | ||||
|   if( c==NULL ) return SQLITE_NOMEM; | ||||
| 
 | ||||
|   c->pInput = pInput; | ||||
|   if( pInput==0 ){ | ||||
|     c->nBytes = 0; | ||||
|   }else if( nBytes<0 ){ | ||||
|     c->nBytes = (int)strlen(pInput); | ||||
|   }else{ | ||||
|     c->nBytes = nBytes; | ||||
|   } | ||||
|   c->iOffset = 0;                 /* start tokenizing at the beginning */ | ||||
|   c->iToken = 0; | ||||
|   c->pToken = NULL;               /* no space allocated, yet. */ | ||||
|   c->nTokenAllocated = 0; | ||||
| 
 | ||||
|   *ppCursor = &c->base; | ||||
|   return SQLITE_OK; | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
| ** Close a tokenization cursor previously opened by a call to | ||||
| ** simpleOpen() above. | ||||
| */ | ||||
| static int simpleClose(sqlite3_tokenizer_cursor *pCursor){ | ||||
|   simple_tokenizer_cursor *c = (simple_tokenizer_cursor *) pCursor; | ||||
|   sqlite3_free(c->pToken); | ||||
|   sqlite3_free(c); | ||||
|   return SQLITE_OK; | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
| ** Extract the next token from a tokenization cursor.  The cursor must | ||||
| ** have been opened by a prior call to simpleOpen(). | ||||
| */ | ||||
| static int simpleNext( | ||||
|   sqlite3_tokenizer_cursor *pCursor,  /* Cursor returned by simpleOpen */ | ||||
|   const char **ppToken,               /* OUT: *ppToken is the token text */ | ||||
|   int *pnBytes,                       /* OUT: Number of bytes in token */ | ||||
|   int *piStartOffset,                 /* OUT: Starting offset of token */ | ||||
|   int *piEndOffset,                   /* OUT: Ending offset of token */ | ||||
|   int *piPosition                     /* OUT: Position integer of token */ | ||||
| ){ | ||||
|   simple_tokenizer_cursor *c = (simple_tokenizer_cursor *) pCursor; | ||||
|   simple_tokenizer *t = (simple_tokenizer *) pCursor->pTokenizer; | ||||
|   unsigned char *p = (unsigned char *)c->pInput; | ||||
| 
 | ||||
|   while( c->iOffset<c->nBytes ){ | ||||
|     int iStartOffset; | ||||
| 
 | ||||
|     /* Scan past delimiter characters */ | ||||
|     while( c->iOffset<c->nBytes && simpleDelim(t, p[c->iOffset]) ){ | ||||
|       c->iOffset++; | ||||
|     } | ||||
| 
 | ||||
|     /* Count non-delimiter characters. */ | ||||
|     iStartOffset = c->iOffset; | ||||
|     while( c->iOffset<c->nBytes && !simpleDelim(t, p[c->iOffset]) ){ | ||||
|       c->iOffset++; | ||||
|     } | ||||
| 
 | ||||
|     if( c->iOffset>iStartOffset ){ | ||||
|       int i, n = c->iOffset-iStartOffset; | ||||
|       if( n>c->nTokenAllocated ){ | ||||
|         c->nTokenAllocated = n+20; | ||||
|         c->pToken = sqlite3_realloc(c->pToken, c->nTokenAllocated); | ||||
|         if( c->pToken==NULL ) return SQLITE_NOMEM; | ||||
|       } | ||||
|       for(i=0; i<n; i++){ | ||||
|         /* TODO(shess) This needs expansion to handle UTF-8
 | ||||
|         ** case-insensitivity. | ||||
|         */ | ||||
|         unsigned char ch = p[iStartOffset+i]; | ||||
|         c->pToken[i] = (ch>='A' && ch<='Z') ? (ch - 'A' + 'a') : ch; | ||||
|       } | ||||
|       *ppToken = c->pToken; | ||||
|       *pnBytes = n; | ||||
|       *piStartOffset = iStartOffset; | ||||
|       *piEndOffset = c->iOffset; | ||||
|       *piPosition = c->iToken++; | ||||
| 
 | ||||
|       return SQLITE_OK; | ||||
|     } | ||||
|   } | ||||
|   return SQLITE_DONE; | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
| ** The set of routines that implement the simple tokenizer | ||||
| */ | ||||
| static const sqlite3_tokenizer_module simpleTokenizerModule = { | ||||
|   0, | ||||
|   simpleCreate, | ||||
|   simpleDestroy, | ||||
|   simpleOpen, | ||||
|   simpleClose, | ||||
|   simpleNext, | ||||
| }; | ||||
| 
 | ||||
| /*
 | ||||
| ** Allocate a new simple tokenizer.  Return a pointer to the new | ||||
| ** tokenizer in *ppModule | ||||
| */ | ||||
| void sqlite3Fts2SimpleTokenizerModule( | ||||
|   sqlite3_tokenizer_module const**ppModule | ||||
| ){ | ||||
|   *ppModule = &simpleTokenizerModule; | ||||
| } | ||||
| 
 | ||||
| #endif /* !defined(SQLITE_CORE) || defined(SQLITE_ENABLE_FTS2) */ | ||||
							
								
								
									
										4
									
								
								third_party/sqlite3/fts3.c
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								third_party/sqlite3/fts3.c
									
										
									
									
										vendored
									
									
								
							|  | @ -289,7 +289,7 @@ | |||
| */ | ||||
| /* clang-format off */ | ||||
| 
 | ||||
| #include "third_party/sqlite3/fts3Int.h" | ||||
| #include "third_party/sqlite3/fts3Int.inc" | ||||
| #if !defined(SQLITE_CORE) || defined(SQLITE_ENABLE_FTS3) | ||||
| 
 | ||||
| #if defined(SQLITE_ENABLE_FTS3) && !defined(SQLITE_CORE) | ||||
|  | @ -300,7 +300,7 @@ | |||
| #include "libc/mem/mem.h" | ||||
| #include "libc/stdio/stdio.h" | ||||
| #include "libc/str/str.h" | ||||
| #include "third_party/sqlite3/fts3.h" | ||||
| #include "third_party/sqlite3/fts3.inc" | ||||
| #ifndef SQLITE_CORE | ||||
| #include "third_party/sqlite3/sqlite3ext.h" | ||||
| SQLITE_EXTENSION_INIT1 | ||||
|  |  | |||
|  | @ -42,8 +42,8 @@ | |||
| SQLITE_EXTENSION_INIT3 | ||||
| #endif
 | ||||
| 
 | ||||
| #include "third_party/sqlite3/fts3_hash.h"
 | ||||
| #include "third_party/sqlite3/fts3_tokenizer.h"
 | ||||
| #include "third_party/sqlite3/fts3_hash.inc"
 | ||||
| #include "third_party/sqlite3/fts3_tokenizer.inc"
 | ||||
| #include "third_party/sqlite3/sqlite3.h"
 | ||||
| 
 | ||||
| /* | ||||
							
								
								
									
										2
									
								
								third_party/sqlite3/fts3_aux.c
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								third_party/sqlite3/fts3_aux.c
									
										
									
									
										vendored
									
									
								
							|  | @ -12,7 +12,7 @@ | |||
| ** | ||||
| */ | ||||
| /* clang-format off */ | ||||
| #include "third_party/sqlite3/fts3Int.h" | ||||
| #include "third_party/sqlite3/fts3Int.inc" | ||||
| #if !defined(SQLITE_CORE) || defined(SQLITE_ENABLE_FTS3) | ||||
| 
 | ||||
| #include "libc/assert.h" | ||||
|  |  | |||
							
								
								
									
										2
									
								
								third_party/sqlite3/fts3_expr.c
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								third_party/sqlite3/fts3_expr.c
									
										
									
									
										vendored
									
									
								
							|  | @ -15,7 +15,7 @@ | |||
| ** syntax is relatively simple, the whole tokenizer/parser system is | ||||
| ** hand-coded. | ||||
| */ | ||||
| #include "third_party/sqlite3/fts3Int.h" | ||||
| #include "third_party/sqlite3/fts3Int.inc" | ||||
| #if !defined(SQLITE_CORE) || defined(SQLITE_ENABLE_FTS3) | ||||
| /* clang-format off */ | ||||
| 
 | ||||
|  |  | |||
							
								
								
									
										4
									
								
								third_party/sqlite3/fts3_hash.c
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								third_party/sqlite3/fts3_hash.c
									
										
									
									
										vendored
									
									
								
							|  | @ -24,13 +24,13 @@ | |||
| **     * The FTS3 module is being built into the core of | ||||
| **       SQLite (in which case SQLITE_ENABLE_FTS3 is defined). | ||||
| */ | ||||
| #include "third_party/sqlite3/fts3Int.h" | ||||
| #include "third_party/sqlite3/fts3Int.inc" | ||||
| #if !defined(SQLITE_CORE) || defined(SQLITE_ENABLE_FTS3) | ||||
| 
 | ||||
| #include "libc/assert.h" | ||||
| #include "libc/mem/mem.h" | ||||
| #include "libc/str/str.h" | ||||
| #include "third_party/sqlite3/fts3_hash.h" | ||||
| #include "third_party/sqlite3/fts3_hash.inc" | ||||
| 
 | ||||
| /*
 | ||||
| ** Malloc and Free functions | ||||
|  |  | |||
							
								
								
									
										4
									
								
								third_party/sqlite3/fts3_icu.c
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								third_party/sqlite3/fts3_icu.c
									
										
									
									
										vendored
									
									
								
							|  | @ -11,7 +11,7 @@ | |||
| ************************************************************************* | ||||
| ** This file implements a tokenizer for fts3 based on the ICU library. | ||||
| */ | ||||
| #include "third_party/sqlite3/fts3Int.h" | ||||
| #include "third_party/sqlite3/fts3Int.inc" | ||||
| #if !defined(SQLITE_CORE) || defined(SQLITE_ENABLE_FTS3) | ||||
| #ifdef SQLITE_ENABLE_ICU | ||||
| /* clang-format off */ | ||||
|  | @ -19,7 +19,7 @@ | |||
| #include "libc/assert.h" | ||||
| #include "libc/str/str.h" | ||||
| #include "libc/unicode/unicode.h" | ||||
| #include "third_party/sqlite3/fts3_tokenizer.h" | ||||
| #include "third_party/sqlite3/fts3_tokenizer.inc" | ||||
| 
 | ||||
| typedef struct IcuTokenizer IcuTokenizer; | ||||
| typedef struct IcuCursor IcuCursor; | ||||
|  |  | |||
							
								
								
									
										4
									
								
								third_party/sqlite3/fts3_porter.c
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								third_party/sqlite3/fts3_porter.c
									
										
									
									
										vendored
									
									
								
							|  | @ -23,14 +23,14 @@ | |||
| **     * The FTS3 module is being built into the core of | ||||
| **       SQLite (in which case SQLITE_ENABLE_FTS3 is defined). | ||||
| */ | ||||
| #include "third_party/sqlite3/fts3Int.h" | ||||
| #include "third_party/sqlite3/fts3Int.inc" | ||||
| #if !defined(SQLITE_CORE) || defined(SQLITE_ENABLE_FTS3) | ||||
| 
 | ||||
| #include "libc/assert.h" | ||||
| #include "libc/mem/mem.h" | ||||
| #include "libc/stdio/stdio.h" | ||||
| #include "libc/str/str.h" | ||||
| #include "third_party/sqlite3/fts3_tokenizer.h" | ||||
| #include "third_party/sqlite3/fts3_tokenizer.inc" | ||||
| 
 | ||||
| /*
 | ||||
| ** Class derived from sqlite3_tokenizer | ||||
|  |  | |||
							
								
								
									
										2
									
								
								third_party/sqlite3/fts3_snippet.c
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								third_party/sqlite3/fts3_snippet.c
									
										
									
									
										vendored
									
									
								
							|  | @ -12,7 +12,7 @@ | |||
| */ | ||||
| /* clang-format off */ | ||||
| 
 | ||||
| #include "third_party/sqlite3/fts3Int.h" | ||||
| #include "third_party/sqlite3/fts3Int.inc" | ||||
| #if !defined(SQLITE_CORE) || defined(SQLITE_ENABLE_FTS3) | ||||
| 
 | ||||
| #include "libc/assert.h" | ||||
|  |  | |||
							
								
								
									
										2
									
								
								third_party/sqlite3/fts3_tokenize_vtab.c
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								third_party/sqlite3/fts3_tokenize_vtab.c
									
										
									
									
										vendored
									
									
								
							|  | @ -38,7 +38,7 @@ | |||
| **   pos:     Token offset of token within input. | ||||
| ** | ||||
| */ | ||||
| #include "third_party/sqlite3/fts3Int.h" | ||||
| #include "third_party/sqlite3/fts3Int.inc" | ||||
| #if !defined(SQLITE_CORE) || defined(SQLITE_ENABLE_FTS3) | ||||
| /* clang-format off */ | ||||
| 
 | ||||
|  |  | |||
							
								
								
									
										8
									
								
								third_party/sqlite3/fts3_tokenizer.c
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										8
									
								
								third_party/sqlite3/fts3_tokenizer.c
									
										
									
									
										vendored
									
									
								
							|  | @ -24,7 +24,7 @@ | |||
| **     * The FTS3 module is being built into the core of | ||||
| **       SQLite (in which case SQLITE_ENABLE_FTS3 is defined). | ||||
| */ | ||||
| #include "third_party/sqlite3/fts3Int.h" | ||||
| #include "third_party/sqlite3/fts3Int.inc" | ||||
| #if !defined(SQLITE_CORE) || defined(SQLITE_ENABLE_FTS3) | ||||
| 
 | ||||
| #include "libc/assert.h" | ||||
|  | @ -226,12 +226,6 @@ int sqlite3Fts3InitTokenizer( | |||
| 
 | ||||
| 
 | ||||
| #ifdef SQLITE_TEST | ||||
| 
 | ||||
| #if defined(INCLUDE_SQLITE_TCL_H) | ||||
| #include "third_party/sqlite3/sqlite_tcl.h" | ||||
| #else | ||||
| #include "third_party/sqlite3/tcl.h" | ||||
| #endif | ||||
| #include "libc/str/str.h" | ||||
| 
 | ||||
| /*
 | ||||
|  |  | |||
							
								
								
									
										4
									
								
								third_party/sqlite3/fts3_tokenizer1.c
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								third_party/sqlite3/fts3_tokenizer1.c
									
										
									
									
										vendored
									
									
								
							|  | @ -23,14 +23,14 @@ | |||
| **     * The FTS3 module is being built into the core of | ||||
| **       SQLite (in which case SQLITE_ENABLE_FTS3 is defined). | ||||
| */ | ||||
| #include "third_party/sqlite3/fts3Int.h" | ||||
| #include "third_party/sqlite3/fts3Int.inc" | ||||
| #if !defined(SQLITE_CORE) || defined(SQLITE_ENABLE_FTS3) | ||||
| 
 | ||||
| #include "libc/assert.h" | ||||
| #include "libc/mem/mem.h" | ||||
| #include "libc/stdio/stdio.h" | ||||
| #include "libc/str/str.h" | ||||
| #include "third_party/sqlite3/fts3_tokenizer.h" | ||||
| #include "third_party/sqlite3/fts3_tokenizer.inc" | ||||
| 
 | ||||
| typedef struct simple_tokenizer { | ||||
|   sqlite3_tokenizer base; | ||||
|  |  | |||
							
								
								
									
										4
									
								
								third_party/sqlite3/fts3_unicode.c
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								third_party/sqlite3/fts3_unicode.c
									
										
									
									
										vendored
									
									
								
							|  | @ -16,14 +16,14 @@ | |||
| 
 | ||||
| #ifndef SQLITE_DISABLE_FTS3_UNICODE | ||||
| 
 | ||||
| #include "third_party/sqlite3/fts3Int.h" | ||||
| #include "third_party/sqlite3/fts3Int.inc" | ||||
| #if !defined(SQLITE_CORE) || defined(SQLITE_ENABLE_FTS3) | ||||
| 
 | ||||
| #include "libc/assert.h" | ||||
| #include "libc/mem/mem.h" | ||||
| #include "libc/stdio/stdio.h" | ||||
| #include "libc/str/str.h" | ||||
| #include "third_party/sqlite3/fts3_tokenizer.h" | ||||
| #include "third_party/sqlite3/fts3_tokenizer.inc" | ||||
| 
 | ||||
| /*
 | ||||
| ** The following two macros - READ_UTF8 and WRITE_UTF8 - have been copied | ||||
|  |  | |||
							
								
								
									
										2
									
								
								third_party/sqlite3/fts3_write.c
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								third_party/sqlite3/fts3_write.c
									
										
									
									
										vendored
									
									
								
							|  | @ -18,7 +18,7 @@ | |||
| */ | ||||
| /* clang-format off */ | ||||
| 
 | ||||
| #include "third_party/sqlite3/fts3Int.h" | ||||
| #include "third_party/sqlite3/fts3Int.inc" | ||||
| #if !defined(SQLITE_CORE) || defined(SQLITE_ENABLE_FTS3) | ||||
| 
 | ||||
| #include "libc/alg/alg.h" | ||||
|  |  | |||
							
								
								
									
										4
									
								
								third_party/sqlite3/func.c
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								third_party/sqlite3/func.c
									
										
									
									
										vendored
									
									
								
							|  | @ -15,11 +15,11 @@ | |||
| */ | ||||
| #include "libc/assert.h" | ||||
| #include "libc/mem/mem.h" | ||||
| #include "third_party/sqlite3/sqliteInt.h" | ||||
| #include "third_party/sqlite3/sqliteInt.inc" | ||||
| #ifndef SQLITE_OMIT_FLOATING_POINT | ||||
| #include "libc/math.h" | ||||
| #endif | ||||
| #include "third_party/sqlite3/vdbeInt.h" | ||||
| #include "third_party/sqlite3/vdbeInt.inc" | ||||
| /* clang-format off */ | ||||
| 
 | ||||
| /*
 | ||||
|  |  | |||
|  | @ -16,11 +16,7 @@ | |||
| ** This file is #include-ed onto the end of "rtree.c" so that it has
 | ||||
| ** access to all of the R-Tree internals. | ||||
| */ | ||||
| #include "libc/fmt/conv.h"
 | ||||
| #include "libc/mem/mem.h"
 | ||||
| #include "third_party/gdtoa/gdtoa.h"
 | ||||
| #include "third_party/sqlite3/rtree.h"
 | ||||
| #include "third_party/sqlite3/sqlite3.h"
 | ||||
| /* clang-format off */ | ||||
| 
 | ||||
| /* Enable -DGEOPOLY_ENABLE_DEBUG for debugging facilities */ | ||||
							
								
								
									
										4
									
								
								third_party/sqlite3/global.c
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								third_party/sqlite3/global.c
									
										
									
									
										vendored
									
									
								
							|  | @ -12,7 +12,7 @@ | |||
| ** | ||||
| ** This file contains definitions of global variables and constants. | ||||
| */ | ||||
| #include "third_party/sqlite3/sqliteInt.h" | ||||
| #include "third_party/sqlite3/sqliteInt.inc" | ||||
| /* clang-format off */ | ||||
| 
 | ||||
| /* An array to map all upper-case characters into their corresponding
 | ||||
|  | @ -307,7 +307,7 @@ int sqlite3PendingByte = 0x40000000; | |||
| u32 sqlite3SelectTrace = 0; | ||||
| u32 sqlite3WhereTrace = 0; | ||||
| 
 | ||||
| #include "third_party/sqlite3/opcodes.h" | ||||
| #include "third_party/sqlite3/opcodes.inc" | ||||
| /*
 | ||||
| ** Properties of opcodes.  The OPFLG_INITIALIZER macro is | ||||
| ** created by mkopcodeh.awk during compilation.  Data is obtained | ||||
|  |  | |||
							
								
								
									
										2
									
								
								third_party/sqlite3/hash.c
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								third_party/sqlite3/hash.c
									
										
									
									
										vendored
									
									
								
							|  | @ -13,7 +13,7 @@ | |||
| ** used in SQLite. | ||||
| */ | ||||
| #include "libc/assert.h" | ||||
| #include "third_party/sqlite3/sqliteInt.h" | ||||
| #include "third_party/sqlite3/sqliteInt.inc" | ||||
| /* clang-format off */ | ||||
| 
 | ||||
| /* Turn bulk memory into a hash table object by initializing the
 | ||||
|  |  | |||
							
								
								
									
										2
									
								
								third_party/sqlite3/insert.c
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								third_party/sqlite3/insert.c
									
										
									
									
										vendored
									
									
								
							|  | @ -12,7 +12,7 @@ | |||
| ** This file contains C code routines that are called by the parser | ||||
| ** to handle INSERT statements in SQLite. | ||||
| */ | ||||
| #include "third_party/sqlite3/sqliteInt.h" | ||||
| #include "third_party/sqlite3/sqliteInt.inc" | ||||
| /* clang-format off */ | ||||
| 
 | ||||
| /*
 | ||||
|  |  | |||
							
								
								
									
										62
									
								
								third_party/sqlite3/inttypes.inc
									
										
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										62
									
								
								third_party/sqlite3/inttypes.inc
									
										
									
									
										vendored
									
									
										Normal file
									
								
							|  | @ -0,0 +1,62 @@ | |||
| #ifndef COSMOPOLITAN_THIRD_PARTY_SQLITE3_INTTYPES_H_
 | ||||
| #define COSMOPOLITAN_THIRD_PARTY_SQLITE3_INTTYPES_H_
 | ||||
| #include "third_party/sqlite3/sqlite3.h"
 | ||||
| #if !(__ASSEMBLER__ + __LINKER__ + 0)
 | ||||
| COSMOPOLITAN_C_START_ | ||||
| 
 | ||||
| /* | ||||
| ** Integers of known sizes.  These typedefs might change for architectures | ||||
| ** where the sizes very.  Preprocessor macros are available so that the | ||||
| ** types can be conveniently redefined at compile-type.  Like this: | ||||
| ** | ||||
| **         cc '-DUINTPTR_TYPE=long long int' ... | ||||
| */ | ||||
| #ifndef UINT32_TYPE
 | ||||
| #ifdef HAVE_UINT32_T
 | ||||
| #define UINT32_TYPE uint32_t
 | ||||
| #else
 | ||||
| #define UINT32_TYPE unsigned int
 | ||||
| #endif
 | ||||
| #endif
 | ||||
| #ifndef UINT16_TYPE
 | ||||
| #ifdef HAVE_UINT16_T
 | ||||
| #define UINT16_TYPE uint16_t
 | ||||
| #else
 | ||||
| #define UINT16_TYPE unsigned short int
 | ||||
| #endif
 | ||||
| #endif
 | ||||
| #ifndef INT16_TYPE
 | ||||
| #ifdef HAVE_INT16_T
 | ||||
| #define INT16_TYPE int16_t
 | ||||
| #else
 | ||||
| #define INT16_TYPE short int
 | ||||
| #endif
 | ||||
| #endif
 | ||||
| #ifndef UINT8_TYPE
 | ||||
| #ifdef HAVE_UINT8_T
 | ||||
| #define UINT8_TYPE uint8_t
 | ||||
| #else
 | ||||
| #define UINT8_TYPE unsigned char
 | ||||
| #endif
 | ||||
| #endif
 | ||||
| #ifndef INT8_TYPE
 | ||||
| #ifdef HAVE_INT8_T
 | ||||
| #define INT8_TYPE int8_t
 | ||||
| #else
 | ||||
| #define INT8_TYPE signed char
 | ||||
| #endif
 | ||||
| #endif
 | ||||
| #ifndef LONGDOUBLE_TYPE
 | ||||
| #define LONGDOUBLE_TYPE long double
 | ||||
| #endif
 | ||||
| typedef sqlite_int64 i64;  /* 8-byte signed integer */ | ||||
| typedef sqlite_uint64 u64; /* 8-byte unsigned integer */ | ||||
| typedef UINT32_TYPE u32;   /* 4-byte unsigned integer */ | ||||
| typedef UINT16_TYPE u16;   /* 2-byte unsigned integer */ | ||||
| typedef INT16_TYPE i16;    /* 2-byte signed integer */ | ||||
| typedef UINT8_TYPE u8;     /* 1-byte unsigned integer */ | ||||
| typedef INT8_TYPE i8;      /* 1-byte signed integer */ | ||||
| 
 | ||||
| COSMOPOLITAN_C_END_ | ||||
| #endif /* !(__ASSEMBLER__ + __LINKER__ + 0) */
 | ||||
| #endif /* COSMOPOLITAN_THIRD_PARTY_SQLITE3_INTTYPES_H_ */
 | ||||
							
								
								
									
										2
									
								
								third_party/sqlite3/legacy.c
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								third_party/sqlite3/legacy.c
									
										
									
									
										vendored
									
									
								
							|  | @ -16,7 +16,7 @@ | |||
| */ | ||||
| /* clang-format off */ | ||||
| 
 | ||||
| #include "third_party/sqlite3/sqliteInt.h" | ||||
| #include "third_party/sqlite3/sqliteInt.inc" | ||||
| 
 | ||||
| /*
 | ||||
| ** Execute SQL code.  Return one of the SQLITE_ success/failure | ||||
|  |  | |||
							
								
								
									
										2
									
								
								third_party/sqlite3/loadext.c
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								third_party/sqlite3/loadext.c
									
										
									
									
										vendored
									
									
								
							|  | @ -18,7 +18,7 @@ | |||
|   #define SQLITE_CORE 1  /* Disable the API redefinition in sqlite3ext.h */ | ||||
| #endif | ||||
| #include "third_party/sqlite3/sqlite3ext.h" | ||||
| #include "third_party/sqlite3/sqliteInt.h" | ||||
| #include "third_party/sqlite3/sqliteInt.inc" | ||||
| 
 | ||||
| #ifndef SQLITE_OMIT_LOAD_EXTENSION | ||||
| /*
 | ||||
|  |  | |||
							
								
								
									
										8
									
								
								third_party/sqlite3/main.c
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										8
									
								
								third_party/sqlite3/main.c
									
										
									
									
										vendored
									
									
								
							|  | @ -14,17 +14,17 @@ | |||
| ** other files are for internal use by SQLite and should not be | ||||
| ** accessed by users of the library. | ||||
| */ | ||||
| #include "third_party/sqlite3/sqliteInt.h" | ||||
| #include "third_party/sqlite3/sqliteInt.inc" | ||||
| /* clang-format off */ | ||||
| 
 | ||||
| #ifdef SQLITE_ENABLE_FTS3 | ||||
| #include "third_party/sqlite3/fts3.h" | ||||
| #include "third_party/sqlite3/fts3.inc" | ||||
| #endif | ||||
| #ifdef SQLITE_ENABLE_RTREE | ||||
| #include "third_party/sqlite3/rtree.h" | ||||
| #include "third_party/sqlite3/rtree.inc" | ||||
| #endif | ||||
| #if defined(SQLITE_ENABLE_ICU) || defined(SQLITE_ENABLE_ICU_COLLATIONS) | ||||
| #include "third_party/sqlite3/sqliteicu.h" | ||||
| #include "third_party/sqlite3/sqliteicu.inc" | ||||
| #endif | ||||
| 
 | ||||
| /*
 | ||||
|  |  | |||
							
								
								
									
										2
									
								
								third_party/sqlite3/malloc.c
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								third_party/sqlite3/malloc.c
									
										
									
									
										vendored
									
									
								
							|  | @ -14,7 +14,7 @@ | |||
| */ | ||||
| /* clang-format off */ | ||||
| 
 | ||||
| #include "third_party/sqlite3/sqliteInt.h" | ||||
| #include "third_party/sqlite3/sqliteInt.inc" | ||||
| 
 | ||||
| /*
 | ||||
| ** Attempt to release up to n bytes of non-essential memory currently | ||||
|  |  | |||
							
								
								
									
										2
									
								
								third_party/sqlite3/mem0.c
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								third_party/sqlite3/mem0.c
									
										
									
									
										vendored
									
									
								
							|  | @ -16,7 +16,7 @@ | |||
| ** are merely placeholders.  Real drivers must be substituted using | ||||
| ** sqlite3_config() before SQLite will operate. | ||||
| */ | ||||
| #include "third_party/sqlite3/sqliteInt.h" | ||||
| #include "third_party/sqlite3/sqliteInt.inc" | ||||
| /* clang-format off */ | ||||
| 
 | ||||
| /*
 | ||||
|  |  | |||
							
								
								
									
										2
									
								
								third_party/sqlite3/mem1.c
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								third_party/sqlite3/mem1.c
									
										
									
									
										vendored
									
									
								
							|  | @ -41,7 +41,7 @@ | |||
| **                                be necessary when compiling for Delphi, | ||||
| **                                for example. | ||||
| */ | ||||
| #include "third_party/sqlite3/sqliteInt.h" | ||||
| #include "third_party/sqlite3/sqliteInt.inc" | ||||
| /* clang-format off */ | ||||
| 
 | ||||
| /*
 | ||||
|  |  | |||
							
								
								
									
										2
									
								
								third_party/sqlite3/mem2.c
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								third_party/sqlite3/mem2.c
									
										
									
									
										vendored
									
									
								
							|  | @ -19,7 +19,7 @@ | |||
| ** This file contains implementations of the low-level memory allocation | ||||
| ** routines specified in the sqlite3_mem_methods object. | ||||
| */ | ||||
| #include "third_party/sqlite3/sqliteInt.h" | ||||
| #include "third_party/sqlite3/sqliteInt.inc" | ||||
| /* clang-format off */ | ||||
| 
 | ||||
| /*
 | ||||
|  |  | |||
							
								
								
									
										2
									
								
								third_party/sqlite3/mem3.c
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								third_party/sqlite3/mem3.c
									
										
									
									
										vendored
									
									
								
							|  | @ -23,7 +23,7 @@ | |||
| ** This version of the memory allocation subsystem is included | ||||
| ** in the build only if SQLITE_ENABLE_MEMSYS3 is defined. | ||||
| */ | ||||
| #include "third_party/sqlite3/sqliteInt.h" | ||||
| #include "third_party/sqlite3/sqliteInt.inc" | ||||
| /* clang-format off */ | ||||
| 
 | ||||
| /*
 | ||||
|  |  | |||
							
								
								
									
										2
									
								
								third_party/sqlite3/mem5.c
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								third_party/sqlite3/mem5.c
									
										
									
									
										vendored
									
									
								
							|  | @ -48,7 +48,7 @@ | |||
| ** The sqlite3_status() logic tracks the maximum values of n and M so | ||||
| ** that an application can, at any time, verify this constraint. | ||||
| */ | ||||
| #include "third_party/sqlite3/sqliteInt.h" | ||||
| #include "third_party/sqlite3/sqliteInt.inc" | ||||
| /* clang-format off */ | ||||
| 
 | ||||
| /*
 | ||||
|  |  | |||
							
								
								
									
										2
									
								
								third_party/sqlite3/memdb.c
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								third_party/sqlite3/memdb.c
									
										
									
									
										vendored
									
									
								
							|  | @ -16,7 +16,7 @@ | |||
| ** This file also implements interface sqlite3_serialize() and | ||||
| ** sqlite3_deserialize(). | ||||
| */ | ||||
| #include "third_party/sqlite3/sqliteInt.h" | ||||
| #include "third_party/sqlite3/sqliteInt.inc" | ||||
| #ifdef SQLITE_ENABLE_DESERIALIZE | ||||
| /* clang-format off */ | ||||
| 
 | ||||
|  |  | |||
							
								
								
									
										2
									
								
								third_party/sqlite3/memjournal.c
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								third_party/sqlite3/memjournal.c
									
										
									
									
										vendored
									
									
								
							|  | @ -23,7 +23,7 @@ | |||
| ** in the common case, they are usually small and no file I/O needs to | ||||
| ** occur. | ||||
| */ | ||||
| #include "third_party/sqlite3/sqliteInt.h" | ||||
| #include "third_party/sqlite3/sqliteInt.inc" | ||||
| /* clang-format off */ | ||||
| 
 | ||||
| /* Forward references to internal structures */ | ||||
|  |  | |||
							
								
								
									
										2
									
								
								third_party/sqlite3/mutex.c
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								third_party/sqlite3/mutex.c
									
										
									
									
										vendored
									
									
								
							|  | @ -13,7 +13,7 @@ | |||
| ** | ||||
| ** This file contains code that is common across all mutex implementations. | ||||
| */ | ||||
| #include "third_party/sqlite3/sqliteInt.h" | ||||
| #include "third_party/sqlite3/sqliteInt.inc" | ||||
| /* clang-format off */ | ||||
| 
 | ||||
| #if defined(SQLITE_DEBUG) && !defined(SQLITE_MUTEX_OMIT) | ||||
|  |  | |||
							
								
								
									
										2
									
								
								third_party/sqlite3/mutex_noop.c
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								third_party/sqlite3/mutex_noop.c
									
										
									
									
										vendored
									
									
								
							|  | @ -25,7 +25,7 @@ | |||
| ** that does error checking on mutexes to make sure they are being | ||||
| ** called correctly. | ||||
| */ | ||||
| #include "third_party/sqlite3/sqliteInt.h" | ||||
| #include "third_party/sqlite3/sqliteInt.inc" | ||||
| /* clang-format off */ | ||||
| 
 | ||||
| #ifndef SQLITE_MUTEX_OMIT | ||||
|  |  | |||
							
								
								
									
										2
									
								
								third_party/sqlite3/mutex_unix.c
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								third_party/sqlite3/mutex_unix.c
									
										
									
									
										vendored
									
									
								
							|  | @ -11,7 +11,7 @@ | |||
| ************************************************************************* | ||||
| ** This file contains the C functions that implement mutexes for pthreads | ||||
| */ | ||||
| #include "third_party/sqlite3/sqliteInt.h" | ||||
| #include "third_party/sqlite3/sqliteInt.inc" | ||||
| /* clang-format off */ | ||||
| 
 | ||||
| /*
 | ||||
|  |  | |||
							
								
								
									
										400
									
								
								third_party/sqlite3/mutex_w32.c
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										400
									
								
								third_party/sqlite3/mutex_w32.c
									
										
									
									
										vendored
									
									
								
							|  | @ -1,400 +0,0 @@ | |||
| /*
 | ||||
| ** 2007 August 14 | ||||
| ** | ||||
| ** The author disclaims copyright to this source code.  In place of | ||||
| ** a legal notice, here is a blessing: | ||||
| ** | ||||
| **    May you do good and not evil. | ||||
| **    May you find forgiveness for yourself and forgive others. | ||||
| **    May you share freely, never taking more than you give. | ||||
| ** | ||||
| ************************************************************************* | ||||
| ** This file contains the C functions that implement mutexes for Win32. | ||||
| */ | ||||
| #include "third_party/sqlite3/sqliteInt.h" | ||||
| /* clang-format off */ | ||||
| 
 | ||||
| #if SQLITE_OS_WIN | ||||
| /*
 | ||||
| ** Include code that is common to all os_*.c files | ||||
| */ | ||||
| #include "third_party/sqlite3/os_common.h" | ||||
| 
 | ||||
| /*
 | ||||
| ** Include the header file for the Windows VFS. | ||||
| */ | ||||
| #include "third_party/sqlite3/os_win.h" | ||||
| #endif | ||||
| 
 | ||||
| /*
 | ||||
| ** The code in this file is only used if we are compiling multithreaded | ||||
| ** on a Win32 system. | ||||
| */ | ||||
| #ifdef SQLITE_MUTEX_W32 | ||||
| 
 | ||||
| /*
 | ||||
| ** Each recursive mutex is an instance of the following structure. | ||||
| */ | ||||
| struct sqlite3_mutex { | ||||
|   CRITICAL_SECTION mutex;    /* Mutex controlling the lock */ | ||||
|   int id;                    /* Mutex type */ | ||||
| #ifdef SQLITE_DEBUG | ||||
|   volatile int nRef;         /* Number of enterances */ | ||||
|   volatile DWORD owner;      /* Thread holding this mutex */ | ||||
|   volatile LONG trace;       /* True to trace changes */ | ||||
| #endif | ||||
| }; | ||||
| 
 | ||||
| /*
 | ||||
| ** These are the initializer values used when declaring a "static" mutex | ||||
| ** on Win32.  It should be noted that all mutexes require initialization | ||||
| ** on the Win32 platform. | ||||
| */ | ||||
| #define SQLITE_W32_MUTEX_INITIALIZER { 0 } | ||||
| 
 | ||||
| #ifdef SQLITE_DEBUG | ||||
| #define SQLITE3_MUTEX_INITIALIZER(id) { SQLITE_W32_MUTEX_INITIALIZER, id, \ | ||||
|                                     0L, (DWORD)0, 0 } | ||||
| #else | ||||
| #define SQLITE3_MUTEX_INITIALIZER(id) { SQLITE_W32_MUTEX_INITIALIZER, id } | ||||
| #endif | ||||
| 
 | ||||
| #ifdef SQLITE_DEBUG | ||||
| /*
 | ||||
| ** The sqlite3_mutex_held() and sqlite3_mutex_notheld() routine are | ||||
| ** intended for use only inside assert() statements. | ||||
| */ | ||||
| static int winMutexHeld(sqlite3_mutex *p){ | ||||
|   return p->nRef!=0 && p->owner==GetCurrentThreadId(); | ||||
| } | ||||
| 
 | ||||
| static int winMutexNotheld2(sqlite3_mutex *p, DWORD tid){ | ||||
|   return p->nRef==0 || p->owner!=tid; | ||||
| } | ||||
| 
 | ||||
| static int winMutexNotheld(sqlite3_mutex *p){ | ||||
|   DWORD tid = GetCurrentThreadId(); | ||||
|   return winMutexNotheld2(p, tid); | ||||
| } | ||||
| #endif | ||||
| 
 | ||||
| /*
 | ||||
| ** Try to provide a memory barrier operation, needed for initialization | ||||
| ** and also for the xShmBarrier method of the VFS in cases when SQLite is | ||||
| ** compiled without mutexes (SQLITE_THREADSAFE=0). | ||||
| */ | ||||
| void sqlite3MemoryBarrier(void){ | ||||
| #if defined(SQLITE_MEMORY_BARRIER) | ||||
|   SQLITE_MEMORY_BARRIER; | ||||
| #elif defined(__GNUC__) | ||||
|   __sync_synchronize(); | ||||
| #elif MSVC_VERSION>=1300 | ||||
|   _ReadWriteBarrier(); | ||||
| #elif defined(MemoryBarrier) | ||||
|   MemoryBarrier(); | ||||
| #endif | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
| ** Initialize and deinitialize the mutex subsystem. | ||||
| */ | ||||
| static sqlite3_mutex winMutex_staticMutexes[] = { | ||||
|   SQLITE3_MUTEX_INITIALIZER(2), | ||||
|   SQLITE3_MUTEX_INITIALIZER(3), | ||||
|   SQLITE3_MUTEX_INITIALIZER(4), | ||||
|   SQLITE3_MUTEX_INITIALIZER(5), | ||||
|   SQLITE3_MUTEX_INITIALIZER(6), | ||||
|   SQLITE3_MUTEX_INITIALIZER(7), | ||||
|   SQLITE3_MUTEX_INITIALIZER(8), | ||||
|   SQLITE3_MUTEX_INITIALIZER(9), | ||||
|   SQLITE3_MUTEX_INITIALIZER(10), | ||||
|   SQLITE3_MUTEX_INITIALIZER(11), | ||||
|   SQLITE3_MUTEX_INITIALIZER(12), | ||||
|   SQLITE3_MUTEX_INITIALIZER(13) | ||||
| }; | ||||
| 
 | ||||
| static int winMutex_isInit = 0; | ||||
| static int winMutex_isNt = -1; /* <0 means "need to query" */ | ||||
| 
 | ||||
| /* As the winMutexInit() and winMutexEnd() functions are called as part
 | ||||
| ** of the sqlite3_initialize() and sqlite3_shutdown() processing, the | ||||
| ** "interlocked" magic used here is probably not strictly necessary. | ||||
| */ | ||||
| static LONG SQLITE_WIN32_VOLATILE winMutex_lock = 0; | ||||
| 
 | ||||
| int sqlite3_win32_is_nt(void); /* os_win.c */ | ||||
| void sqlite3_win32_sleep(DWORD milliseconds); /* os_win.c */ | ||||
| 
 | ||||
| static int winMutexInit(void){ | ||||
|   /* The first to increment to 1 does actual initialization */ | ||||
|   if( InterlockedCompareExchange(&winMutex_lock, 1, 0)==0 ){ | ||||
|     int i; | ||||
|     for(i=0; i<ArraySize(winMutex_staticMutexes); i++){ | ||||
| #if SQLITE_OS_WINRT | ||||
|       InitializeCriticalSectionEx(&winMutex_staticMutexes[i].mutex, 0, 0); | ||||
| #else | ||||
|       InitializeCriticalSection(&winMutex_staticMutexes[i].mutex); | ||||
| #endif | ||||
|     } | ||||
|     winMutex_isInit = 1; | ||||
|   }else{ | ||||
|     /* Another thread is (in the process of) initializing the static
 | ||||
|     ** mutexes */ | ||||
|     while( !winMutex_isInit ){ | ||||
|       sqlite3_win32_sleep(1); | ||||
|     } | ||||
|   } | ||||
|   return SQLITE_OK; | ||||
| } | ||||
| 
 | ||||
| static int winMutexEnd(void){ | ||||
|   /* The first to decrement to 0 does actual shutdown
 | ||||
|   ** (which should be the last to shutdown.) */ | ||||
|   if( InterlockedCompareExchange(&winMutex_lock, 0, 1)==1 ){ | ||||
|     if( winMutex_isInit==1 ){ | ||||
|       int i; | ||||
|       for(i=0; i<ArraySize(winMutex_staticMutexes); i++){ | ||||
|         DeleteCriticalSection(&winMutex_staticMutexes[i].mutex); | ||||
|       } | ||||
|       winMutex_isInit = 0; | ||||
|     } | ||||
|   } | ||||
|   return SQLITE_OK; | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
| ** The sqlite3_mutex_alloc() routine allocates a new | ||||
| ** mutex and returns a pointer to it.  If it returns NULL | ||||
| ** that means that a mutex could not be allocated.  SQLite | ||||
| ** will unwind its stack and return an error.  The argument | ||||
| ** to sqlite3_mutex_alloc() is one of these integer constants: | ||||
| ** | ||||
| ** <ul> | ||||
| ** <li>  SQLITE_MUTEX_FAST | ||||
| ** <li>  SQLITE_MUTEX_RECURSIVE | ||||
| ** <li>  SQLITE_MUTEX_STATIC_MAIN | ||||
| ** <li>  SQLITE_MUTEX_STATIC_MEM | ||||
| ** <li>  SQLITE_MUTEX_STATIC_OPEN | ||||
| ** <li>  SQLITE_MUTEX_STATIC_PRNG | ||||
| ** <li>  SQLITE_MUTEX_STATIC_LRU | ||||
| ** <li>  SQLITE_MUTEX_STATIC_PMEM | ||||
| ** <li>  SQLITE_MUTEX_STATIC_APP1 | ||||
| ** <li>  SQLITE_MUTEX_STATIC_APP2 | ||||
| ** <li>  SQLITE_MUTEX_STATIC_APP3 | ||||
| ** <li>  SQLITE_MUTEX_STATIC_VFS1 | ||||
| ** <li>  SQLITE_MUTEX_STATIC_VFS2 | ||||
| ** <li>  SQLITE_MUTEX_STATIC_VFS3 | ||||
| ** </ul> | ||||
| ** | ||||
| ** The first two constants cause sqlite3_mutex_alloc() to create | ||||
| ** a new mutex.  The new mutex is recursive when SQLITE_MUTEX_RECURSIVE | ||||
| ** is used but not necessarily so when SQLITE_MUTEX_FAST is used. | ||||
| ** The mutex implementation does not need to make a distinction | ||||
| ** between SQLITE_MUTEX_RECURSIVE and SQLITE_MUTEX_FAST if it does | ||||
| ** not want to.  But SQLite will only request a recursive mutex in | ||||
| ** cases where it really needs one.  If a faster non-recursive mutex | ||||
| ** implementation is available on the host platform, the mutex subsystem | ||||
| ** might return such a mutex in response to SQLITE_MUTEX_FAST. | ||||
| ** | ||||
| ** The other allowed parameters to sqlite3_mutex_alloc() each return | ||||
| ** a pointer to a static preexisting mutex.  Six static mutexes are | ||||
| ** used by the current version of SQLite.  Future versions of SQLite | ||||
| ** may add additional static mutexes.  Static mutexes are for internal | ||||
| ** use by SQLite only.  Applications that use SQLite mutexes should | ||||
| ** use only the dynamic mutexes returned by SQLITE_MUTEX_FAST or | ||||
| ** SQLITE_MUTEX_RECURSIVE. | ||||
| ** | ||||
| ** Note that if one of the dynamic mutex parameters (SQLITE_MUTEX_FAST | ||||
| ** or SQLITE_MUTEX_RECURSIVE) is used then sqlite3_mutex_alloc() | ||||
| ** returns a different mutex on every call.  But for the static | ||||
| ** mutex types, the same mutex is returned on every call that has | ||||
| ** the same type number. | ||||
| */ | ||||
| static sqlite3_mutex *winMutexAlloc(int iType){ | ||||
|   sqlite3_mutex *p; | ||||
| 
 | ||||
|   switch( iType ){ | ||||
|     case SQLITE_MUTEX_FAST: | ||||
|     case SQLITE_MUTEX_RECURSIVE: { | ||||
|       p = sqlite3MallocZero( sizeof(*p) ); | ||||
|       if( p ){ | ||||
|         p->id = iType; | ||||
| #ifdef SQLITE_DEBUG | ||||
| #ifdef SQLITE_WIN32_MUTEX_TRACE_DYNAMIC | ||||
|         p->trace = 1; | ||||
| #endif | ||||
| #endif | ||||
| #if SQLITE_OS_WINRT | ||||
|         InitializeCriticalSectionEx(&p->mutex, 0, 0); | ||||
| #else | ||||
|         InitializeCriticalSection(&p->mutex); | ||||
| #endif | ||||
|       } | ||||
|       break; | ||||
|     } | ||||
|     default: { | ||||
| #ifdef SQLITE_ENABLE_API_ARMOR | ||||
|       if( iType-2<0 || iType-2>=ArraySize(winMutex_staticMutexes) ){ | ||||
|         (void)SQLITE_MISUSE_BKPT; | ||||
|         return 0; | ||||
|       } | ||||
| #endif | ||||
|       p = &winMutex_staticMutexes[iType-2]; | ||||
| #ifdef SQLITE_DEBUG | ||||
| #ifdef SQLITE_WIN32_MUTEX_TRACE_STATIC | ||||
|       InterlockedCompareExchange(&p->trace, 1, 0); | ||||
| #endif | ||||
| #endif | ||||
|       break; | ||||
|     } | ||||
|   } | ||||
|   assert( p==0 || p->id==iType ); | ||||
|   return p; | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| /*
 | ||||
| ** This routine deallocates a previously | ||||
| ** allocated mutex.  SQLite is careful to deallocate every | ||||
| ** mutex that it allocates. | ||||
| */ | ||||
| static void winMutexFree(sqlite3_mutex *p){ | ||||
|   assert( p ); | ||||
|   assert( p->nRef==0 && p->owner==0 ); | ||||
|   if( p->id==SQLITE_MUTEX_FAST || p->id==SQLITE_MUTEX_RECURSIVE ){ | ||||
|     DeleteCriticalSection(&p->mutex); | ||||
|     sqlite3_free(p); | ||||
|   }else{ | ||||
| #ifdef SQLITE_ENABLE_API_ARMOR | ||||
|     (void)SQLITE_MISUSE_BKPT; | ||||
| #endif | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
| ** The sqlite3_mutex_enter() and sqlite3_mutex_try() routines attempt | ||||
| ** to enter a mutex.  If another thread is already within the mutex, | ||||
| ** sqlite3_mutex_enter() will block and sqlite3_mutex_try() will return | ||||
| ** SQLITE_BUSY.  The sqlite3_mutex_try() interface returns SQLITE_OK | ||||
| ** upon successful entry.  Mutexes created using SQLITE_MUTEX_RECURSIVE can | ||||
| ** be entered multiple times by the same thread.  In such cases the, | ||||
| ** mutex must be exited an equal number of times before another thread | ||||
| ** can enter.  If the same thread tries to enter any other kind of mutex | ||||
| ** more than once, the behavior is undefined. | ||||
| */ | ||||
| static void winMutexEnter(sqlite3_mutex *p){ | ||||
| #if defined(SQLITE_DEBUG) || defined(SQLITE_TEST) | ||||
|   DWORD tid = GetCurrentThreadId(); | ||||
| #endif | ||||
| #ifdef SQLITE_DEBUG | ||||
|   assert( p ); | ||||
|   assert( p->id==SQLITE_MUTEX_RECURSIVE || winMutexNotheld2(p, tid) ); | ||||
| #else | ||||
|   assert( p ); | ||||
| #endif | ||||
|   assert( winMutex_isInit==1 ); | ||||
|   EnterCriticalSection(&p->mutex); | ||||
| #ifdef SQLITE_DEBUG | ||||
|   assert( p->nRef>0 || p->owner==0 ); | ||||
|   p->owner = tid; | ||||
|   p->nRef++; | ||||
|   if( p->trace ){ | ||||
|     OSTRACE(("ENTER-MUTEX tid=%lu, mutex(%d)=%p (%d), nRef=%d\n", | ||||
|              tid, p->id, p, p->trace, p->nRef)); | ||||
|   } | ||||
| #endif | ||||
| } | ||||
| 
 | ||||
| static int winMutexTry(sqlite3_mutex *p){ | ||||
| #if defined(SQLITE_DEBUG) || defined(SQLITE_TEST) | ||||
|   DWORD tid = GetCurrentThreadId(); | ||||
| #endif | ||||
|   int rc = SQLITE_BUSY; | ||||
|   assert( p ); | ||||
|   assert( p->id==SQLITE_MUTEX_RECURSIVE || winMutexNotheld2(p, tid) ); | ||||
|   /*
 | ||||
|   ** The sqlite3_mutex_try() routine is very rarely used, and when it | ||||
|   ** is used it is merely an optimization.  So it is OK for it to always | ||||
|   ** fail. | ||||
|   ** | ||||
|   ** The TryEnterCriticalSection() interface is only available on WinNT. | ||||
|   ** And some windows compilers complain if you try to use it without | ||||
|   ** first doing some #defines that prevent SQLite from building on Win98. | ||||
|   ** For that reason, we will omit this optimization for now.  See | ||||
|   ** ticket #2685. | ||||
|   */ | ||||
| #if defined(_WIN32_WINNT) && _WIN32_WINNT >= 0x0400 | ||||
|   assert( winMutex_isInit==1 ); | ||||
|   assert( winMutex_isNt>=-1 && winMutex_isNt<=1 ); | ||||
|   if( winMutex_isNt<0 ){ | ||||
|     winMutex_isNt = sqlite3_win32_is_nt(); | ||||
|   } | ||||
|   assert( winMutex_isNt==0 || winMutex_isNt==1 ); | ||||
|   if( winMutex_isNt && TryEnterCriticalSection(&p->mutex) ){ | ||||
| #ifdef SQLITE_DEBUG | ||||
|     p->owner = tid; | ||||
|     p->nRef++; | ||||
| #endif | ||||
|     rc = SQLITE_OK; | ||||
|   } | ||||
| #else | ||||
|   UNUSED_PARAMETER(p); | ||||
| #endif | ||||
| #ifdef SQLITE_DEBUG | ||||
|   if( p->trace ){ | ||||
|     OSTRACE(("TRY-MUTEX tid=%lu, mutex(%d)=%p (%d), owner=%lu, nRef=%d, rc=%s\n", | ||||
|              tid, p->id, p, p->trace, p->owner, p->nRef, sqlite3ErrName(rc))); | ||||
|   } | ||||
| #endif | ||||
|   return rc; | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
| ** The sqlite3_mutex_leave() routine exits a mutex that was | ||||
| ** previously entered by the same thread.  The behavior | ||||
| ** is undefined if the mutex is not currently entered or | ||||
| ** is not currently allocated.  SQLite will never do either. | ||||
| */ | ||||
| static void winMutexLeave(sqlite3_mutex *p){ | ||||
| #if defined(SQLITE_DEBUG) || defined(SQLITE_TEST) | ||||
|   DWORD tid = GetCurrentThreadId(); | ||||
| #endif | ||||
|   assert( p ); | ||||
| #ifdef SQLITE_DEBUG | ||||
|   assert( p->nRef>0 ); | ||||
|   assert( p->owner==tid ); | ||||
|   p->nRef--; | ||||
|   if( p->nRef==0 ) p->owner = 0; | ||||
|   assert( p->nRef==0 || p->id==SQLITE_MUTEX_RECURSIVE ); | ||||
| #endif | ||||
|   assert( winMutex_isInit==1 ); | ||||
|   LeaveCriticalSection(&p->mutex); | ||||
| #ifdef SQLITE_DEBUG | ||||
|   if( p->trace ){ | ||||
|     OSTRACE(("LEAVE-MUTEX tid=%lu, mutex(%d)=%p (%d), nRef=%d\n", | ||||
|              tid, p->id, p, p->trace, p->nRef)); | ||||
|   } | ||||
| #endif | ||||
| } | ||||
| 
 | ||||
| sqlite3_mutex_methods const *sqlite3DefaultMutex(void){ | ||||
|   static const sqlite3_mutex_methods sMutex = { | ||||
|     winMutexInit, | ||||
|     winMutexEnd, | ||||
|     winMutexAlloc, | ||||
|     winMutexFree, | ||||
|     winMutexEnter, | ||||
|     winMutexTry, | ||||
|     winMutexLeave, | ||||
| #ifdef SQLITE_DEBUG | ||||
|     winMutexHeld, | ||||
|     winMutexNotheld | ||||
| #else | ||||
|     0, | ||||
|     0 | ||||
| #endif | ||||
|   }; | ||||
|   return &sMutex; | ||||
| } | ||||
| 
 | ||||
| #endif /* SQLITE_MUTEX_W32 */ | ||||
							
								
								
									
										4
									
								
								third_party/sqlite3/notify.c
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								third_party/sqlite3/notify.c
									
										
									
									
										vendored
									
									
								
							|  | @ -13,8 +13,8 @@ | |||
| ** This file contains the implementation of the sqlite3_unlock_notify() | ||||
| ** API method and its associated functionality. | ||||
| */ | ||||
| #include "third_party/sqlite3/btreeInt.h" | ||||
| #include "third_party/sqlite3/sqliteInt.h" | ||||
| #include "third_party/sqlite3/btreeInt.inc" | ||||
| #include "third_party/sqlite3/sqliteInt.inc" | ||||
| /* clang-format off */ | ||||
| 
 | ||||
| /* Omit this entire file if SQLITE_ENABLE_UNLOCK_NOTIFY is not defined. */ | ||||
|  |  | |||
							
								
								
									
										2
									
								
								third_party/sqlite3/os.c
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								third_party/sqlite3/os.c
									
										
									
									
										vendored
									
									
								
							|  | @ -13,7 +13,7 @@ | |||
| ** This file contains OS interface code that is common to all | ||||
| ** architectures. | ||||
| */ | ||||
| #include "third_party/sqlite3/sqliteInt.h" | ||||
| #include "third_party/sqlite3/sqliteInt.inc" | ||||
| /* clang-format off */ | ||||
| 
 | ||||
| /*
 | ||||
|  |  | |||
|  | @ -25,7 +25,7 @@ | |||
| ** Attempt to automatically detect the operating system and setup the | ||||
| ** necessary pre-processor macros for it. | ||||
| */ | ||||
| #include "third_party/sqlite3/os_setup.h"
 | ||||
| #include "third_party/sqlite3/os_setup.inc"
 | ||||
| 
 | ||||
| /* If the SET_FULLSYNC macro is not defined above, then make it | ||||
| ** a no-op | ||||
|  | @ -40,7 +40,7 @@ | |||
| ** hwtime.h contains inline assembler code for implementing | ||||
| ** high-performance timing routines. | ||||
| */ | ||||
| #include "third_party/sqlite3/hwtime.h"
 | ||||
| #include "third_party/sqlite3/hwtime.inc"
 | ||||
| 
 | ||||
| static sqlite_uint64 g_start; | ||||
| static sqlite_uint64 g_elapsed; | ||||
							
								
								
									
										12
									
								
								third_party/sqlite3/os_unix.c
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										12
									
								
								third_party/sqlite3/os_unix.c
									
										
									
									
										vendored
									
									
								
							|  | @ -43,7 +43,7 @@ | |||
| **   *  Definitions of sqlite3_vfs objects for all locking methods | ||||
| **      plus implementations of sqlite3_os_init() and sqlite3_os_end(). | ||||
| */ | ||||
| #include "third_party/sqlite3/sqliteInt.h" | ||||
| #include "third_party/sqlite3/sqliteInt.inc" | ||||
| #if SQLITE_OS_UNIX /* This file is used on unix only */ | ||||
|                    /* clang-format off */ | ||||
| 
 | ||||
|  | @ -102,15 +102,7 @@ | |||
| #include "libc/sysv/consts/prot.h" | ||||
| #include "libc/time/struct/tm.h" | ||||
| #include "libc/time/time.h" | ||||
| #if !defined(SQLITE_OMIT_WAL) || SQLITE_MAX_MMAP_SIZE > 0 | ||||
| #include "libc/mem/mem.h" | ||||
| #endif | ||||
| 
 | ||||
| #if SQLITE_ENABLE_LOCKING_STYLE | ||||
| # include <sys/ioctl.h> | ||||
| # include <sys/file.h> | ||||
| # include <sys/param.h> | ||||
| #endif /* SQLITE_ENABLE_LOCKING_STYLE */ | ||||
| 
 | ||||
| /*
 | ||||
| ** Try to determine if gethostuuid() is available based on standard | ||||
|  | @ -305,7 +297,7 @@ static pid_t randomnessPid = 0; | |||
| /*
 | ||||
| ** Include code that is common to all os_*.c files | ||||
| */ | ||||
| #include "third_party/sqlite3/os_common.h" | ||||
| #include "third_party/sqlite3/os_common.inc" | ||||
| 
 | ||||
| /*
 | ||||
| ** Define various macros that are missing from some systems. | ||||
|  |  | |||
							
								
								
									
										9
									
								
								third_party/sqlite3/os_win.c
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										9
									
								
								third_party/sqlite3/os_win.c
									
										
									
									
										vendored
									
									
								
							|  | @ -12,19 +12,14 @@ | |||
| ** | ||||
| ** This file contains code that is specific to Windows. | ||||
| */ | ||||
| #include "third_party/sqlite3/sqliteInt.h" | ||||
| #include "third_party/sqlite3/sqliteInt.inc" | ||||
| #if SQLITE_OS_WIN /* This file is used for Windows only */ | ||||
| /* clang-format off */ | ||||
| 
 | ||||
| /*
 | ||||
| ** Include code that is common to all os_*.c files | ||||
| */ | ||||
| #include "third_party/sqlite3/os_common.h" | ||||
| 
 | ||||
| /*
 | ||||
| ** Include the header file for the Windows VFS. | ||||
| */ | ||||
| #include "third_party/sqlite3/os_win.h" | ||||
| #include "third_party/sqlite3/os_common.inc" | ||||
| 
 | ||||
| /*
 | ||||
| ** Compiling and using WAL mode requires several APIs that are only | ||||
|  |  | |||
							
								
								
									
										90
									
								
								third_party/sqlite3/os_win.h
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										90
									
								
								third_party/sqlite3/os_win.h
									
										
									
									
										vendored
									
									
								
							|  | @ -1,90 +0,0 @@ | |||
| /*
 | ||||
| ** 2013 November 25 | ||||
| ** | ||||
| ** The author disclaims copyright to this source code.  In place of | ||||
| ** a legal notice, here is a blessing: | ||||
| ** | ||||
| **    May you do good and not evil. | ||||
| **    May you find forgiveness for yourself and forgive others. | ||||
| **    May you share freely, never taking more than you give. | ||||
| ** | ||||
| ****************************************************************************** | ||||
| ** | ||||
| ** This file contains code that is specific to Windows. | ||||
| */ | ||||
| #ifndef SQLITE_OS_WIN_H | ||||
| #define SQLITE_OS_WIN_H | ||||
| /* clang-format off */ | ||||
| 
 | ||||
| /*
 | ||||
| ** Include the primary Windows SDK header file. | ||||
| */ | ||||
| #include "third_party/sqlite3/windows.h" | ||||
| 
 | ||||
| #ifdef __CYGWIN__ | ||||
| #include <sys/cygwin.h> | ||||
| 
 | ||||
| #include "libc/errno.h" /* amalgamator: dontcache */ | ||||
| #endif | ||||
| 
 | ||||
| /*
 | ||||
| ** Determine if we are dealing with Windows NT. | ||||
| ** | ||||
| ** We ought to be able to determine if we are compiling for Windows 9x or | ||||
| ** Windows NT using the _WIN32_WINNT macro as follows: | ||||
| ** | ||||
| ** #if defined(_WIN32_WINNT) | ||||
| ** # define SQLITE_OS_WINNT 1 | ||||
| ** #else | ||||
| ** # define SQLITE_OS_WINNT 0 | ||||
| ** #endif | ||||
| ** | ||||
| ** However, Visual Studio 2005 does not set _WIN32_WINNT by default, as | ||||
| ** it ought to, so the above test does not work.  We'll just assume that | ||||
| ** everything is Windows NT unless the programmer explicitly says otherwise | ||||
| ** by setting SQLITE_OS_WINNT to 0. | ||||
| */ | ||||
| #if SQLITE_OS_WIN && !defined(SQLITE_OS_WINNT) | ||||
| # define SQLITE_OS_WINNT 1 | ||||
| #endif | ||||
| 
 | ||||
| /*
 | ||||
| ** Determine if we are dealing with Windows CE - which has a much reduced | ||||
| ** API. | ||||
| */ | ||||
| #if defined(_WIN32_WCE) | ||||
| # define SQLITE_OS_WINCE 1 | ||||
| #else | ||||
| # define SQLITE_OS_WINCE 0 | ||||
| #endif | ||||
| 
 | ||||
| /*
 | ||||
| ** Determine if we are dealing with WinRT, which provides only a subset of | ||||
| ** the full Win32 API. | ||||
| */ | ||||
| #if !defined(SQLITE_OS_WINRT) | ||||
| # define SQLITE_OS_WINRT 0 | ||||
| #endif | ||||
| 
 | ||||
| /*
 | ||||
| ** For WinCE, some API function parameters do not appear to be declared as | ||||
| ** volatile. | ||||
| */ | ||||
| #if SQLITE_OS_WINCE | ||||
| # define SQLITE_WIN32_VOLATILE | ||||
| #else | ||||
| # define SQLITE_WIN32_VOLATILE volatile | ||||
| #endif | ||||
| 
 | ||||
| /*
 | ||||
| ** For some Windows sub-platforms, the _beginthreadex() / _endthreadex() | ||||
| ** functions are not available (e.g. those not using MSVC, Cygwin, etc). | ||||
| */ | ||||
| #if SQLITE_OS_WIN && !SQLITE_OS_WINCE && !SQLITE_OS_WINRT && \ | ||||
|     SQLITE_THREADSAFE>0 && !defined(__CYGWIN__) | ||||
| # define SQLITE_OS_WIN_THREADS 1 | ||||
| #else | ||||
| # define SQLITE_OS_WIN_THREADS 0 | ||||
| #endif | ||||
| 
 | ||||
| #endif /* SQLITE_OS_WIN_H */ | ||||
							
								
								
									
										4
									
								
								third_party/sqlite3/pager.c
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								third_party/sqlite3/pager.c
									
										
									
									
										vendored
									
									
								
							|  | @ -19,8 +19,8 @@ | |||
| ** another is writing. | ||||
| */ | ||||
| #ifndef SQLITE_OMIT_DISKIO | ||||
| #include "third_party/sqlite3/sqliteInt.h" | ||||
| #include "third_party/sqlite3/wal.h" | ||||
| #include "third_party/sqlite3/sqliteInt.inc" | ||||
| #include "third_party/sqlite3/wal.inc" | ||||
| /* clang-format off */ | ||||
| 
 | ||||
| 
 | ||||
|  |  | |||
							
								
								
									
										2
									
								
								third_party/sqlite3/parse.c
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								third_party/sqlite3/parse.c
									
										
									
									
										vendored
									
									
								
							|  | @ -23,7 +23,7 @@ | |||
| #line 58 "parse.y" | ||||
| /* clang-format off */ | ||||
| 
 | ||||
| #include "third_party/sqlite3/sqliteInt.h" | ||||
| #include "third_party/sqlite3/sqliteInt.inc" | ||||
| 
 | ||||
| /*
 | ||||
| ** Disable all error recovery processing in the parser push-down | ||||
|  |  | |||
							
								
								
									
										2
									
								
								third_party/sqlite3/pcache.c
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								third_party/sqlite3/pcache.c
									
										
									
									
										vendored
									
									
								
							|  | @ -11,7 +11,7 @@ | |||
| ************************************************************************* | ||||
| ** This file implements that page cache. | ||||
| */ | ||||
| #include "third_party/sqlite3/sqliteInt.h" | ||||
| #include "third_party/sqlite3/sqliteInt.inc" | ||||
| /* clang-format off */ | ||||
| 
 | ||||
| /*
 | ||||
|  |  | |||
							
								
								
									
										2
									
								
								third_party/sqlite3/pcache1.c
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								third_party/sqlite3/pcache1.c
									
										
									
									
										vendored
									
									
								
							|  | @ -80,7 +80,7 @@ | |||
| ** show that method (3) with N==100 provides about a 5% performance boost for | ||||
| ** common workloads. | ||||
| */ | ||||
| #include "third_party/sqlite3/sqliteInt.h" | ||||
| #include "third_party/sqlite3/sqliteInt.inc" | ||||
| /* clang-format off */ | ||||
| 
 | ||||
| typedef struct PCache1 PCache1; | ||||
|  |  | |||
							
								
								
									
										4
									
								
								third_party/sqlite3/pragma.c
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								third_party/sqlite3/pragma.c
									
										
									
									
										vendored
									
									
								
							|  | @ -11,7 +11,7 @@ | |||
| ************************************************************************* | ||||
| ** This file contains code used to implement the PRAGMA command. | ||||
| */ | ||||
| #include "third_party/sqlite3/sqliteInt.h" | ||||
| #include "third_party/sqlite3/sqliteInt.inc" | ||||
| /* clang-format off */ | ||||
| 
 | ||||
| #if !defined(SQLITE_ENABLE_LOCKING_STYLE) | ||||
|  | @ -29,7 +29,7 @@ | |||
| ** lexicographical order to facility a binary search of the pragma name. | ||||
| ** Do not edit pragma.h directly.  Edit and rerun the script in at  | ||||
| ** ../tool/mkpragmatab.tcl. */ | ||||
| #include "third_party/sqlite3/pragma.h" | ||||
| #include "third_party/sqlite3/pragma.inc" | ||||
| 
 | ||||
| /*
 | ||||
| ** Interpret the given string as a safety level.  Return 0 for OFF, | ||||
|  |  | |||
							
								
								
									
										2
									
								
								third_party/sqlite3/prepare.c
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								third_party/sqlite3/prepare.c
									
										
									
									
										vendored
									
									
								
							|  | @ -13,7 +13,7 @@ | |||
| ** interface, and routines that contribute to loading the database schema | ||||
| ** from disk. | ||||
| */ | ||||
| #include "third_party/sqlite3/sqliteInt.h" | ||||
| #include "third_party/sqlite3/sqliteInt.inc" | ||||
| /* clang-format off */ | ||||
| 
 | ||||
| /*
 | ||||
|  |  | |||
							
								
								
									
										2
									
								
								third_party/sqlite3/printf.c
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								third_party/sqlite3/printf.c
									
										
									
									
										vendored
									
									
								
							|  | @ -9,7 +9,7 @@ | |||
| ** library, though the implementation here has enhancements to support | ||||
| ** SQLite. | ||||
| */ | ||||
| #include "third_party/sqlite3/sqliteInt.h" | ||||
| #include "third_party/sqlite3/sqliteInt.inc" | ||||
| /* clang-format off */ | ||||
| 
 | ||||
| /*
 | ||||
|  |  | |||
							
								
								
									
										2
									
								
								third_party/sqlite3/random.c
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								third_party/sqlite3/random.c
									
										
									
									
										vendored
									
									
								
							|  | @ -15,7 +15,7 @@ | |||
| ** Random numbers are used by some of the database backends in order | ||||
| ** to generate random integer keys for tables or random filenames. | ||||
| */ | ||||
| #include "third_party/sqlite3/sqliteInt.h" | ||||
| #include "third_party/sqlite3/sqliteInt.inc" | ||||
| /* clang-format off */ | ||||
| 
 | ||||
| 
 | ||||
|  |  | |||
Some files were not shown because too many files have changed in this diff Show more
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue