From d3080efe1ea6393f72f4b3a01c6cda6843b72662 Mon Sep 17 00:00:00 2001 From: oehhar Date: Mon, 16 Dec 2024 11:50:32 +0000 Subject: [PATCH 1/7] Ticket [70f3b23cad]: propose new doc wording for requirement strings --- doc/InitStubs.3 | 10 +++-- doc/package.n | 117 +++++++++++++++++++++++++----------------------- 2 files changed, 67 insertions(+), 60 deletions(-) diff --git a/doc/InitStubs.3 b/doc/InitStubs.3 index 80a21de8ab3..9ac4f24515e 100644 --- a/doc/InitStubs.3 +++ b/doc/InitStubs.3 @@ -14,21 +14,23 @@ Tcl_InitStubs \- initialize the Tcl stubs mechanism \fB#include \fR .sp const char * -\fBTcl_InitStubs\fR(\fIinterp, version, exact\fR) +\fBTcl_InitStubs\fR(\fIinterp, requirements, exact\fR) .fi .SH ARGUMENTS .AS Tcl_Interp *interp .AP Tcl_Interp *interp in Tcl interpreter handle. -.AP "const char" *version in -A version string consisting of one or more decimal numbers -separated by dots. +.AP "const char" *requirements in +The requirement string of the compatible Tcl versions. +It consists of a space separated list of requirements as described on the PACKAGE manual page in the section REQUIREMENT. .AP int exact in 1 means that only the particular version specified by \fIversion\fR is acceptable. +An exact version number must be given as \fIrequirement\fR. 0 means that versions newer than \fIversion\fR are also acceptable as long as they have the same major version number as \fIversion\fR. Other bits have no effect. +See the \fB-require\fR option of the \fBpackage require\fR command. .BE .SH INTRODUCTION .PP diff --git a/doc/package.n b/doc/package.n index d27a44a3868..9f67e9aaf61 100644 --- a/doc/package.n +++ b/doc/package.n @@ -205,62 +205,7 @@ commands. \fBpackage vsatisfies \fIversion requirement...\fR . Returns 1 if the \fIversion\fR satisfies at least one of the given -requirements, and 0 otherwise. Each \fIrequirement\fR is allowed to -have any of the forms: -.RS -.IP \fImin\fR -This form is called -.QW min-bounded . -.IP \fImin\fB\-\fR -This form is called -.QW min-unbound . -.IP \fImin\fB\-\fImax\fR -This form is called -.QW bounded . -.PP -where -.QW \fImin\fR -and -.QW \fImax\fR -are valid version numbers. The legacy syntax is -a special case of the extended syntax, keeping backward -compatibility. Regarding satisfaction the rules are: -.IP [1] -The \fIversion\fR has to pass at least one of the listed -\fIrequirement\fRs to be satisfactory. -.IP [2] -A version satisfies a -.QW bounded -requirement when -.RS -.IP [a] -For \fImin\fR equal to the \fImax\fR if, and only if the \fIversion\fR -is equal to the \fImin\fR. -.IP [b] -Otherwise if, and only if the \fIversion\fR is greater than or equal -to the \fImin\fR, and less than the \fImax\fR, where both \fImin\fR -and \fImax\fR have been padded internally with -.QW a0 . -Note that while the comparison to \fImin\fR is inclusive, the -comparison to \fImax\fR is exclusive. -.RE -.IP [3] -A -.QW min-bounded -requirement is a -.QW bounded -requirement in disguise, -with the \fImax\fR part implicitly specified as the next higher major -version number of the \fImin\fR part. A version satisfies it per the -rules above. -.IP [4] -A \fIversion\fR satisfies a -.QW min-unbound -requirement if, and only if it is greater than or equal to the -\fImin\fR, where the \fImin\fR has been padded internally with -.QW a0 . -There is no constraint to a maximum. -.RE +requirements, and 0 otherwise. \fIrequirements\fR are defined in the REQUIREMENTS section below. .\" METHOD: prefer .TP \fBpackage prefer \fR?\fBlatest\fR|\fBstable\fR? @@ -350,6 +295,66 @@ and \fBpackage provide\fR commands in scripts, and use the procedure Once you have done this, packages will be loaded automatically in response to \fBpackage require\fR commands. See the documentation for \fBpkg_mkIndex\fR for details. +.SH "REQUIREMENT" +.PP +A \fIrequirement\fR string checks, if a compatible version number of a package is present. +Most commands accept a list of requirements where the highest suitable version is matched. +.PP +Each \fIrequirement\fR is allowed to have any of the forms: +.RS +.IP \fImin\fR +This form is called +.QW min-bounded . +.IP \fImin\fB\-\fR +This form is called +.QW min-unbound . +.IP \fImin\fB\-\fImax\fR +This form is called +.QW bounded . +.PP +where +.QW \fImin\fR +and +.QW \fImax\fR +are valid version numbers. The legacy syntax is +a special case of the extended syntax, keeping backward +compatibility. Regarding satisfaction the rules are: +.IP [1] +The \fIversion\fR has to pass at least one of the listed +\fIrequirement\fRs to be satisfactory. +.IP [2] +A version satisfies a +.QW bounded +requirement when +.RS +.IP [a] +For \fImin\fR equal to the \fImax\fR if, and only if the \fIversion\fR +is equal to the \fImin\fR. +.IP [b] +Otherwise if, and only if the \fIversion\fR is greater than or equal +to the \fImin\fR, and less than the \fImax\fR, where both \fImin\fR +and \fImax\fR have been padded internally with +.QW a0 . +Note that while the comparison to \fImin\fR is inclusive, the +comparison to \fImax\fR is exclusive. +.RE +.IP [3] +A +.QW min-bounded +requirement is a +.QW bounded +requirement in disguise, +with the \fImax\fR part implicitly specified as the next higher major +version number of the \fImin\fR part. A version satisfies it per the +rules above. +.IP [4] +A \fIversion\fR satisfies a +.QW min-unbound +requirement if, and only if it is greater than or equal to the +\fImin\fR, where the \fImin\fR has been padded internally with +.QW a0 . +There is no constraint to a maximum. +.RE .SH EXAMPLES .PP To state that a Tcl script requires the Tk and http packages, put this From 4e6acfdfcd15415430f992cf3e77c0d389ed2dd1 Mon Sep 17 00:00:00 2001 From: oehhar Date: Mon, 16 Dec 2024 14:16:12 +0000 Subject: [PATCH 2/7] Add Tcl_InitStubs macro information. Update man page links. --- doc/InitStubs.3 | 14 ++++++++++---- doc/package.n | 8 ++++---- 2 files changed, 14 insertions(+), 8 deletions(-) diff --git a/doc/InitStubs.3 b/doc/InitStubs.3 index 9ac4f24515e..d6edc8797bb 100644 --- a/doc/InitStubs.3 +++ b/doc/InitStubs.3 @@ -14,15 +14,16 @@ Tcl_InitStubs \- initialize the Tcl stubs mechanism \fB#include \fR .sp const char * -\fBTcl_InitStubs\fR(\fIinterp, requirements, exact\fR) +\fBTcl_InitStubs\fR(\fIinterp, requirement, exact\fR) .fi .SH ARGUMENTS .AS Tcl_Interp *interp .AP Tcl_Interp *interp in Tcl interpreter handle. -.AP "const char" *requirements in -The requirement string of the compatible Tcl versions. -It consists of a space separated list of requirements as described on the PACKAGE manual page in the section REQUIREMENT. +.AP "const char" *requirement in +The requirement string of compatible Tcl versions. +It consists of one requirement as described for \fBpackage require\fR (\fBPACKAGE\fR manual page in the section \fBREQUIREMENT\fR). +Multiple requirement strings like with \fBpackage require\fR are not supported. .AP int exact in 1 means that only the particular version specified by \fIversion\fR is acceptable. @@ -86,7 +87,12 @@ non-zero means that only the specified \fIversion\fR is acceptable. \fBTcl_InitStubs\fR returns a string containing the actual version of Tcl satisfying the request, or NULL if the Tcl version is not acceptable, does not support stubs, or any other error condition occurred. +.PP +\fBTcl_InitStubs\fR is implemented as a macro. +If \fBUSE_TCL_STUBS\fR is not defined, it translates to a TCL version verification only. +Thus, the call may also be used if the TCL library is directly linked. .SH "SEE ALSO" Tk_InitStubs +package .SH KEYWORDS stubs diff --git a/doc/package.n b/doc/package.n index 9f67e9aaf61..ba95c316df2 100644 --- a/doc/package.n +++ b/doc/package.n @@ -123,8 +123,8 @@ loaded; otherwise it generates an error. .RS .PP A suitable version of the package is any version which satisfies at -least one of the requirements, per the rules of \fBpackage -vsatisfies\fR. If multiple versions are suitable the implementation +least one of the requirements as defined in the section \fBREQUIREMENT\fR below. +If multiple versions are suitable the implementation with the highest version is chosen. This last part is additionally influenced by the selection mode set with \fBpackage prefer\fR. .PP @@ -205,7 +205,7 @@ commands. \fBpackage vsatisfies \fIversion requirement...\fR . Returns 1 if the \fIversion\fR satisfies at least one of the given -requirements, and 0 otherwise. \fIrequirements\fR are defined in the REQUIREMENTS section below. +requirements, and 0 otherwise. \fIrequirements\fR are defined in the \fBREQUIREMENT\fR section below. .\" METHOD: prefer .TP \fBpackage prefer \fR?\fBlatest\fR|\fBstable\fR? @@ -298,7 +298,7 @@ See the documentation for \fBpkg_mkIndex\fR for details. .SH "REQUIREMENT" .PP A \fIrequirement\fR string checks, if a compatible version number of a package is present. -Most commands accept a list of requirements where the highest suitable version is matched. +Most commands accept a list of requirement strings where the highest suitable version is matched. .PP Each \fIrequirement\fR is allowed to have any of the forms: .RS From e20406d838b6c93686dc974560705e61076d2ee7 Mon Sep 17 00:00:00 2001 From: "jan.nijtmans" Date: Fri, 3 Jan 2025 14:48:06 +0000 Subject: [PATCH 3/7] Document underlying Tcl_InitStubs() behavior, when circumventing the macro in tcl.h --- doc/InitStubs.3 | 12 +++++++++++- 1 file changed, 11 insertions(+), 1 deletion(-) diff --git a/doc/InitStubs.3 b/doc/InitStubs.3 index d6edc8797bb..62b53f52982 100644 --- a/doc/InitStubs.3 +++ b/doc/InitStubs.3 @@ -83,12 +83,22 @@ and a Boolean flag indicating whether the extension requires an exact version match or not. If \fIexact\fR is 0, then the extension is indicating that newer versions of Tcl are acceptable as long as they have the same major version number as \fIversion\fR; -non-zero means that only the specified \fIversion\fR is acceptable. +1 means that only the specified \fIversion\fR is acceptable. \fBTcl_InitStubs\fR returns a string containing the actual version of Tcl satisfying the request, or NULL if the Tcl version is not acceptable, does not support stubs, or any other error condition occurred. .PP \fBTcl_InitStubs\fR is implemented as a macro. +If \fBUSE_TCL_STUBS\fR is defined, this macro calls the real \fBTcl_InitStubs\fR +function which has an additional \fBmagic\fR integer argument and uses the \fBexact\fR +parameter to know the exact Tcl major version number. If you circumvent +the macro in tcl.h, it should be called like this: +.CS + if (Tcl_InitStubs(interp, "8.6-", TCL_MAJOR_VERSION<<8, TCL_STUB_MAGIC)) == NULL) { + return TCL_ERROR; + } +} +.CE If \fBUSE_TCL_STUBS\fR is not defined, it translates to a TCL version verification only. Thus, the call may also be used if the TCL library is directly linked. .SH "SEE ALSO" From 997d70e625ef35643b5632c3d22d15ea357e88e2 Mon Sep 17 00:00:00 2001 From: "jan.nijtmans" Date: Sun, 5 Jan 2025 20:38:40 +0000 Subject: [PATCH 4/7] Simplification for [d58e315172] --- generic/tclBasic.c | 4 +- win/configure | 32 +++++------ win/configure.ac | 27 ++++------ win/tclWin32Dll.c | 131 ++------------------------------------------- 4 files changed, 30 insertions(+), 164 deletions(-) diff --git a/generic/tclBasic.c b/generic/tclBasic.c index 622651af9fb..c0e65577fda 100644 --- a/generic/tclBasic.c +++ b/generic/tclBasic.c @@ -68,7 +68,7 @@ * compatible with AddressSanitizer (ASan) use-after-return detection. */ -#if defined(_MSC_VER) && defined(HAVE_INTRIN_H) +#if defined(_MSC_VER) #include /* for _AddressOfReturnAddress() */ #endif @@ -85,7 +85,7 @@ TclGetCStackPtr(void) { #if defined( __GNUC__ ) || __has_builtin(__builtin_frame_address) return __builtin_frame_address(0); -#elif defined(_MSC_VER) && defined(HAVE_INTRIN_H) +#elif defined(_MSC_VER) return _AddressOfReturnAddress(); #else ptrdiff_t unused = 0; diff --git a/win/configure b/win/configure index 801ff648117..10b63648e09 100755 --- a/win/configure +++ b/win/configure @@ -5361,50 +5361,44 @@ fi # See if the compiler supports cpuid header. -if test "${GCC}" = "yes" ; then - { printf "%s\n" "$as_me:${as_lineno-$LINENO}: checking for cpuid-header support in compiler" >&5 -printf %s "checking for cpuid-header support in compiler... " >&6; } -if test ${tcl_cv_cpuidhead+y} +{ printf "%s\n" "$as_me:${as_lineno-$LINENO}: checking for cpuid.h" >&5 +printf %s "checking for cpuid.h... " >&6; } +if test ${tcl_cv_cpuid_h+y} then : printf %s "(cached) " >&6 else case e in #( e) cat confdefs.h - <<_ACEOF >conftest.$ac_ext /* end confdefs.h. */ - #if defined (__clang__) || ((__GNUC__) && ((__GNUC__ > 4) || ((__GNUC__ == 4) && (__GNUC_MINOR__ >= 8)))) - # include - #endif +#include int main (void) { - unsigned int regs; - __get_cpuid(0, ®s, ®s, ®s, ®s); + __get_cpuid(0, 0, 0, 0, 0); ; return 0; } _ACEOF -if ac_fn_c_try_link "$LINENO" +if ac_fn_c_try_compile "$LINENO" then : - tcl_cv_cpuidhead=yes + tcl_cv_cpuid_h=yes else case e in #( - e) tcl_cv_cpuidhead=no ;; + e) tcl_cv_cpuid_h=no ;; esac fi -rm -f core conftest.err conftest.$ac_objext conftest.beam \ - conftest$ac_exeext conftest.$ac_ext - ;; +rm -f core conftest.err conftest.$ac_objext conftest.beam conftest.$ac_ext + ;; esac fi -{ printf "%s\n" "$as_me:${as_lineno-$LINENO}: result: $tcl_cv_cpuidhead" >&5 -printf "%s\n" "$tcl_cv_cpuidhead" >&6; } - if test "$tcl_cv_cpuidhead" = "yes"; then +{ printf "%s\n" "$as_me:${as_lineno-$LINENO}: result: $tcl_cv_cpuid_h" >&5 +printf "%s\n" "$tcl_cv_cpuid_h" >&6; } +if test "$tcl_cv_cpuid_h" = "yes"; then printf "%s\n" "#define HAVE_CPUID_H 1" >>confdefs.h - fi fi # See if the header file is present diff --git a/win/configure.ac b/win/configure.ac index 630d204d8a2..b7ede2076f6 100644 --- a/win/configure.ac +++ b/win/configure.ac @@ -257,24 +257,19 @@ fi # See if the compiler supports cpuid header. -if test "${GCC}" = "yes" ; then - AC_CACHE_CHECK(for cpuid-header support in compiler, - tcl_cv_cpuidhead, - AC_LINK_IFELSE([AC_LANG_PROGRAM([[ - #if defined (__clang__) || ((__GNUC__) && ((__GNUC__ > 4) || ((__GNUC__ == 4) && (__GNUC_MINOR__ >= 8)))) - # include - #endif - ]], [[ - unsigned int regs; - __get_cpuid(0, ®s, ®s, ®s, ®s); +AC_CACHE_CHECK(for cpuid.h, + tcl_cv_cpuid_h, +AC_COMPILE_IFELSE([AC_LANG_PROGRAM([[ +#include +]], [[ + __get_cpuid(0, 0, 0, 0, 0); ]])], - [tcl_cv_cpuidhead=yes], - [tcl_cv_cpuidhead=no]) - ) - if test "$tcl_cv_cpuidhead" = "yes"; then + [tcl_cv_cpuid_h=yes], + [tcl_cv_cpuid_h=no]) +) +if test "$tcl_cv_cpuid_h" = "yes"; then AC_DEFINE(HAVE_CPUID_H, 1, - [Defined when the compilers supports cpuid header]) - fi + [Defined when cpuid.h exists]) fi # See if the header file is present diff --git a/win/tclWin32Dll.c b/win/tclWin32Dll.c index 12dfab9c3d8..81db0f2502d 100644 --- a/win/tclWin32Dll.c +++ b/win/tclWin32Dll.c @@ -13,8 +13,8 @@ #include "tclWinInt.h" #if defined(HAVE_CPUID_H) -# include -#elif defined(HAVE_INTRIN_H) +# xinclude +#elif defined(_MSC_VER) # include #endif @@ -445,134 +445,12 @@ TclWinCPUID( __get_cpuid(index, ®s[0], ®s[1], ®s[2], ®s[3]); status = TCL_OK; -#elif defined(HAVE_INTRIN_H) && defined(_WIN64) && defined(HAVE_CPUID) +#elif defined(_MSC_VER) && defined(_WIN64) && defined(HAVE_CPUID) __cpuid((int *)regsPtr, index); status = TCL_OK; -#elif defined(__GNUC__) && defined(HAVE_CPUID) -# if defined(_WIN64) - /* - * Execute the CPUID instruction with the given index, and store results - * off 'regPtr'. - */ - - __asm__ __volatile__( - /* - * Do the CPUID instruction, and save the results in the 'regsPtr' - * area. - */ - - "movl %[rptr], %%edi" "\n\t" - "movl %[index], %%eax" "\n\t" - "cpuid" "\n\t" - "movl %%eax, 0x0(%%edi)" "\n\t" - "movl %%ebx, 0x4(%%edi)" "\n\t" - "movl %%ecx, 0x8(%%edi)" "\n\t" - "movl %%edx, 0xC(%%edi)" "\n\t" - - : - /* No outputs */ - : - [index] "m" (index), - [rptr] "m" (regsPtr) - : - "%eax", "%ebx", "%ecx", "%edx", "%esi", "%edi", "memory"); - status = TCL_OK; - -# else - - TCLEXCEPTION_REGISTRATION registration; - - /* - * Execute the CPUID instruction with the given index, and store results - * off 'regPtr'. - */ - - __asm__ __volatile__( - /* - * Construct an TCLEXCEPTION_REGISTRATION to protect the CPUID - * instruction (early 486's don't have CPUID) - */ - - "leal %[registration], %%edx" "\n\t" - "movl %%fs:0, %%eax" "\n\t" - "movl %%eax, 0x0(%%edx)" "\n\t" /* link */ - "leal 1f, %%eax" "\n\t" - "movl %%eax, 0x4(%%edx)" "\n\t" /* handler */ - "movl %%ebp, 0x8(%%edx)" "\n\t" /* ebp */ - "movl %%esp, 0xC(%%edx)" "\n\t" /* esp */ - "movl %[error], 0x10(%%edx)" "\n\t" /* status */ - - /* - * Link the TCLEXCEPTION_REGISTRATION on the chain - */ - - "movl %%edx, %%fs:0" "\n\t" - - /* - * Do the CPUID instruction, and save the results in the 'regsPtr' - * area. - */ - - "movl %[rptr], %%edi" "\n\t" - "movl %[index], %%eax" "\n\t" - "cpuid" "\n\t" - "movl %%eax, 0x0(%%edi)" "\n\t" - "movl %%ebx, 0x4(%%edi)" "\n\t" - "movl %%ecx, 0x8(%%edi)" "\n\t" - "movl %%edx, 0xC(%%edi)" "\n\t" - - /* - * Come here on a normal exit. Recover the TCLEXCEPTION_REGISTRATION and - * store a TCL_OK status. - */ - - "movl %%fs:0, %%edx" "\n\t" - "movl %[ok], %%eax" "\n\t" - "movl %%eax, 0x10(%%edx)" "\n\t" - "jmp 2f" "\n" - - /* - * Come here on an exception. Get the TCLEXCEPTION_REGISTRATION that we - * previously put on the chain. - */ - - "1:" "\t" - "movl %%fs:0, %%edx" "\n\t" - "movl 0x8(%%edx), %%edx" "\n\t" - - /* - * Come here however we exited. Restore context from the - * TCLEXCEPTION_REGISTRATION in case the stack is unbalanced. - */ - - "2:" "\t" - "movl 0xC(%%edx), %%esp" "\n\t" - "movl 0x8(%%edx), %%ebp" "\n\t" - "movl 0x0(%%edx), %%eax" "\n\t" - "movl %%eax, %%fs:0" "\n\t" - - : - /* No outputs */ - : - [index] "m" (index), - [rptr] "m" (regsPtr), - [registration] "m" (registration), - [ok] "i" (TCL_OK), - [error] "i" (TCL_ERROR) - : - "%eax", "%ebx", "%ecx", "%edx", "%esi", "%edi", "memory"); - status = registration.status; - -# endif /* !_WIN64 */ -#elif defined(_MSC_VER) && defined(HAVE_CPUID) -# if defined(_WIN64) - - __cpuid(regsPtr, index); - status = TCL_OK; - -# elif defined (_M_IX86) +#elif defined (_M_IX86) /* * Define a structure in the stack frame to hold the registers. */ @@ -619,7 +497,6 @@ TclWinCPUID( /* do nothing */ } -# endif #else (void)index; (void)regsPtr; From a0a8e2f294ec770dbb8713ce29dda0865d7028f0 Mon Sep 17 00:00:00 2001 From: "jan.nijtmans" Date: Sun, 5 Jan 2025 20:41:05 +0000 Subject: [PATCH 5/7] oops --- win/tclWin32Dll.c | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/win/tclWin32Dll.c b/win/tclWin32Dll.c index 81db0f2502d..3b7bbea3ae0 100644 --- a/win/tclWin32Dll.c +++ b/win/tclWin32Dll.c @@ -13,7 +13,7 @@ #include "tclWinInt.h" #if defined(HAVE_CPUID_H) -# xinclude +# include #elif defined(_MSC_VER) # include #endif From 8183691a4f59cd8825eb70046924731ff91d0425 Mon Sep 17 00:00:00 2001 From: "jan.nijtmans" Date: Mon, 6 Jan 2025 21:15:03 +0000 Subject: [PATCH 6/7] This would be my advise. Using TCL_VERSION in Tcl_InitStubs() is a bad idea (IMHO) --- doc/InitStubs.3 | 41 +++++++++++++++++++++++++++++++---------- 1 file changed, 31 insertions(+), 10 deletions(-) diff --git a/doc/InitStubs.3 b/doc/InitStubs.3 index 62b53f52982..da659fc7eac 100644 --- a/doc/InitStubs.3 +++ b/doc/InitStubs.3 @@ -24,13 +24,33 @@ Tcl interpreter handle. The requirement string of compatible Tcl versions. It consists of one requirement as described for \fBpackage require\fR (\fBPACKAGE\fR manual page in the section \fBREQUIREMENT\fR). Multiple requirement strings like with \fBpackage require\fR are not supported. +Examples: +.CS +Tcl_InitStubs(interp, "8.5", 0); /* This extension only uses the Tcl 8.5 API. It will work with 8.6 as well */ +Tcl_InitStubs(interp, "8.6", 0); /* This extension only uses the Tcl 8.6 API. It will work with 8.6 only */ +Tcl_InitStubs(interp, "9.0", 0); /* This extension only uses the Tcl 9.0 API. It will work with 9.0 only */ +.CE +If any such extension is ported to Tcl 9.0, without additional API usage (only int -> Tcl_Size conversions, nothing more): +.CS +Tcl_InitStubs(interp, "8.5-", 0); /* This extension only uses the Tcl 8.5 API. It will work with 8.6 and (recompiled) with 9.0 as well */ +Tcl_InitStubs(interp, "8.6-", 0); /* This extension only uses the Tcl 8.6 API. It will work (recompiled) with 9.0 as well */ +.CE +If you don't know which Tcl version you are developing for, do the following: +.CS +$ tclsh +% info tclversion +9.0 +.CE +Now you know what requirement string to fill. .AP int exact in 1 means that only the particular version specified by -\fIversion\fR is acceptable. +\fIrequirement\fR is acceptable. An exact version number must be given as \fIrequirement\fR. -0 means that versions newer than \fIversion\fR are also -acceptable as long as they have the same major version number -as \fIversion\fR. Other bits have no effect. +0 means that versions newer than \fIrequirement\fR are also +acceptable. If the requirement string ends with \fB-\fR, +higher major versions are acceptable as well, otherwise +the major version must be the same as in the requirement string. +Other bits have no effect. See the \fB-require\fR option of the \fBpackage require\fR command. .BE .SH INTRODUCTION @@ -41,7 +61,7 @@ This provides two significant benefits to Tcl users: .IP 1) 5 Extensions that use the stubs mechanism can be loaded into multiple versions of Tcl without being recompiled or -relinked. +relinked, as long as the major Tcl version is the same. .IP 2) 5 Extensions that use the stubs mechanism can be dynamically loaded into statically-linked Tcl applications. @@ -81,8 +101,10 @@ and ensure that the correct version of Tcl is loaded. In addition to an interpreter handle, it accepts as arguments a version number and a Boolean flag indicating whether the extension requires an exact version match or not. If \fIexact\fR is 0, then the -extension is indicating that newer versions of Tcl are acceptable -as long as they have the same major version number as \fIversion\fR; +extension is indicating that newer versions of Tcl are acceptable. +If the requirement string ends with \fB-\fR, +higher major versions are acceptable as well, otherwise +the major version must be the same as in the requirement string. 1 means that only the specified \fIversion\fR is acceptable. \fBTcl_InitStubs\fR returns a string containing the actual version of Tcl satisfying the request, or NULL if the Tcl version is not @@ -94,9 +116,8 @@ function which has an additional \fBmagic\fR integer argument and uses the \fBex parameter to know the exact Tcl major version number. If you circumvent the macro in tcl.h, it should be called like this: .CS - if (Tcl_InitStubs(interp, "8.6-", TCL_MAJOR_VERSION<<8, TCL_STUB_MAGIC)) == NULL) { - return TCL_ERROR; - } +if (Tcl_InitStubs(interp, "9.0", TCL_MAJOR_VERSION<<8, TCL_STUB_MAGIC)) == NULL) { + return TCL_ERROR; } .CE If \fBUSE_TCL_STUBS\fR is not defined, it translates to a TCL version verification only. From bee9aff93622bd5fd3b1f15eaeb2d88b3bde0677 Mon Sep 17 00:00:00 2001 From: "jan.nijtmans" Date: Fri, 10 Jan 2025 13:59:22 +0000 Subject: [PATCH 7/7] Another attempt to make it clear, without too much bloat --- doc/InitStubs.3 | 69 ++++++++++++++----------------------------------- 1 file changed, 19 insertions(+), 50 deletions(-) diff --git a/doc/InitStubs.3 b/doc/InitStubs.3 index da659fc7eac..decea876e4f 100644 --- a/doc/InitStubs.3 +++ b/doc/InitStubs.3 @@ -14,44 +14,23 @@ Tcl_InitStubs \- initialize the Tcl stubs mechanism \fB#include \fR .sp const char * -\fBTcl_InitStubs\fR(\fIinterp, requirement, exact\fR) +\fBTcl_InitStubs\fR(\fIinterp, version, exact\fR) .fi .SH ARGUMENTS .AS Tcl_Interp *interp .AP Tcl_Interp *interp in Tcl interpreter handle. -.AP "const char" *requirement in -The requirement string of compatible Tcl versions. -It consists of one requirement as described for \fBpackage require\fR (\fBPACKAGE\fR manual page in the section \fBREQUIREMENT\fR). -Multiple requirement strings like with \fBpackage require\fR are not supported. -Examples: -.CS -Tcl_InitStubs(interp, "8.5", 0); /* This extension only uses the Tcl 8.5 API. It will work with 8.6 as well */ -Tcl_InitStubs(interp, "8.6", 0); /* This extension only uses the Tcl 8.6 API. It will work with 8.6 only */ -Tcl_InitStubs(interp, "9.0", 0); /* This extension only uses the Tcl 9.0 API. It will work with 9.0 only */ -.CE -If any such extension is ported to Tcl 9.0, without additional API usage (only int -> Tcl_Size conversions, nothing more): -.CS -Tcl_InitStubs(interp, "8.5-", 0); /* This extension only uses the Tcl 8.5 API. It will work with 8.6 and (recompiled) with 9.0 as well */ -Tcl_InitStubs(interp, "8.6-", 0); /* This extension only uses the Tcl 8.6 API. It will work (recompiled) with 9.0 as well */ -.CE -If you don't know which Tcl version you are developing for, do the following: -.CS -$ tclsh -% info tclversion -9.0 -.CE -Now you know what requirement string to fill. +.AP "const char" *version in +A version string, indicating which minimal version of Tcl is accepted. +Normally just \fB"9.0"\fR. Or \fB"8.6-"\fR if both 8.6 and 9.0 are accepted. .AP int exact in 1 means that only the particular version specified by -\fIrequirement\fR is acceptable. -An exact version number must be given as \fIrequirement\fR. -0 means that versions newer than \fIrequirement\fR are also -acceptable. If the requirement string ends with \fB-\fR, -higher major versions are acceptable as well, otherwise -the major version must be the same as in the requirement string. +\fIversion\fR is accepted. +0 means that versions newer than \fIversion\fR are also +accepted. If the\fIversion\fR ends with \fB-\fR, +higher major versions are accepted as well, otherwise +the major version must be the same as in \fIversion\fR. Other bits have no effect. -See the \fB-require\fR option of the \fBpackage require\fR command. .BE .SH INTRODUCTION .PP @@ -100,28 +79,18 @@ more information. and ensure that the correct version of Tcl is loaded. In addition to an interpreter handle, it accepts as arguments a version number and a Boolean flag indicating whether the extension requires -an exact version match or not. If \fIexact\fR is 0, then the -extension is indicating that newer versions of Tcl are acceptable. -If the requirement string ends with \fB-\fR, -higher major versions are acceptable as well, otherwise -the major version must be the same as in the requirement string. -1 means that only the specified \fIversion\fR is acceptable. +an exact version match or not. If \fIexact\fR is 0, then versions +newer than \fIversion\fR are also accepted. If the\fIversion\fR +ends with \fB-\fR, higher major versions are accepted as well, +otherwise the major version must be the same as in \fIversion\fR. +1 means that only the specified \fIversion\fR is accepted. +\fIversion\fR can be any construct as described for \fBpackage require\fR +(\fBPACKAGE\fR manual page in the section \fBREQUIREMENT\fR). +Multiple requirement strings like with \fBpackage require\fR are not supported. + \fBTcl_InitStubs\fR returns a string containing the actual version of Tcl satisfying the request, or NULL if the Tcl version is not -acceptable, does not support stubs, or any other error condition occurred. -.PP -\fBTcl_InitStubs\fR is implemented as a macro. -If \fBUSE_TCL_STUBS\fR is defined, this macro calls the real \fBTcl_InitStubs\fR -function which has an additional \fBmagic\fR integer argument and uses the \fBexact\fR -parameter to know the exact Tcl major version number. If you circumvent -the macro in tcl.h, it should be called like this: -.CS -if (Tcl_InitStubs(interp, "9.0", TCL_MAJOR_VERSION<<8, TCL_STUB_MAGIC)) == NULL) { - return TCL_ERROR; -} -.CE -If \fBUSE_TCL_STUBS\fR is not defined, it translates to a TCL version verification only. -Thus, the call may also be used if the TCL library is directly linked. +accepted, does not support stubs, or any other error condition occurred. .SH "SEE ALSO" Tk_InitStubs package