1#!/usr/bin/tclsh 2# 3# This script constructs the "sqlite3.h" header file from the following 4# sources: 5# 6# 1) The src/sqlite.h.in source file. This is the template for sqlite3.h. 7# 2) The VERSION file containing the current SQLite version number. 8# 3) The manifest file from the fossil SCM. This gives use the date. 9# 4) The manifest.uuid file from the fossil SCM. This gives the SHA1 hash. 10# 11# Run this script by specifying the root directory of the source tree 12# on the command-line. 13# 14# This script performs processing on src/sqlite.h.in. It: 15# 16# 1) Adds SQLITE_EXTERN in front of the declaration of global variables, 17# 2) Adds SQLITE_API in front of the declaration of API functions, 18# 3) Replaces the string --VERS-- with the current library version, 19# formatted as a string (e.g. "3.6.17"), and 20# 4) Replaces the string --VERSION-NUMBER-- with current library version, 21# formatted as an integer (e.g. "3006017"). 22# 5) Replaces the string --SOURCE-ID-- with the date and time and sha1 23# hash of the fossil-scm manifest for the source tree. 24# 6) Adds the SQLITE_CALLBACK calling convention macro in front of all 25# callback declarations. 26# 27# This script outputs to stdout. 28# 29# Example usage: 30# 31# tclsh mksqlite3h.tcl ../sqlite >sqlite3.h 32# 33 34 35# Get the source tree root directory from the command-line 36# 37set TOP [lindex $argv 0] 38 39# Enable use of SQLITE_APICALL macros at the right points? 40# 41set useapicall 0 42 43if {[lsearch -regexp [lrange $argv 1 end] {^-+useapicall}] != -1} { 44 set useapicall 1 45} 46 47# Get the SQLite version number (ex: 3.6.18) from the $TOP/VERSION file. 48# 49set in [open $TOP/VERSION] 50set zVersion [string trim [read $in]] 51close $in 52set nVersion [eval format "%d%03d%03d" [split $zVersion .]] 53 54# Get the source-id 55# 56set PWD [pwd] 57cd $TOP 58set zSourceId [exec $PWD/mksourceid manifest] 59cd $PWD 60 61# Set up patterns for recognizing API declarations. 62# 63set varpattern {^[a-zA-Z][a-zA-Z_0-9 *]+sqlite3_[_a-zA-Z0-9]+(\[|;| =)} 64set declpattern1 {^ *([a-zA-Z][a-zA-Z_0-9 ]+ \**)(sqlite3_[_a-zA-Z0-9]+)(\(.*)$} 65 66set declpattern2 \ 67 {^ *([a-zA-Z][a-zA-Z_0-9 ]+ \**)(sqlite3session_[_a-zA-Z0-9]+)(\(.*)$} 68 69set declpattern3 \ 70 {^ *([a-zA-Z][a-zA-Z_0-9 ]+ \**)(sqlite3changeset_[_a-zA-Z0-9]+)(\(.*)$} 71 72set declpattern4 \ 73 {^ *([a-zA-Z][a-zA-Z_0-9 ]+ \**)(sqlite3changegroup_[_a-zA-Z0-9]+)(\(.*)$} 74 75set declpattern5 \ 76 {^ *([a-zA-Z][a-zA-Z_0-9 ]+ \**)(sqlite3rebaser_[_a-zA-Z0-9]+)(\(.*)$} 77 78# Force the output to use unix line endings, even on Windows. 79fconfigure stdout -translation lf 80 81set filelist [subst { 82 $TOP/src/sqlite.h.in 83 $TOP/ext/rtree/sqlite3rtree.h 84 $TOP/ext/session/sqlite3session.h 85 $TOP/ext/fts5/fts5.h 86}] 87 88# These are the functions that accept a variable number of arguments. They 89# always need to use the "cdecl" calling convention even when another calling 90# convention (e.g. "stcall") is being used for the rest of the library. 91set cdecllist { 92 sqlite3_config 93 sqlite3_db_config 94 sqlite3_log 95 sqlite3_mprintf 96 sqlite3_snprintf 97 sqlite3_test_control 98 sqlite3_vtab_config 99} 100 101# Process the source files. 102# 103foreach file $filelist { 104 set in [open $file] 105 if {![regexp {sqlite\.h\.in} $file]} { 106 puts "/******** Begin file [file tail $file] *********/" 107 } 108 while {![eof $in]} { 109 110 set line [string trimright [gets $in]] 111 112 # File sqlite3rtree.h contains a line "#include <sqlite3.h>". Omit this 113 # line when copying sqlite3rtree.h into sqlite3.h. 114 # 115 if {[string match {*#include*[<"]sqlite3.h[>"]*} $line]} continue 116 117 regsub -- --VERS-- $line $zVersion line 118 regsub -- --VERSION-NUMBER-- $line $nVersion line 119 regsub -- --SOURCE-ID-- $line "$zSourceId" line 120 121 if {[regexp $varpattern $line] && ![regexp {^ *typedef} $line]} { 122 set line "SQLITE_API $line" 123 } else { 124 if {[regexp $declpattern1 $line all rettype funcname rest] || \ 125 [regexp $declpattern2 $line all rettype funcname rest] || \ 126 [regexp $declpattern3 $line all rettype funcname rest] || \ 127 [regexp $declpattern4 $line all rettype funcname rest] || \ 128 [regexp $declpattern5 $line all rettype funcname rest]} { 129 set line SQLITE_API 130 append line " " [string trim $rettype] 131 if {[string index $rettype end] ne "*"} { 132 append line " " 133 } 134 if {$useapicall} { 135 if {[lsearch -exact $cdecllist $funcname] >= 0} { 136 append line SQLITE_CDECL " " 137 } else { 138 append line SQLITE_APICALL " " 139 } 140 } 141 append line $funcname $rest 142 } 143 } 144 if {$useapicall} { 145 set line [string map [list (*sqlite3_syscall_ptr) \ 146 "(SQLITE_SYSAPI *sqlite3_syscall_ptr)"] $line] 147 regsub {\(\*} $line {(SQLITE_CALLBACK *} line 148 } 149 puts $line 150 } 151 close $in 152 if {![regexp {sqlite\.h\.in} $file]} { 153 puts "/******** End of [file tail $file] *********/" 154 } 155} 156