mirror of
				https://github.com/sqlite/sqlite.git
				synced 2025-10-21 11:13:54 +03:00 
			
		
		
		
	
		
			
				
	
	
		
			98 lines
		
	
	
		
			2.6 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			98 lines
		
	
	
		
			2.6 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*
 | |
| ** 2024-05-24
 | |
| **
 | |
| ** 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.
 | |
| **
 | |
| ******************************************************************************
 | |
| **
 | |
| ** An SQL function that return pseudo-random non-negative integers.
 | |
| **
 | |
| **      SELECT stmtrand(123);
 | |
| **
 | |
| ** A special feature of this function is that the same sequence of random
 | |
| ** integers is returned for each invocation of the statement.  This makes
 | |
| ** the results repeatable, and hence useful for testing.  The argument is
 | |
| ** an integer which is the seed for the random number sequence.  The seed
 | |
| ** is used by the first invocation of this function only and is ignored
 | |
| ** for all subsequent calls within the same statement.
 | |
| **
 | |
| ** Resetting a statement (sqlite3_reset()) also resets the random number
 | |
| ** sequence.
 | |
| */
 | |
| #include "sqlite3ext.h"
 | |
| SQLITE_EXTENSION_INIT1
 | |
| #include <assert.h>
 | |
| #include <string.h>
 | |
| 
 | |
| /* State of the pseudo-random number generator */
 | |
| typedef struct Stmtrand {
 | |
|   unsigned int x, y;
 | |
| } Stmtrand;
 | |
| 
 | |
| /* auxdata key */
 | |
| #define STMTRAND_KEY  (-4418371)
 | |
| 
 | |
| /*
 | |
| ** Function:     stmtrand(SEED)
 | |
| **
 | |
| ** Return a pseudo-random number.
 | |
| */
 | |
| static void stmtrandFunc(
 | |
|   sqlite3_context *context,
 | |
|   int argc,
 | |
|   sqlite3_value **argv
 | |
| ){
 | |
|   Stmtrand *p;
 | |
| 
 | |
|   p = (Stmtrand*)sqlite3_get_auxdata(context, STMTRAND_KEY);
 | |
|   if( p==0 ){
 | |
|     unsigned int seed;
 | |
|     p = sqlite3_malloc( sizeof(*p) );
 | |
|     if( p==0 ){
 | |
|       sqlite3_result_error_nomem(context);
 | |
|       return;
 | |
|     }
 | |
|     if( argc>=1 ){
 | |
|       seed = (unsigned int)sqlite3_value_int(argv[0]);
 | |
|     }else{
 | |
|       seed = 0;
 | |
|     }
 | |
|     p->x = seed | 1;
 | |
|     p->y = seed;
 | |
|     sqlite3_set_auxdata(context, STMTRAND_KEY, p, sqlite3_free);
 | |
|     p = (Stmtrand*)sqlite3_get_auxdata(context, STMTRAND_KEY);
 | |
|     if( p==0 ){
 | |
|       sqlite3_result_error_nomem(context);
 | |
|       return;
 | |
|     }
 | |
|   }
 | |
|   p->x = (p->x>>1) ^ ((1+~(p->x&1)) & 0xd0000001);
 | |
|   p->y = p->y*1103515245 + 12345;
 | |
|   sqlite3_result_int(context, (int)((p->x ^ p->y)&0x7fffffff));
 | |
| }
 | |
| 
 | |
| #ifdef _WIN32
 | |
| __declspec(dllexport)
 | |
| #endif
 | |
| int sqlite3_stmtrand_init(
 | |
|   sqlite3 *db, 
 | |
|   char **pzErrMsg, 
 | |
|   const sqlite3_api_routines *pApi
 | |
| ){
 | |
|   int rc = SQLITE_OK;
 | |
|   SQLITE_EXTENSION_INIT2(pApi);
 | |
|   (void)pzErrMsg;  /* Unused parameter */
 | |
|   rc = sqlite3_create_function(db, "stmtrand", 1, SQLITE_UTF8, 0,
 | |
|                                stmtrandFunc, 0, 0);
 | |
|   if( rc==SQLITE_OK ){
 | |
|     rc = sqlite3_create_function(db, "stmtrand", 0, SQLITE_UTF8, 0,
 | |
|                                  stmtrandFunc, 0, 0);
 | |
|   }
 | |
|   return rc;
 | |
| }
 | 
