Introduction

This is meant to be a comprehensive list of commands in Deal which are not native to Tcl. If you want to learn Tcl, I will be providing a set of pointers later.
Text in blue is meant for Tcl afficianado - reading this text without knowledge of Tcl might confuse more than enlighten.

Text in green is meant for programmers interested in finding code definitions.

Hand commands: north, east, south, west, hand

Usage

east [ -void string] [ suitname ... ]
hand {hand string} [ -void string] [ suitname ... ]

south subcommand [ ... args ... ]
hand {hand string} subcommand [ ... args ... ]
These are very strange commands - they really have too much stuffed into them. That's a historical anomoly which I'm reluctant to abandon.

With no subcommands, the routine is used for formatting the hand as a string. For example, if south is:

S: AJ5432
H: KT94
D: void
C: K93
Then the results of various commands are:
south                    =>    {AJ5432 KT94 {} K93}
south spades             =>    AJ5432
south hearts clubs       =>    {KT94 K93}
south -void -            =>    {AJ5432 KT94 - K93}
south -void --- diamonds =>    ---

set h {AK4 {} A95432 JT98}
hand $h                  =>    {AK4 {} A95432 JT98}
hand $h spades           =>    AK4
hand $h hearts clubs     =>    {{} JT98}
hand $h -void -          =>    {AK4 - A95432 JT98}
The -void switch exists precisely for formatting the output.

The various subcommands will be treated in seperate sections. The hand version of this command only works with some subcommands.

For Programmers

Implementation:
C code
Location:
tcl_deal.c, function tcl_hand_cmd

Subcommand: shape

Usage

north shape
hand {AJTxx Axx JTxx x} shape

Summary

Returns a list of suit lengths, in standard order.

For example, the second command above would return the list:

5 3 4 1

Subcommand: pattern

Usage

north pattern
hand {AJ32 A5 JTxx xxx} pattern

Summary

Returns a sorted list of the suit lengths, starting with the longest. For example, the second command above would return the list:
4 4 3 2

Subcommand: is

Usage

south is {handstring}

Summary

This command pre-places a specific 13 cards. For voids, you can use the "-" character:
south is {AJ32 KT932 - Q876}
Calls to this subcommand must occur before dealing begins, outside the "main" command.

This subcommand calls the stack_hand command. Inside the main code, the deal is already dealt.

Subcommand: gets

Usage

handname gets card [card ...]

Summary

Puts a specific list of cards in the hand named. As with the "is" subcommand, this must be called before dealing starts, outside the "main" command. The card is specified in two characters, a rank character and a suit character:
AS KH JH 9D
This routine dispatches its data to the stack_cards command.

Subcommand: has

Usage

handname has card [card ...]
hand {string} has card [...]

Summary

This returns a count of the cards listed that occur in the hand named.
% south
AJ54 JT54 43 654
% south has SA HK HJ HT
3
 

Control commands

Commands: accept and reject

Usage

accept [ [ if | unless ] expr expr ...]
reject [ [ if | unless ] expr expr ...]

Summary

Without arguments, accept and reject are the same as return 1 and return 0, respectively.

With arguments it's a little more obscure.

accept if {$h>4}
This returns 1 if the expression matches. It is equivalent to:
if {$h>4} { return 1 }
The code:
accept if {$h>4} {$s>4}
Is logically equivalent to:
if {$h>4 || $s>4} { return 1}
That is, the values are accepted if either of the expressions evaluates as true.

The unless option does the opposite:

accept unless {$h>4} {$s>4}
This is equivalent to:
if {!($h>4) || !($s>4)} { return 1 }

The means we return with a true value unless one of the expressions is true. If one of the values is true, we simply go on to the next line.

Examples

Virtually all of the examples included in the release contain an instance of "accept" or "reject."
For Programmers:

Implementation:
C code
Location:
tcl_deal.c, procedure tcl_deal_control.
This construct is borrowed from the Perl programming language. Originally, I added it to Deal 2.0 for performance reasons. Those reasons are obsolote with Tcl 8.x, but I like the mnemonic enough to keep it, and removing it might confuse old users.

For Tcl Experts:

There actually is one subtle and occasionally useful distinction between accept/reject and the stated "if {...} { return ... }" version. In reality:

accept if {$h>4} {$s>4}
is equivalent to:
if {$h>4||$s>4} { 
    return 1
}
expr 0
This only matters when the command is the last command in a procedure. The two procedures:
proc A {h s} {
    if {$h>4||$s>4} {
        return 1
    }	
}

proc B {h s} {
     accept if {$h>4} {$s>4}
}
are slightly different. A call of A 3 3 returns an empty string, while a call of B 3 3 returns 0. That can be useful, as you can see here.

Command: main

Usage

main {
   block of code
}

Summary

This defines the primary block of code evaluated after each deal is generated. If the code returns true, Deal will treat the deal as a "match" and call write_deal. It will also increment its count of deals found and exit if the limit is reached.
For Programmers

Implementation:
C code
Location:
tcl_deal.c, procedure tcl_deal_control.

Command: whogets

Usage

whogets cardname
This returns the name of the person holding the card cardname.
For Programmers

Implementation:
C code
Location:
tcl_deal.c, procedure tcl_deal_to_whom.

Command: deal_finished

Usage

deal_finished {
   block of code
}
This defines the code called at the completion of generating all deals. This is code you would use to dump statistics, for example.
For Programmers

Implementation:
C code
Location:
tcl_deal.c, procedure tcl_after_set.

Bridge Evaluators

Common interfaces

There are some standard interfaces to bridge evaluation functions.

Hand Procedures

Any procedure defined entirely based on the values in one hand is considered a hand procedure. These procedures can be called in one of two ways:
Weak2Bid handname

Weak2Bid hand {AKQxxx xxx xxx x}
The handname parameter can be one of north, south, east, west. Shape procedures and holding procedures fit the hand procedure calling method, along with other options.

Shape procedures

Any procedure defined on the shape of a hand can be called with one of the following methods:
balanced handname

balanced hand {AJ43 AKxx xxx K5}

balanced eval 4 3 4 2

balanced shape {4 3 4 2}
This follows the hand procedure outline with the addition of the eval option.


Holding procedures

A holding procedure is a hand procedure which is evaluated on a bridge hand by evaluating each suit holding one at a time, and then accumulate the results. There are two common ways of defining holding procedures, defvector and holdingProc.

This is an abstraction of a very common bridge evaluation technique. The most commonly used holding functions are high card points, losing trick count, controls. For example, when counting the losers in the hand AKxx Axx KQJxxx x, we find 1 loser in spades, 2 losers in hearts, 1 loser in diamonds, and one loser in clubs. We define the total losers to be the sum across all suits, and get 5 losers, total.

The interface lets you evaluate the entire hand, or only selected suits in a hand, or a specific list of holdings.

hcp handname [ suitname ...  ]

hcp hand {AKJxxx K xxx xxx} [ suitname ...  ]

hcp holding AK43 [ ... ]
In the first two cases, if no suits are named, all suits are evaluated, and the results are accumulated. In the case of the holding call, the specific holdings passed in are evaluated.

Note, I've been saying "accumulated" rather than added. In some cases, you might have an accumulation method different from addition. For example, you might return the standard opening leads from a holding:

    openingLead holding KQJ7    =>   K
    openingLead south           => {K} {7 2} {4} {J}
Each suit would create a list of proposed leads, and the result would be a list of lists of all standard leads from all holdings. Here, the accumulation is to simply make a list.

Command: balanced and semibalanced

Usage

balanced handname

balanced hand {text hand}

balanced eval s h d c

Summary

These are both shape procedures.

balanced returns true if the hand is some 4333, 4432, or 5332 without a 5-card major.

semibalanced returns true if the hand has no singletons or voids, not six-card majors, and no seven-card minors.

For Programmers

Implementation:
Tcl with shapecond
Location:
deal.tcl
These use to be hard-coded in the C code, but it made more sense to add them to the library code - the user can change the definitions as he likes.

Commands: clubs, diamonds, hearts, spades

Usage

     spades handname
     spades hand {AKxx Axx AJxx Kx}
These implement the hand procedure interface, and return the suit lengths for the appropriate suit.
For Programmers

Implementation:
Location:
tcl_deal.c, function tcl_suit_count
These could be shape procedures but for speed reasons, I've left them this way.

Holding Evaluators

Command: hcp

Usage

hcp handname
hcp handname suitname [suitname ...]
hcp hand {s h d c}
hcp hand {s h d c} suitname [suitname ...]
This procedure computes standard high card points - ace is 4, king is 3, queen is 2, jack is 1. The procedure implements the holding procedure interface.
For Programmers

Implementation:
C code
Location:
Built with additive.c, tcl_create_additive and deal.c, count_hcp

Command: losers

Usage

losers handname
losers handname suitname [suitname ...]
losers hand {s h d c}
losers hand {s h d c} suitname [suitname ...]
This implements a holding procedure which computes the number of half losers. This is for historical reasons - the only holding functions allowed in past releases returned integer values, but I wanted to have some refinements over raw losing trick count.
For Programmers

Implementation:
C code
Location:
Built with additive.c, tcl_create_additive and deal.c, count_hcp
It is probably better to reimplement in Tcl using holdingProc.

Bridge Logic and Utilities

Commands: lho, partner, rho

Usage

lho handname
rho handname
partner handname

Summary

These routines accept one hand name and return another. For example, "lho north" returns "east," and "partner east" returns "west."
For Programmers

Implementation:
C code
Location:
tcl_deal.c, function tcl_other_hand

Commands: holding

Usage

holding length AJxx           =>     4

holding contains AJ64 A6      =>     1 (true)
holding contains AJ64 A65     =>     0 (false)

holding encode AJ4            =>     4612  (binary: 1001000000100 )

holding decode 4612           =>     AJ4

holding matches AKxx  AK42    =>     1 (true)

holding disjoint AJ65 KT82    =>     1 (true)
holding disjoint A65  A32     =>     0 (false)

holding index AKJ4 0          =>     A
holding index AKJ4 1          =>     K
holding index AKJ4 3          =>     4
holding index AKJ4 -2         =>     J
holding index AKJ4 10         =>     ""

Summary

For Programmers

Implementation:
C code
Location:
holdings.c, function IDeal_HoldingCmd

Command: score

Usage

source lib/score.tcl
   ...
score contract vulnerability tricks

Summary

This routine computes the standard duplicate bridge score for a contract from the point of view of declarer, if declaring side takes the given number of tricks.

The contract is passed as a list:

2 spades

2 spades undoubled 

4 clubs doubled

6 notrump redoubled
The vulnerablity is one of the words vul or nonvul.

The tricks is the number of tricks taken by declarer.

Examples

score {2 spades} nonvul 8           =>  110
score {2 spades doubled} nonvul 8   =>  470
score {2 spades redoubled} nonvul 8 =>  640
score {2 spades doubled} vul 8      =>  670
score {2 spades} nonvul 7           =>  -50
score {2 spades doubled} nonvul 7   =>  -100
score {2 spades doubled} vul 7      =>  -200
For Programmers

Implementation:
Tcl
Location:
score.tcl

GIB-related routines

These next routines use calls to the GIB double-dummy engine. You must have a licensed version of GIB installed on your computer for these routines to work. Also, these routines only work on Windows at the moment.

Command: gib::directory

Usage

source lib/gib.tcl
   ...
gib::directory path

Summary

This tells the GIB-related routines where GIB is installed.

If this routine is not called, it defaults to GIB's default install directory, C:/Program Files/GIB.

Note: your path should include forward slashes '/', even on Windows.

For Programmers

Implementation:
Tcl
Location:
gib.tcl

Command: gib::tricks

Usage

source lib/gib.tcl
   ...
gib::tricks declarer denomination

Summary

Denomination can be a suit name or "notrump." Declarer can be any hand name.

This routine returns the double-dummy number of tricks available in this deal, in the given denomination by the given declarer.

If GIB is installed anywhere unusual, you will need to call gib::directory before gib::tricks is called.

For Programmers

Implementation:
Tcl
Location:
gib.tcl

Command: parscore

Usage

source lib/parscore.tcl
   ...
set result [parscore dealer vul]

set contract [lindex $result 0]
set declarer [lindex $result 1]
set score [lindex $result 2]
set tricks [lindex $result 3]
set auction [lindex $result 4]

Summary

This computes the double-dummy par result for the hand.

The parscore routine returns a list of five values - the contract (in the same form that is used by the score routine above), the declarer, the score for north/south, the number of tricks taken, and a "stupid" auction to the contract suitable for putting into a PBN file.

Dealer is relevant on a few occasions. If both south and west can make 1NT, for example, and no 2-level contracts make, then "par" is 1NT by whomever gets a chance to bid it first.

If GIB is installed anywhere unusual, you will need to call gib::directory before gib::tricks is called.

For Programmers

Implementation:
Tcl
Location:
parscore.tcl

Bridge Command Builders

A number of common types of bridge functions can easily be implemented to run quickly via lookup tables, including holding and shape procedures. Deal lets the user take advantage of these two sorts of lookup procedures relatively easily.

Command: defvector

Usage

defvector vec aceVal [ kingVal ... ]

vec handname

vec hand {text hand}

vec handname suitname [ suitname ... ]

vec handname suitname [ suitname ... ]

vec hand {text hand} suitname ...

Summary

This defines a holding procedure which assigns integer values to the various cards in the hand. For example
defvector hcp6421 6 4 2 1
Defines a holding procedure which counts the ace as worth 6 points, the king as worth 4, the queen as 2, and the jack as 1.

Vectors are limited to being integer-valued. For more complicated holding procedures, use holdingProc.

For Programmers

Implementation: C
Location: vector.c Vectors are slighly faster than their equivalent holdingProc. They are an old optimization which remain mostly for backwards compatibility.


Command: holdingProc

Usage

Temporarily, see the seperate documentation.

For Programmers
Implementation: C
Location: holdings.c, function IDeal_DefHoldingProc

Commands: shapeclass, shapecond, and shapefunc

Usage

shapeclass name {... code ...}

shapecond name {expr}

shapefunc name {... code ...}

name [ north | south | east | west ]

name eval s h d c

name shape {s h d c}

A shape class (or equivalently, a shape condition) also has the list subcommand:
name list

Summary

These commands create new procedures which fit the shape function interface.

shapeclass and shapecond define procedures which returns boolean values. shapefunc can return any type of data. shapecond is actually an alias:

shapecond name {expr}
is the equivalent of:
shapeclass name {
  if {expr} {
    return 1
  } else {
    return 0
}
The code or expr is allowed to use the variables s, h, d, or c.

More details can be found in the Advanced Guide.

The list subcommand for shape classes returns a list of shapes that are in the class.

Why are there two subcommands, "eval" and "shape" which do the roughly the same things?

Let's say you have a class named "SemiBalanced" already defined, which includes 5-card majors. You want to define a "Balanced" class which excludes the 5-card majors. You can do this with the call:

shapecond Balanced {[SemiBalanced eval $s $h $d $c] && $s<5 && $h<5}
On the other hand, if you have a shape - output by the, say, a call to [north shape] you can evaluate it as:
    SemiBalanced shape [north shape]
In fact, this is exactly equivalent to calling
    SemiBalanced north
only slightly slower.
For Programmers
Implementation: C
Location: dist.c, functions tcl_shapeclass_define, tcl_shapecond_define, and tcl_shapefunc_define

Statistics

Command: sdev

Usage

sdev name

name add data [ data ... ]

name count

name average

name sdev

name rms

Summary

The "sdev" command defines a new Tcl procedure with the given name, which behaves as a data collection object. You can add data points to it, or you can retrieve the current number of data points, the average of the data points, the standard deviation of the data points, or the "root mean square" of the data points.

For Programmers
Implementation:
C code
Location:
stat.c, function tcl_sdev_define.
This was implemented in C for efficiency reasons - most real number computations really need to be done in C if they are going to be done frequently, and here, the "add" command is called quite often in normal usage.

Command: correlation

Usage

     correlation name

     name add xval yval [ xval yval ... ]

     name count

     name correlate

     name average [ x | y ]

     name sdev [ x | y ]

     name rms [ x | y ]

Summary

The correlation declaration defines a routine much like the sdev command, but each datapoint is a pair of values, and it computes the linear correlation between the x and y values.

You can also get individual data for the x and y values.

If there is no data, it returns an error on average, correlate, sdev and rms.

If there is only one pair entered, it will throw an error on sdev.

For Programmers
Implementation:
C code
Location:
stat.c, function tcl_correlation_define.
This was implemented in C for efficiency reasons - most real number computations really need to be done in C if they are going to be done frequently, and here, the "add" command is called quite often in normal usage.

Utility Procedures

Command: deal_deck

Usage

deal_deck

Summary

deal_deck is called at every new deal, even when all 52 cards are specified (stacked) precisely. Imagine stacking cards as stacking them in an undealt deck, but that the cards are only distributed to the hands when deal_deck is called.

Most of the time, you won't need to call deal_deck directly, but it is useful to understand its behavior if you are going to write advanced procedures like input formats.

Stacked cards are permanently stacked until the deck is reset. In this code:

    stack_hand south {AKQ32 AJ53 K54 2}
    deal_deck
    deal_deck
    puts [south spades]
The output is "AKQ32". Use reset_deck to undo card stackings.

The deal_deck procedure does one thing rather complicated when it is called, the first thing it does is execute all code registered with the deal_reset_cmds. This allows the script writer to do one of several things before the deal is generated:

  • Delete metadata related to the deal. For example, the first time the user calls gib::tricks south spades it calls the GIB double-dummy processor. Each time after that, it uses a stored value for this function call up until the next call to deal_deck.
  • The reset command might, instead, read the next deal from a file, stack the deck, and then re-register itself. Then when deal_deck is is called, it just deals that hand from the file. This is a crude way of allowing Deal to transparently read deals from a file (or generate deals in other ways, such as smart stacking.
  • For Programmers

    Implementation:
    C
    Location:
    tcl_deal.c, function tcl_deal_deck

    Command: reset_deck

    Usage

    reset_deck
    

    Summary

    This forgets about all stacked cards. The deck, from Deal's point of view, plus a list of cards which must go to specific hands. The cards which are assigned to specific hands are "stacked." The cards which are not stacked can go anywhere at the next call to deal_deck.
    For Programmers

    Implementation:
    C
    Location:
    tcl_deal.c, function tcl_reset_deck

    Commands: stack_hand and stack_cards

    Usage

    stack_hand handname hand
    stack_cards handname suitname holding [...]
    

    Summary

    By default, these two routines just call deck_stack_hand and deck_stack_cards, respectively - that is, they forcibly place cards in the deck.

    But these routines are meant to be overridden. For example, when using one of the input formats which reads deals from a file, the format will redefine these two procedures to give errors.

    % deal -I "line foo.txt" -e "stack_cards south spades AJ"
    Tcl stack dump of error info:
    No card stacking with input format ::line
    ...
    
    A more complicated re-definition occurs in the smartstack input format, which alters its hand generation depending on what cards are stacked where.
    For Programmers

    Location:
    hand.c, function tcl_stack_hand and tcl_stack_cards

    Commands: deck_stack_hand and deck_stack_cards

    Usage

    deck_stack_hand handname hand
    deck_stack_cards handname suitname holding [...]
    

    Summary

    These routines are the "underlying" deck-stacking routines. Any cards stacked with these routines remain stacked until the next call to reset_deck
    For Programmers

    Location:
    hand.c, function tcl_stack_hand and tcl_stack_cards

    Command: stacked

    Usage

    stacked handname
    

    Summary

    Determines what cards are stacked to this hand, returning them as a list of holdings:
    south gets AS KH 3H QD JD TC 9C 8C
    puts [stacked south]
    
    writes out the list:
    A K3 QJ T98
    
    This is useful for the smartstacker, because we don't want to force the user to stack cards *after* specifying conditions.
    stack_hand north {AJT KT3 KJ 75432}
    deal::input smartstack south balanced hcp 20 21
    
    The call to stack_hand occurs before smartstack is loaded, so stack_hand has not been redefined. So what smartstack does is read the cards already stacked, and use that for its initial conditions.

    For Programmers

    Implementation:
    Location:
    tcl_deal.c, function tcl_stacked_cards

    Command: deal::nostacking

    Usage

    ::deal::nostacking
    

    Summary

    This procedure redefines the stack_hand and stack_cards procedures to generate error messages, and generates an error if any cards are currently stacked.

    This is used in all of the input formats which read complete deals from files, and thus are incompatible with stacking.

    For Programmers

    Implementation:
    Location:
    deal.tcl, function deal::nostacking

    Command: deal_reset_cmds

    Usage

    deal_reset_cmds {...code...} [ ... ]
    

    Summary

    This adds a set of commands that are to be called before the next deal. The code is executed at the next call to the deal_deck procedure, which, unless you are doing something complicated, means it is called at the beginning of each time through the evaluation loop.

    deal_reset_cmds can be used so that metadata about the previous deal, such as cached values and the like, are unset. See deal::metadata.

    It is also used for defining input formats, so that deals can be read from files. For example, the "line" input format has the following routine declared:

    namespace eval line {
        #....
        proc next {} {
    	variable handle
    	set length -1
    	catch { set length [gets $handle line] }
    
            # ... process the line, or handle oef stuff ...
            # ...
    	deal_reset_cmds {::line::next}
        }
    }
    

    The key is that when line::next is done, it re-registers itself, making sure that it is called next time through as well.

    The code passed in to deal_reset_cmds is called only once, at the next request for a deal - think of it as registering a one-time trigger. Multiple triggers can be registered - they are called in the reverse order that they are added, which can seem counter-intuitive until you think of the process as an "undo" process.

    For Programmers

    Implementation:
    Location:
    tcl_deal.c, function add_reset_cmds
    This callback idea is fairly powerful, and will probably be clarified and elaborated in later releases. Unfortunately, initial efforts to use it to alter the deal source (by say reading deals from a file) have been mixed - Windows/DOS seems to choke occasionally on it.

    Command: deal::metadata

    Usage

    deal::metadata name {...code...}
    

    Summary

    Currently, this is just a peculiar way of caching slow evaluation routines. At the moment, the only place it is used is in gib::tricks.

    When you call deal::metadata, it will check to see if there is a value associated with the name requested already. If there is, the value is returned. If it does not, it evaluates the specified code and associates the result with the name. The key is, when the next deal is being analyzed, all the cached values are pitched.

    This isn't really necessary or efficient in most cases, but with evaluations which take some time, e.g. GIB calls, it improves things.

    For Programmers

    Implementation:
    Tcl
    Location:
    deal.tcl
    In later releases, metadata read from input streams (say, the fields from a PBN file) will also be stored here.

    This procedure uses deal_reset_cmds. The metadata is cleared each time the deal_deck command is called.


    Input formats

    Command: deal::input

    Usage

    deal::input formatName args
    

    Summary

    The deal::input command is used to define an input source for deals. It works as follows:
    1. The file input/<formatName>.tcl is loaded. This Tcl file should define a new Tcl 'namespace' with the name formatName, with member procedures named set_input and next.
    2. Deal then calls <formatName>::set_input args, which should initialize the format reading. Usually, the argument(s) are one argument representing the source file.
    3. The next deal, <formatName>::next is called.
    For Programmers:

    Implementation:
    Tcl code
    Location:
    deal.tcl

    Input Format: giblib

    Usage

    deal::input giblib [filename]
    
    or on the command line:
    % deal -I giblib
    
    or
    % deal -I "giblib filename"
    

    Summary

    Specifies that deals are read from the specified file in the format of Matt Ginsberg's library.dat file. This includes double-dummy tricks data, so that later calls to gib::tricks will not entail calls to the GIB binaries.

    If no filename is given, the library tries to read from a file called "library.dat" in the current directory.

    The -I command-line flag is a quick alias for deal::input, passing the next argument as Tcl arguments to the deal::input command.

    For Programmers:

    Implementation:
    Tcl code
    Location:
    input/giblib.tcl
    This procedure uses deal_reset_cmds.

    Input Format: line

    Usage

    deal::input line [filename]
    

    Summary

    Specifies that deals are read from the specified file in the format written by Deal's "-l" option.

    If no filename is given, the library reads from standard input. This way, you can create a single data file and then run several different queries against it:

    % deal -l 100000 > sample
    % deal -e "deal::input line sample" -i query1 
    % deal -e "deal::input line sample" -i query2
    
    [ The -e option just evaluates the code in the next argument as Tcl code. Alternative, you can use the -I option, which is shorthand for input formats:
    % deal -I "line sample" -i query1 
    
    The -I args option is exactly the same as -e "deal::input args"
    For Programmers:

    Implementation:
    Tcl code
    Location:
    input/line.tcl

    Input: smartstack

    Usage

    deal::input smartstack hand shapeclass [holdproc min max]
    
    or on the command line:
    % deal -I "smartstack shapeclass ..."
    

    Summary

    This is the most complicated Deal input method in the current release. It does not read deals from a file - it is, instead, a different path to generation of deals. The purpose of this format is to allow fast generations of deals where one hand fits a very specific description. For example, if you want to find hands where south has a balanced 27-count, you could write:
    deal::input smartstack south balanced hcp 27 27
    
    On my computer, that returns the first deal in 14 seconds, and every deal after that takes a tiny fraction of a second. That's because smartstack first builds a large "factory" in memory, but once it is built, the factory spits out hands matching this condition very quickly.

    By contrast, a standard "deal" script to find balanced 27-counts takes about 2.8 seconds to find each deal. That means that if you only want to find about five deals of this sort, the old style program is faster, but if you want a lot more, use smartstack. For example, if you wanted 1000 examples, smartstack takes about 15 seconds, while the old deal will take about 45 minutes.

    One interesting feature of smartstack is that it is often faster if the hand type is less frequent. For example, it takes about 6 seconds to find ten deals with 9-card spade suits, and about 5 seconds to find ten deals with 10-card spade suits. Similarly, it is faster at finding hands with exactly 9 controls than it is at finding hands with 9-12 controls.

    The smartstack routine only works on one hand - after it places cards in that hand, it generates the rest using the usual algorithm.

    For Programmers:

    Implementation:
    Tcl code
    Location:
    input/smartstack.tcl, lib/handFactory.tcl
    Notes
    See the two articles from the bridge developers mailing list describing the algorithm: My first stab at describing it and some corrections.

    Formatting

    Command: write_deal

    Usage

    write_deal
    

    Summary

    This is the the name of a procedure which is called when deals are accepted. By default, it writes the result in the format:
              S: 98632
              H: A4
              D: AJ754
              C: J
     S: K              S: AJT7
     H: J3             H: QT95
     D: T98            D: Q2
     C: AKT7532        C: 984
              S: Q54
              H: K8762
              D: K63
              C: Q6
    --------------------------
    
    New output formats are defined by redefining this routine.
    For Programmers

    Implementation:
    Tcl
    Location:
    format/default

    Command: flush_deal

    Usage

    flush_deal
    

    Summary

    This routine is called at the very end of a run for deal. It does nothing, by default, but some output formats may require it if data is cached.
    For Programmers

    Implementation:
    Location:
    YYYYY, function ZZZZ