/* ** 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. ** ** Compiling: ** ** (linux) gcc -fPIC -shared zorder.c -o zorder.so ** (mac) clang -fPIC -dynamiclib zorder.c -o zorder.dylib ** (windows) cl zorder.c -link -dll -out:zorder.dll ** ** Usage example: ** ** .load ./zorder ** SELECT zorder(1,2,3,4); */ #include "sqlite3ext.h" SQLITE_EXTENSION_INIT1 #include #include /* ** Functions: zorder(X0,X1,....) ** ** Convert integers X0, X1, ... into morton code. There must be at least ** two arguments. There may be no more than 24 arguments. ** ** The output is a signed 64-bit integer. If any argument is too large ** to be successfully encoded into a morton code, an error is raised. */ static void zorderFunc( sqlite3_context *context, int argc, sqlite3_value **argv ){ sqlite3_int64 z, x[24]; int i, j; z = 0; if( argc<2 || argc>24 ){ sqlite3_result_error(context, "zorder() needs between 2 and 24 arguments4", -1); return; } for(i=0; i>= 1; } sqlite3_result_int64(context, z); for(i=0; i24 ){ sqlite3_result_error(context, "N argument to unzorder(Z,N,K) should be between 2 and 24", -1); return; } i = sqlite3_value_int64(argv[2]); if( i<0 || i>=n ){ sqlite3_result_error(context, "K argument to unzorder(Z,N,K) should be between 0 and N-1", -1); return; } x = 0; for(k=0, j=i; j<63; j+=n, k++){ x |= ((z>>j)&1)<