Update contrib.
4 ** Portions Copyright (c) 2008 Nokia Corporation and/or its subsidiaries. All rights reserved.
6 ** The author disclaims copyright to this source code. In place of
7 ** a legal notice, here is a blessing:
9 ** May you do good and not evil.
10 ** May you find forgiveness for yourself and forgive others.
11 ** May you share freely, never taking more than you give.
13 *************************************************************************
14 ** Code for testing all sorts of SQLite interfaces. This code
15 ** implements TCL commands for reading and writing the binary
16 ** database files and displaying the content of those files as
17 ** hexadecimal. We could, in theory, use the built-in "binary"
18 ** command of TCL to do a lot of this, but there are some issues
19 ** with historical versions of the "binary" command. So it seems
20 ** easier and safer to build our own mechanism.
22 ** $Id: test_hexio.c,v 1.7 2008/05/12 16:17:42 drh Exp $
24 #include "sqliteInt.h"
30 #include <sys/param.h>
32 extern char* FullFilePath(char* aPath, const char* aFileName);
34 /* Symbian OS - often TCL scripts attempt to open a file just using the file name without path.
35 This is not working on the hardware. The function bellow will contstruct the full file name. */
36 extern char* GetFullFilePath(char* aPath, const char* aFileName)
38 return FullFilePath(aPath, aFileName);
42 ** Convert binary to hex. The input zBuf[] contains N bytes of
43 ** binary data. zBuf[] is 2*n+1 bytes long. Overwrite zBuf[]
44 ** with a hexadecimal representation of its original binary input.
46 void sqlite3TestBinToHex(unsigned char *zBuf, int N){
47 const unsigned char zHex[] = "0123456789ABCDEF";
52 for(j=N-1; j>=0; j--){
54 zBuf[i--] = zHex[c&0xf];
55 zBuf[i--] = zHex[c>>4];
61 ** Convert hex to binary. The input zIn[] contains N bytes of
62 ** hexadecimal. Convert this into binary and write aOut[] with
63 ** the binary data. Spaces in the original input are ignored.
64 ** Return the number of bytes of binary rendered.
66 int sqlite3TestHexToBin(const unsigned char *zIn, int N, unsigned char *aOut){
67 const unsigned char aMap[] = {
68 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
69 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
70 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
71 1, 2, 3, 4, 5, 6, 7, 8, 9,10, 0, 0, 0, 0, 0, 0,
72 0,11,12,13,14,15,16, 0, 0, 0, 0, 0, 0, 0, 0, 0,
73 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
74 0,11,12,13,14,15,16, 0, 0, 0, 0, 0, 0, 0, 0, 0,
75 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
76 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
77 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
78 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
79 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
80 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
81 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
82 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
83 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
105 ** Usage: hexio_read FILENAME OFFSET AMT
107 ** Read AMT bytes from file FILENAME beginning at OFFSET from the
108 ** beginning of the file. Convert that information to hexadecimal
109 ** and return the resulting HEX string.
111 static int hexio_read(
115 Tcl_Obj *CONST objv[]
122 char fnamebuf[MAXPATHLEN + 1];
125 Tcl_WrongNumArgs(interp, 1, objv, "FILENAME OFFSET AMT");
128 if( Tcl_GetIntFromObj(interp, objv[2], &offset) ) return TCL_ERROR;
129 if( Tcl_GetIntFromObj(interp, objv[3], &amt) ) return TCL_ERROR;
130 zFile = Tcl_GetString(objv[1]);
131 zBuf = sqlite3_malloc( amt*2+1 );
135 if(GetFullFilePath(fnamebuf, zFile) == 0)
137 in = fopen(fnamebuf, "rb");
139 in = fopen(fnamebuf, "r");
142 Tcl_AppendResult(interp, "cannot open input file ", fnamebuf, 0);
145 fseek(in, offset, SEEK_SET);
146 got = fread(zBuf, 1, amt, in);
151 sqlite3TestBinToHex(zBuf, got);
152 Tcl_AppendResult(interp, zBuf, 0);
159 ** Usage: hexio_write FILENAME OFFSET DATA
161 ** Write DATA into file FILENAME beginning at OFFSET from the
162 ** beginning of the file. DATA is expressed in hexadecimal.
164 static int hexio_write(
168 Tcl_Obj *CONST objv[]
171 int nIn, nOut, written;
173 const unsigned char *zIn;
176 char fnamebuf[MAXPATHLEN + 1];
179 Tcl_WrongNumArgs(interp, 1, objv, "FILENAME OFFSET HEXDATA");
182 if( Tcl_GetIntFromObj(interp, objv[2], &offset) ) return TCL_ERROR;
183 zFile = Tcl_GetString(objv[1]);
184 zIn = (const unsigned char *)Tcl_GetStringFromObj(objv[3], &nIn);
185 aOut = sqlite3_malloc( nIn/2 );
189 nOut = sqlite3TestHexToBin(zIn, nIn, aOut);
190 if(GetFullFilePath(fnamebuf, zFile) == 0)
192 out = fopen(fnamebuf, "r+b");
194 out = fopen(fnamebuf, "r+");
197 Tcl_AppendResult(interp, "cannot open output file ", fnamebuf, 0);
200 fseek(out, offset, SEEK_SET);
201 written = fwrite(aOut, 1, nOut, out);
204 Tcl_SetObjResult(interp, Tcl_NewIntObj(written));
209 ** USAGE: hexio_get_int HEXDATA
211 ** Interpret the HEXDATA argument as a big-endian integer. Return
212 ** the value of that integer. HEXDATA can contain between 2 and 8
213 ** hexadecimal digits.
215 static int hexio_get_int(
219 Tcl_Obj *CONST objv[]
223 const unsigned char *zIn;
225 unsigned char aNum[4];
228 Tcl_WrongNumArgs(interp, 1, objv, "HEXDATA");
231 zIn = (const unsigned char *)Tcl_GetStringFromObj(objv[1], &nIn);
232 aOut = sqlite3_malloc( nIn/2 );
236 nOut = sqlite3TestHexToBin(zIn, nIn, aOut);
238 memcpy(aNum, aOut, 4);
240 memset(aNum, 0, sizeof(aNum));
241 memcpy(&aNum[4-nOut], aOut, nOut);
244 val = (aNum[0]<<24) | (aNum[1]<<16) | (aNum[2]<<8) | aNum[3];
245 Tcl_SetObjResult(interp, Tcl_NewIntObj(val));
251 ** USAGE: hexio_render_int16 INTEGER
253 ** Render INTEGER has a 16-bit big-endian integer in hexadecimal.
255 static int hexio_render_int16(
259 Tcl_Obj *CONST objv[]
262 unsigned char aNum[10];
265 Tcl_WrongNumArgs(interp, 1, objv, "INTEGER");
268 if( Tcl_GetIntFromObj(interp, objv[1], &val) ) return TCL_ERROR;
271 sqlite3TestBinToHex(aNum, 2);
272 Tcl_SetObjResult(interp, Tcl_NewStringObj((char*)aNum, 4));
278 ** USAGE: hexio_render_int32 INTEGER
280 ** Render INTEGER has a 32-bit big-endian integer in hexadecimal.
282 static int hexio_render_int32(
286 Tcl_Obj *CONST objv[]
289 unsigned char aNum[10];
292 Tcl_WrongNumArgs(interp, 1, objv, "INTEGER");
295 if( Tcl_GetIntFromObj(interp, objv[1], &val) ) return TCL_ERROR;
300 sqlite3TestBinToHex(aNum, 4);
301 Tcl_SetObjResult(interp, Tcl_NewStringObj((char*)aNum, 8));
306 ** USAGE: utf8_to_utf8 HEX
308 ** The argument is a UTF8 string represented in hexadecimal.
309 ** The UTF8 might not be well-formed. Run this string through
310 ** sqlite3Utf8to8() convert it back to hex and return the result.
312 static int utf8_to_utf8(
316 Tcl_Obj *CONST objv[]
321 const unsigned char *zOrig;
324 Tcl_WrongNumArgs(interp, 1, objv, "HEX");
327 zOrig = (unsigned char *)Tcl_GetStringFromObj(objv[1], &n);
328 z = sqlite3_malloc( n+3 );
329 n = sqlite3TestHexToBin(zOrig, n, z);
331 nOut = sqlite3Utf8To8(z);
332 sqlite3TestBinToHex(z,nOut);
333 Tcl_AppendResult(interp, (char*)z, 0);
341 ** Register commands with the TCL interpreter.
343 int Sqlitetest_hexio_Init(Tcl_Interp *interp){
346 Tcl_ObjCmdProc *xProc;
348 { "hexio_read", hexio_read },
349 { "hexio_write", hexio_write },
350 { "hexio_get_int", hexio_get_int },
351 { "hexio_render_int16", hexio_render_int16 },
352 { "hexio_render_int32", hexio_render_int32 },
353 { "utf8_to_utf8", utf8_to_utf8 },
356 for(i=0; i<sizeof(aObjCmd)/sizeof(aObjCmd[0]); i++){
357 Tcl_CreateObjCommand(interp, aObjCmd[i].zName, aObjCmd[i].xProc, 0, 0);