summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--hash.c31
-rw-r--r--hash.h2
-rw-r--r--txr.1120
3 files changed, 114 insertions, 39 deletions
diff --git a/hash.c b/hash.c
index 1720aa10..c5330daf 100644
--- a/hash.c
+++ b/hash.c
@@ -37,6 +37,7 @@
#include "lib.h"
#include "gc.h"
#include "args.h"
+#include "txr.h"
#include "signal.h"
#include "unwind.h"
#include "stream.h"
@@ -72,7 +73,7 @@ struct hash_iter {
val cons;
};
-val weak_keys_k, weak_vals_k, equal_based_k, userdata_k;
+val weak_keys_k, weak_vals_k, equal_based_k, eql_based_k, userdata_k;
/*
* Dynamic lists built up during gc.
@@ -1059,17 +1060,40 @@ void hash_process_weak(void)
do_iters();
}
+static val equal_based_p(val equal, val eql, val wkeys)
+{
+ if (opt_compat && opt_compat <= 187)
+ return equal;
+
+ if (equal && eql)
+ uw_throwf(error_s,
+ lit("make-hash: mutually exclusive :equal-based and :eql-based"),
+ nao);
+
+ if (wkeys) {
+ if (equal)
+ uw_throwf(error_s,
+ lit("make-hash: mutually exclusive :equal-based and :weak-keys"),
+ nao);
+ else
+ eql = t;
+ }
+
+ return null(eql);
+}
+
val hashv(struct args *args)
{
- val wkeys = nil, wvals = nil, equal = nil, userdata = nil;
+ val wkeys = nil, wvals = nil, equal = nil, eql = nil, userdata = nil;
struct args_bool_key akv[] = {
{ weak_keys_k, nil, &wkeys },
{ weak_vals_k, nil, &wvals },
{ equal_based_k, nil, &equal },
+ { eql_based_k, nil, &eql },
{ userdata_k, t, &userdata }
};
val hash = (args_keys_extract(args, akv, sizeof akv / sizeof akv[0]),
- make_hash(wkeys, wvals, equal));
+ make_hash(wkeys, wvals, equal_based_p(equal, eql, wkeys)));
if (userdata)
set_hash_userdata(hash, userdata);
return hash;
@@ -1431,6 +1455,7 @@ void hash_init(void)
weak_keys_k = intern(lit("weak-keys"), keyword_package);
weak_vals_k = intern(lit("weak-vals"), keyword_package);
equal_based_k = intern(lit("equal-based"), keyword_package);
+ eql_based_k = intern(lit("eql-based"), keyword_package);
userdata_k = intern(lit("userdata"), keyword_package);
val ghu = func_n1(get_hash_userdata);
diff --git a/hash.h b/hash.h
index f5af5c0a..13c21a76 100644
--- a/hash.h
+++ b/hash.h
@@ -25,7 +25,7 @@
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
-extern val weak_keys_k, weak_vals_k, equal_based_k;
+extern val weak_keys_k, weak_vals_k, equal_based_k, eql_based_k, userdata_k;
cnum equal_hash(val obj, int *count);
val make_hash(val weak_keys, val weak_vals, val equal_based);
diff --git a/txr.1 b/txr.1
index 645c4e04..c40542c7 100644
--- a/txr.1
+++ b/txr.1
@@ -6851,7 +6851,7 @@ Example:
.cblk
@; match text into variables a and b, then insert into hash table h
- @(bind h (hash :equal-based))
+ @(bind h (hash))
@a:@b
@(do (set [h a] b))
.cble
@@ -11293,7 +11293,9 @@ arguments and the key-value pairs. For instance:
where
.code (:equal-based)
-is the list of construction arguments and the pairs
+indicates that this hash table's keys are treated using
+.code equal
+equality, and
.code "(a 1)"
and
.code "(b 2)"
@@ -11380,6 +11382,7 @@ The first item in the syntax is a list of keywords. These are the same
keywords as are used when calling the function hash to construct
a hash table. Allowed keywords are:
.codn :equal-based ,
+.codn :eql-based ,
.codn :weak-keys ,
.codn :weak-values ,
and
@@ -11392,6 +11395,11 @@ specifies the hash table's user data, which
can be retrieved using the
.code hash-userdata
function.
+The
+.code :equal-based
+and
+.code :eql-based
+keywords are mutually exclusive.
An empty list can be specified as
.code nil
@@ -17251,7 +17259,7 @@ Two hashes are
.code equal
if they use the same equality (both are
.codn :equal-based ,
-or both are the default
+or both are
.codn :eql-based ),
if their associated user data elements are equal (see the function
.codn hash-userdata ),
@@ -29062,27 +29070,16 @@ function. The first occurrence of each element is retained,
and the subsequent duplicates of that element, of any, are suppressed,
such that the order of the elements is otherwise preserved.
-The following equivalence holds between
+The
.code uniq
-and
-.codn unique :
+function is an alias for the one-argument case of
+.codn unique .
+That is to say, this equivalence holds:
.cblk
- (uniq s) <--> [unique s : :equal-based]
+ (uniq s) <--> (unique s)
.cble
-That is,
-.code uniq
-is like
-.code unique
-with the default
-.meta keyfun
-argument (the
-.code identity
-function) and an
-.codn equal -based
-hash table.
-
.coNP Function @ unique
.synb
.mets (unique < sequence >> [ keyfun <> { hash-arg }* ])
@@ -29123,13 +29120,7 @@ the internal hash table used by
.codn unique .
The arguments are like those of the
.code hash
-function. In particular, the argument
-.code :equal-based
-causes
-.code unique
-to use
-.code equal
-equality.
+function.
.coNP Function @ tuples
.synb
@@ -31264,7 +31255,7 @@ The function is then memoized.
(defun ensure-memo (sym)
(or (gethash %memo% sym)
- (sethash %memo% sym (hash :equal-based))))
+ (sethash %memo% sym (hash))))
(define-param-expander :memo (param body)
(let* ((memo-parm [param 0..(posq : param)])
@@ -38987,7 +38978,8 @@ and
.coNP Functions @ make-hash and @ hash
.synb
.mets (make-hash < weak-keys < weak-vals << equal-based )
-.mets (hash {:weak-keys | :weak-vals | :equal-based |
+.mets (hash {:weak-keys | :weak-vals |
+.mets \ \ \ \ \ \ :eql-based | :equal-based |
.mets \ \ \ \ \ \ :userdata << obj }*)
.syne
.desc
@@ -39018,17 +39010,42 @@ It is an error to attempt to construct an
.codn equal -based
hash table which has weak keys.
-The hash function provides an alternative interface. It accepts optional
-arguments which are keyword symbols. Any combination of the four symbols
+The
+.code hash
+function provides an alternative interface. It accepts optional
+keyword arguments. The supported keyword symbols are:
.codn :weak-keys ,
.codn :weak-vals ,
-.code :equal-based
+.codn :equal-based ,
+.code :eql-based
and
.code :userdata
-can be specified in any order
-to turn on the corresponding properties in the newly constructed hash table.
-If any of the keywords is not specified, the corresponding property defaults to
-.codn nil .
+which can be specified in any order to turn on the corresponding properties in
+the newly constructed hash table.
+
+Only one of
+.code :equal-based
+and
+.code :eql-based
+may be specified. If specified, then the hash table uses
+.code equal
+or
+.code eql
+equality, respectively, for considering two keys to be the same key.
+If neither is specified, the
+.code hash
+function produces an
+.code :equal-based
+hash table by default.
+
+If
+.code :weak-keys
+is specified, then it automatically implies
+.code :eql-based
+and, therefore,
+.code :equal-based
+may not be specified.
+
If
.code :userdata
is present, it must be followed by an argument value; that value
@@ -62055,6 +62072,39 @@ of these version values, the described behaviors are provided if
is given an argument which is equal or lower. For instance
.code "-C 103"
selects the behaviors described below for version 105, but not those for 102.
+.IP 187
+Until \*(TX 187, hash tables constructed by the
+.code hash
+function were based on
+.code eql
+equality by default; the
+.code :equal-based
+keyword argument had to be specified to override this default, and the
+.code :eql-based
+keyword didn't exist. Selecting 187 or lower compatibility restores the
+behavior of
+.code eql
+equality being default, and the
+.code :eql-based
+keyword being unrecognized. This affects all functions which implicitly
+rely on
+.codn hash ,
+those being :
+.codn uniq ,
+.codn unique ,
+and
+.codn group-by .
+In spite of these changes, the printed representation of hash tables continues
+to use the
+.code :equal-based
+keyword to indicate hash tables based on
+.code equal
+and its absence to indicate
+.code eql
+equality. The new
+.code :eql-based
+keyword may be used in hash literals (unless 187 compatibility is
+in effect, in which case it is ignored).
.IP 184
A value of 184 or lower switches to the old implementation of the
.code op