| /* |
| ** 2018-02-09 |
| ** |
| ** 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. |
| ** |
| ****************************************************************************** |
| ** |
| ** SQL functions for z-order (Morton code) transformations. |
| ** |
| ** zorder(X0,X0,..,xN) Generate an N+1 dimension Morton code |
| ** |
| ** unzorder(Z,N,I) Extract the I-th dimension from N-dimensional |
| ** Morton code Z. |
| */ |
| #include "sqlite3ext.h" |
| SQLITE_EXTENSION_INIT1 |
| #include <assert.h> |
| #include <string.h> |
| |
| /* |
| ** Functions: zorder(X0,X1,....) |
| ** |
| ** Convert integers X0, X1, ... into morton code. |
| ** |
| ** The output is a signed 64-bit integer. If any argument is too large, |
| ** an error is thrown. |
| */ |
| static void zorderFunc( |
| sqlite3_context *context, |
| int argc, |
| sqlite3_value **argv |
| ){ |
| sqlite3_int64 z, x[63]; |
| int i, j; |
| z = 0; |
| for(i=0; i<argc; i++){ |
| x[i] = sqlite3_value_int64(argv[i]); |
| } |
| if( argc>0 ){ |
| for(i=0; i<63; i++){ |
| j = i%argc; |
| z |= (x[j]&1)<<i; |
| x[j] >>= 1; |
| } |
| } |
| sqlite3_result_int64(context, z); |
| for(i=0; i<argc; i++){ |
| if( x[i] ){ |
| sqlite3_result_error(context, "parameter too large", -1); |
| } |
| } |
| } |
| |
| |
| /* |
| ** Functions: unzorder(Z,N,I) |
| ** |
| ** Assuming that Z is an N-dimensional Morton code, extract the I-th |
| ** dimension. |
| */ |
| static void unzorderFunc( |
| sqlite3_context *context, |
| int argc, |
| sqlite3_value **argv |
| ){ |
| sqlite3_int64 z, n, i, x; |
| int j, k; |
| z = sqlite3_value_int64(argv[0]); |
| n = sqlite3_value_int64(argv[1]); |
| i = sqlite3_value_int64(argv[2]); |
| x = 0; |
| for(k=0, j=i; j<63; j+=n, k++){ |
| x |= ((z>>j)&1)<<k; |
| } |
| sqlite3_result_int64(context, x); |
| } |
| |
| |
| #ifdef _WIN32 |
| __declspec(dllexport) |
| #endif |
| int sqlite3_zorder_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, "zorder", -1, SQLITE_UTF8, 0, |
| zorderFunc, 0, 0); |
| if( rc==SQLITE_OK ){ |
| rc = sqlite3_create_function(db, "unzorder", 3, SQLITE_UTF8, 0, |
| unzorderFunc, 0, 0); |
| } |
| return rc; |
| } |