<!DOCTYPE html>
<html>
<head>
<title>configdef-doc.md</title>
<meta http-equiv="Content-type" content="text/html;charset=UTF-8">
<style>
/* https://github.com/microsoft/vscode/blob/master/extensions/markdown-language-features/media/markdown.css */
/*---------------------------------------------------------------------------------------------
* Copyright (c) Microsoft Corporation. All rights reserved.
* Licensed under the MIT License. See License.txt in the project root for license information.
*--------------------------------------------------------------------------------------------*/
body {
font-family: var(--vscode-markdown-font-family, -apple-system, BlinkMacSystemFont, "Segoe WPC", "Segoe UI", "Ubuntu", "Droid Sans", sans-serif);
font-size: var(--vscode-markdown-font-size, 14px);
padding: 0 26px;
line-height: var(--vscode-markdown-line-height, 22px);
word-wrap: break-word;
}
#code-csp-warning {
position: fixed;
top: 0;
right: 0;
color: white;
margin: 16px;
text-align: center;
font-size: 12px;
font-family: sans-serif;
background-color:#444444;
cursor: pointer;
padding: 6px;
box-shadow: 1px 1px 1px rgba(0,0,0,.25);
}
#code-csp-warning:hover {
text-decoration: none;
background-color:#007acc;
box-shadow: 2px 2px 2px rgba(0,0,0,.25);
}
body.scrollBeyondLastLine {
margin-bottom: calc(100vh - 22px);
}
body.showEditorSelection .code-line {
position: relative;
}
body.showEditorSelection .code-active-line:before,
body.showEditorSelection .code-line:hover:before {
content: "";
display: block;
position: absolute;
top: 0;
left: -12px;
height: 100%;
}
body.showEditorSelection li.code-active-line:before,
body.showEditorSelection li.code-line:hover:before {
left: -30px;
}
.vscode-light.showEditorSelection .code-active-line:before {
border-left: 3px solid rgba(0, 0, 0, 0.15);
}
.vscode-light.showEditorSelection .code-line:hover:before {
border-left: 3px solid rgba(0, 0, 0, 0.40);
}
.vscode-light.showEditorSelection .code-line .code-line:hover:before {
border-left: none;
}
.vscode-dark.showEditorSelection .code-active-line:before {
border-left: 3px solid rgba(255, 255, 255, 0.4);
}
.vscode-dark.showEditorSelection .code-line:hover:before {
border-left: 3px solid rgba(255, 255, 255, 0.60);
}
.vscode-dark.showEditorSelection .code-line .code-line:hover:before {
border-left: none;
}
.vscode-high-contrast.showEditorSelection .code-active-line:before {
border-left: 3px solid rgba(255, 160, 0, 0.7);
}
.vscode-high-contrast.showEditorSelection .code-line:hover:before {
border-left: 3px solid rgba(255, 160, 0, 1);
}
.vscode-high-contrast.showEditorSelection .code-line .code-line:hover:before {
border-left: none;
}
img {
max-width: 100%;
max-height: 100%;
}
a {
text-decoration: none;
}
a:hover {
text-decoration: underline;
}
a:focus,
input:focus,
select:focus,
textarea:focus {
outline: 1px solid -webkit-focus-ring-color;
outline-offset: -1px;
}
hr {
border: 0;
height: 2px;
border-bottom: 2px solid;
}
h1 {
padding-bottom: 0.3em;
line-height: 1.2;
border-bottom-width: 1px;
border-bottom-style: solid;
}
h1, h2, h3 {
font-weight: normal;
}
table {
border-collapse: collapse;
}
table > thead > tr > th {
text-align: left;
border-bottom: 1px solid;
}
table > thead > tr > th,
table > thead > tr > td,
table > tbody > tr > th,
table > tbody > tr > td {
padding: 5px 10px;
}
table > tbody > tr + tr > td {
border-top: 1px solid;
}
blockquote {
margin: 0 7px 0 5px;
padding: 0 16px 0 10px;
border-left-width: 5px;
border-left-style: solid;
}
code {
font-family: Menlo, Monaco, Consolas, "Droid Sans Mono", "Courier New", monospace, "Droid Sans Fallback";
font-size: 1em;
line-height: 1.357em;
}
body.wordWrap pre {
white-space: pre-wrap;
}
pre:not(.hljs),
pre.hljs code > div {
padding: 16px;
border-radius: 3px;
overflow: auto;
}
pre code {
color: var(--vscode-editor-foreground);
tab-size: 4;
}
/** Theming */
.vscode-light pre {
background-color: rgba(220, 220, 220, 0.4);
}
.vscode-dark pre {
background-color: rgba(10, 10, 10, 0.4);
}
.vscode-high-contrast pre {
background-color: rgb(0, 0, 0);
}
.vscode-high-contrast h1 {
border-color: rgb(0, 0, 0);
}
.vscode-light table > thead > tr > th {
border-color: rgba(0, 0, 0, 0.69);
}
.vscode-dark table > thead > tr > th {
border-color: rgba(255, 255, 255, 0.69);
}
.vscode-light h1,
.vscode-light hr,
.vscode-light table > tbody > tr + tr > td {
border-color: rgba(0, 0, 0, 0.18);
}
.vscode-dark h1,
.vscode-dark hr,
.vscode-dark table > tbody > tr + tr > td {
border-color: rgba(255, 255, 255, 0.18);
}
</style>
<style>
/* Tomorrow Theme */
/* http://jmblog.github.com/color-themes-for-google-code-highlightjs */
/* Original theme - https://github.com/chriskempson/tomorrow-theme */
/* Tomorrow Comment */
.hljs-comment,
.hljs-quote {
color: #8e908c;
}
/* Tomorrow Red */
.hljs-variable,
.hljs-template-variable,
.hljs-tag,
.hljs-name,
.hljs-selector-id,
.hljs-selector-class,
.hljs-regexp,
.hljs-deletion {
color: #c82829;
}
/* Tomorrow Orange */
.hljs-number,
.hljs-built_in,
.hljs-builtin-name,
.hljs-literal,
.hljs-type,
.hljs-params,
.hljs-meta,
.hljs-link {
color: #f5871f;
}
/* Tomorrow Yellow */
.hljs-attribute {
color: #eab700;
}
/* Tomorrow Green */
.hljs-string,
.hljs-symbol,
.hljs-bullet,
.hljs-addition {
color: #718c00;
}
/* Tomorrow Blue */
.hljs-title,
.hljs-section {
color: #4271ae;
}
/* Tomorrow Purple */
.hljs-keyword,
.hljs-selector-tag {
color: #8959a8;
}
.hljs {
display: block;
overflow-x: auto;
color: #4d4d4c;
padding: 0.5em;
}
.hljs-emphasis {
font-style: italic;
}
.hljs-strong {
font-weight: bold;
}
</style>
<style>
/*
* Markdown PDF CSS
*/
body {
font-family: -apple-system, BlinkMacSystemFont, "Segoe WPC", "Segoe UI", "Ubuntu", "Droid Sans", sans-serif, "Meiryo";
padding: 0 12px;
}
pre {
background-color: #f8f8f8;
border: 1px solid #cccccc;
border-radius: 3px;
overflow-x: auto;
white-space: pre-wrap;
overflow-wrap: break-word;
}
pre:not(.hljs) {
padding: 23px;
line-height: 19px;
}
blockquote {
background: rgba(127, 127, 127, 0.1);
border-color: rgba(0, 122, 204, 0.5);
}
.emoji {
height: 1.4em;
}
code {
font-size: 14px;
line-height: 19px;
}
/* for inline code */
:not(pre):not(.hljs) > code {
color: #C9AE75; /* Change the old color so it seems less like an error */
font-size: inherit;
}
/* Page Break : use <div class="page"/> to insert page break
-------------------------------------------------------- */
.page {
page-break-after: always;
}
</style>
<script src="https://unpkg.com/mermaid/dist/mermaid.min.js"></script>
</head>
<body>
<script>
mermaid.initialize({
startOnLoad: true,
theme: document.body.classList.contains('vscode-dark') || document.body.classList.contains('vscode-high-contrast')
? 'dark'
: 'default'
});
</script>
<!-- configdef-doc.md rev1.0.1 2023-06-17 -->
<h1 id="configdef"><strong>configdef</strong></h1>
<h2 id="intro"><strong>Intro</strong></h2>
<p><em><strong>configdef</strong></em> is a configuration utility created for Tcl developers, Tcl extension writers in particular. While <em>configdef</em> may also be useful in non-Tcl projects, at this point it doesn't provide the comprehensive capabilities of Gnu configure or cmake. <em>configdef</em> does offer simplicity, flexibility and works on any system that can run an up-to-date Tcl interpreter (which is virtually all of them).</p>
<h2 id="system-requirements"><strong>System requirements</strong></h2>
<p>The <em>configdef</em> archive contains ready-to-run binaries for Windows and Linux systems.</p>
<p>For other OS, or when custom features are needed, binaries can be regenerated. For such purposes it's necessary to have a statically compiled Tcl 8.7 or 9.0. This will be described in detail in a later section.</p>
<h2 id="installation"><strong>Installation</strong></h2>
<p>Decompress the downloaded archive in any convenient directory. Archives contain the following file structure:</p>
<pre><code>+ configdef
+ bin
- config
- config.exe
- genPkgIndex
- genPkgIndex.exe
+ doc
- configdef-doc.md
- configdef-doc.pdf
+ generic
- config.tcl
- genPkgIndex.tcl
- mkconfig.tcl
- mkGenPkgIndex.tcl
- tcltkparams.tcl
+ tools
+ config8.vfs
+ config9.vfs
+ genPkgIndex8.vfs
+ genPkgIndex9.vfs
+ unix
+ win
</code></pre>
<p>It's only necessary to copy config and genPkgIndex (or the *.exe files) to a directory on the exec path. For unix users that's typically /usr/local/bin. Windows users might want to create a "bin" directory in their home directory and add it to their personal or system-wide path.</p>
<h2 id="using-configdef"><strong>Using <em>configdef</em></strong></h2>
<h3 id="the-configdef-file"><strong>The <em>config.def</em> file</strong></h3>
<p>The <em>config.def</em> file describes and defines configuration options that are transformed into a usable <em>Makefile</em>. The syntax of this Makefile is compatible with any recent version of Gnu make.</p>
<p><em>config.def</em> is intended to be processed by the <em>config</em> utility on a <em>per directory basis</em>. IOW every directory with configurable options will have its own <em>config.def</em> file. It's possible to cascade configuration such that <em>config.def</em> in a directory sets up or points to <em>config.def</em> in one or more subdirectories.</p>
<h4 id="configdef-syntax"><strong><em>config.def</em> syntax</strong></h4>
<p>Note that <em>config.def</em> is a normal Tcl file and can contain additional <em>procs</em>, global variables or <em>source</em> other Tcl files as needed. The basic components are described in this section.</p>
<p><strong>Vars</strong> variable contains a list of undefined variables that must be defined on the <em>config</em> command line. For example:</p>
<pre><code>set Vars {filename}
config -filename=myfile.tcl
# if filename not supplied:
ERROR: missing input: -filename=...
</code></pre>
<p><strong>Defaults</strong> variable has a list of variable+default value pairs. Variables listed here don't have to be supplied on the <em>config</em> command line except to override the default:</p>
<pre><code>set Defaults {
tcldir c:/tcl87b1s
...
}
# global variable "tcldir" is visible in config.def
# This invocation overrides the default value:
config -tcldir=c:/tcl90b1s
</code></pre>
<p><strong>helptext</strong>. Optional list of strings providing help for <em>mandatory</em> arguments to <em>config</em>. (As described above re: <strong>Vars</strong>.) These strings are displayed when <code>config -help</code> is called.</p>
<p><strong>helptext_optional</strong>. Optional list of strings with help text for optional params to <em>config</em>. Optional params correspond to variables given under <strong>Defaults</strong>.</p>
<p><strong>DEFINES</strong>. The DEFINES variable contains a list of variable/value pairs. Here the first member of the pair becomes a Makefile variable and the second is the value.</p>
<pre><code>set DEFINES {
tclinc $tclinc
tcllib $tcllib
...
}
# in the Makefile:
TCLINC = c:/tcl87b1/include
TCLLIB = c:/tcl87b1/lib
...
</code></pre>
<p>When there are several Makefile variables to initialize it's convenient to use this special syntax:</p>
<pre><code>set DEFINES {
* {tclinc tcllib tclsrc ...}
...
}
</code></pre>
<p>The output is identical to the prior example.</p>
<p><em>configdef</em> provides a large number of Tcl-related variables automatically when "tcldir" or "tclsrc" is specified in <strong>Defaults</strong> or on the <em>config</em> command line. ("tcldir" is the directory where Tcl is installed, "tclsrc" is the Tcl source directory. Similarly, Tk variables are generated when "tkdir" and "tksrc" are provided.)</p>
<p>DEFINES don't have to be predefined in Defaults. It's quite possible and often useful to create values for arbitrary variables. DEFINES values can make use of the full range of Tcl facilities:</p>
<pre><code>set DEFINES {
...
config_vfs {../config[string index $::tclver 0].vfs}
progs {\$(CONFIG) \$(PKGNDX)}
...
}
# Makefile:
CONFIG_VFS = ../config8.vfs
PROGS = $(CONFIG) $(PKGNDX)
</code></pre>
<p>Note that a value must be enclosed in {} when it contains spaces, Tcl variables or commands. Also "$" must be escaped with "\" when the $ refers to Makefile and not Tcl variables. Furthermore, values defined in <strong>Defaults</strong> or command line and used directly need not be namespace-qualified (":<img class="emoji" alt="blush" src="" />. However when used in a command in the value (right-hand side) then the global namespace qualifiers are necessary.</p>
<p>The <strong>RULES</strong> variable is slightly more complex. This variable defines Makefile rules, with each rule consisting of 3 parts: the target, list of dependencies, and a list of operations. <strong>RULES</strong> looks like this:</p>
<div style="page-break-after: always;"></div>
<pre><code>set RULES {
all {$(PROGS)} {
}
cfginit {} {
{rm -rf $(CONFIG_VFSLIB)}
{cp -r $(TCLLBRY) $(CONFIG_VFSLIB)}
{cp $(GEN)/config.tcl $(GEN)/tcltkparams.tcl $(CONFIG_VFS)}
}
$(CONFIG) {$(CONFIG_VFS)/config.tcl \
$(CONFIG_VFS)/tcltkparams.tcl} {
{make cfginit}
{$(TCLSH) $(GEN)/mkconfig.tcl}
}
...
}
# Makefile:
all: $(PROGS)
cfginit:
rm -rf $(CONFIG_VFSLIB)
cp -r $(TCLLBRY) $(CONFIG_VFSLIB)
cp $(GEN)/config.tcl $(GEN)/tcltkparams.tcl $(CONFIG_VFS)
$(CONFIG): $(CONFIG_VFS)/config.tcl $(CONFIG_VFS)/tcltkparams.tcl
make cfginit
$(TCLSH) $(GEN)/mkconfig.tcl
</code></pre>
<p>Note that the rules list is evaluated but the target and dependencies are not. Because Makefile variables are common and essential in rules, $... is not treated as a Tcl variable during <em>configdef</em> evaluation, except when enclosed in a Tcl command invocation:</p>
<pre><code>$(MYTARGET) {<dependency list>} {
{[expr {$::my_tcl_variable ? <do something> : <do something else>}]}
{$(CC) -o $@ -c $< ...}
}
# <$my_tcl_variable> is evaluated as expected, but $(CC), etc., are ignored
</code></pre>
<h4 id="cascading-configdef"><strong>Cascading <em>config.def</em></strong></h4>
<p>As noted above, configuration can proceed from one directory to another. All that's necessary is defining a <em>proc</em> named "nextcfg" that takes no arguments. The proc returns a list of directories/config arguments. Each directory will be visited in turn and config arguments applied.</p>
<pre><code>proc nextcfg {} {
lappend ls ../subdirA [list -tcldir=$::tcldir] \
../subdirB [list -tcllib=$::tcllib]
return $ls
}
</code></pre>
<p>nextcfg can make use of Defaults, procs, and other Tcl features defined in config.def, assuming that the section was already processed. For this reason, nextcfg is typically placed at the bottom of the config.def file.</p>
<h3 id="genpkgindex"><strong>genPkgIndex</strong></h3>
<p><em>configdef</em> includes the <strong>genPkgIndex</strong> utility which unsurprisingly generates a <em>pkgIndex.tcl</em> file for binary extensions.</p>
<p>genPkgIndex takes 2 mandatory and 2 optional arguments. The syntax is as follows:</p>
<pre><code>genPkgIndex(.exe) -pkgnm <package name> -pkgver <pkg version> ?-script <string>? ?-filename <output file>?
</code></pre>
<p>If <filename> is not given then output goes to stdout. If <script> is absent then the built-in standard pkgIndex.tcl is generated:</p>
<pre><code>genPkgIndex -pkgnm mypkg -pkgver 1.1
# writes to stdout:
# if {[package vsatisfies [package provide Tcl] 9.0-]} {
# package ifneeded mypkg 1.0 [list load [file join $dir tcl9mypkg10.dll]]
# } else {
# package ifneeded mypkg 1.0 [list load [file join $ dir mypkg10.dll]]
# }
</code></pre>
<p>Of course, a script can be provided. The script should have embedded $pkgnm, $pkgver, $tcl9libnm and $libnm variables at the appropriate places. Otherwise $, [, ] characters must be backslash escaped.</p>
<h3 id="generating-binaries-from-source"><strong>Generating binaries from source</strong></h3>
<p>Complete source code is included in the release archive. Generating binaries for supported platforms is pretty straightforward. The only required dependency is a statically compiled Tcl 8.7 or 9.0 <em>tclsh</em>. Consult the Tcl documentation for details of compiling Tcl.</p>
<p>If <em>configdef</em> sources have been modified, very likely the default <em>config</em> and <em>genPkgIndex</em> are no longer valid. To generate these files follow these steps:</p>
<ol>
<li>Locate the appropriate directory, unix or win under the tools directory.</li>
<li>Modify config.def to match the user's platform. The main options are in Defaults re: "tcldir" and "tclsrc" which should point to valid locations. (Note that "tcldir" and "tclsrc" are usable after successful compilation and installation of Tcl 8.7 or 9.0.)</li>
<li>Make sure config8.vfs or config9.vfs is up to date and has the appropriate tcl_library (same Tcl version as intended to use in the next step).</li>
<li>Running "/path/to/tclsh* ../../generic/mkconfig.tcl" should generate ./config(.exe) in tools/unix(or win) directory.</li>
<li>Now running "./config(.exe)" should generate a new (and correct) Makefile.</li>
<li>Run "make clean", "make" and "make install".</li>
<li>It's only necessary to use one of Tcl 8.7 or 9.0 as either will produce valid, functionally equivalent <em>config</em> and <em>genPkgIndex</em> programs.</li>
</ol>
<h2 id="built-in-variables"><strong>Built-in variables</strong></h2>
<p>As a convenience to users <em>configdef</em> supplies a number of Tcl global variables. These can be redefined in <strong>Defaults</strong> or <strong>DEFINES</strong> lists.</p>
<h3 id="makefile-defines"><strong>Makefile DEFINES</strong></h3>
<p>Several variables are computed by <em>configdef</em> that automatically appear in the Makefile.</p>
<table>
<thead>
<tr>
<th>variable</th>
<th>Makefile</th>
<th>value</th>
<th>(info)</th>
</tr>
</thead>
<tbody>
<tr>
<td>cc</td>
<td>CC</td>
<td>gcc</td>
<td></td>
</tr>
<tr>
<td>libext</td>
<td>LIBEXT</td>
<td>dll, so</td>
<td></td>
</tr>
<tr>
<td>objext</td>
<td>OBJEXT</td>
<td>obj, o</td>
<td></td>
</tr>
<tr>
<td>exe</td>
<td>EXE</td>
<td>.exe, ""</td>
<td></td>
</tr>
<tr>
<td>starext</td>
<td>STAREXT</td>
<td>.lib, .a</td>
<td>(static archive extension)</td>
</tr>
</tbody>
</table>
<h3 id="tcltk-specific-variables"><strong>Tcl/Tk-specific variables</strong></h3>
<p>Establishing certain variables in <strong>Defaults</strong> or on the command line leads to automatically produced variables.</p>
<p>(Defaults) <code>tcldir /path/to/installed_Tcl</code><br>
(cmdline) <code>-tcldir=/path/to/installed_Tcl</code></p>
<table>
<thead>
<tr>
<th>variable</th>
<th>value</th>
</tr>
</thead>
<tbody>
<tr>
<td>tclbin</td>
<td>$tcldir/bin</td>
</tr>
<tr>
<td>tclsh</td>
<td>$tcldir/bin/tclsh*</td>
</tr>
<tr>
<td>tclinc</td>
<td>$tcldir/include</td>
</tr>
<tr>
<td>tcllib</td>
<td>$tcldir/lib</td>
</tr>
<tr>
<td>tclstub</td>
<td>$tcldir/lib/tcl*stub*</td>
</tr>
<tr>
<td>tclver</td>
<td>tcl version</td>
</tr>
</tbody>
</table>
<p>(Defaults) <code>tclsrc /path/to/Tcl_source_dir</code><br>
(cmdline) <code>-tclsrc=/path/to/Tcl_source_dir</code></p>
<table>
<thead>
<tr>
<th>variable</th>
<th>value</th>
<th>info</th>
</tr>
</thead>
<tbody>
<tr>
<td>tcllbry</td>
<td>$tclsrc/library</td>
<td></td>
</tr>
<tr>
<td>tclobjd</td>
<td>$tclsrc/obj_dir</td>
<td>dir containing object files</td>
</tr>
<tr>
<td>tclshsd</td>
<td>$tclsrc/tclsh*</td>
<td>tclsh (in source directory)</td>
</tr>
<tr>
<td>tclstubsd</td>
<td>$tclsrc/tcl*stub*</td>
<td>stub (in source directory)</td>
</tr>
<tr>
<td>tclgen</td>
<td>$tclsrc/generic</td>
<td></td>
</tr>
<tr>
<td>tclver</td>
<td>tcl version</td>
<td></td>
</tr>
</tbody>
</table>
<p>(Defaults) <code>tkdir /path/to/installed_Tk</code><br>
(cmdline) <code>-tkdir=/path/to/installed_Tk</code></p>
<table>
<thead>
<tr>
<th>variable</th>
<th>value</th>
<th>info</th>
</tr>
</thead>
<tbody>
<tr>
<td>tkbin</td>
<td>$tkdir/bin</td>
<td></td>
</tr>
<tr>
<td>tksh</td>
<td>$tkdir/bin/wish*</td>
<td></td>
</tr>
<tr>
<td>tkinc</td>
<td>$tkdir/include</td>
<td></td>
</tr>
<tr>
<td>tklib</td>
<td>$tkdir/lib</td>
<td></td>
</tr>
<tr>
<td>tkstub</td>
<td>$tkdir/lib/tk*stub*</td>
<td></td>
</tr>
<tr>
<td>tkver</td>
<td>Tk version</td>
<td></td>
</tr>
</tbody>
</table>
<p>(Defaults) <code>tclsrc /path/to/Tk_source_dir</code><br>
(cmdline) <code>-tclsrc=/path/to/Tk_source_dir</code></p>
<table>
<thead>
<tr>
<th>variable</th>
<th>value</th>
<th>info</th>
</tr>
</thead>
<tbody>
<tr>
<td>tklbry</td>
<td>$tksrc/library</td>
<td></td>
</tr>
<tr>
<td>tkobjd</td>
<td>$tksrc/obj_dir</td>
<td>dir containing Tk object files</td>
</tr>
<tr>
<td>tkwishsd</td>
<td>$tksrc/wish*</td>
<td></td>
</tr>
<tr>
<td>tkstubsd</td>
<td>$tksrc/tk*stub*</td>
<td></td>
</tr>
<tr>
<td>tkgen</td>
<td>$tksrc/generic</td>
<td></td>
</tr>
<tr>
<td>tkver</td>
<td>Tk version</td>
<td></td>
</tr>
</tbody>
</table>
</body>
</html>