summaryrefslogtreecommitdiffstats
path: root/lib.c
diff options
context:
space:
mode:
authorKaz Kylheku <kaz@kylheku.com>2021-06-22 07:15:58 -0700
committerKaz Kylheku <kaz@kylheku.com>2021-06-22 07:15:58 -0700
commit27ece3ffc30437f4a70228b96764e306bd5ed4f1 (patch)
tree1e31550044e236af58c3edce542eeb3efde99569 /lib.c
parent08b4921e616a17ed8492527c6a89c08ccc33ea35 (diff)
downloadtxr-27ece3ffc30437f4a70228b96764e306bd5ed4f1.tar.gz
txr-27ece3ffc30437f4a70228b96764e306bd5ed4f1.tar.bz2
txr-27ece3ffc30437f4a70228b96764e306bd5ed4f1.zip
lib: optimize mismatch, rmismatch for strings.
* lib (mismatch, rmismatch): If the arguments are strings or literals, other than lazy strings, keyfun is identity, and equality is by character identity, the operation can be done with an efficient loop over the wchar_t strings. * tests/012/seq.tl: Tests for string case of mismatch, via starts-with function. Test mismatch via ends-with, and also directly for vectors and strings.
Diffstat (limited to 'lib.c')
-rw-r--r--lib.c49
1 files changed, 46 insertions, 3 deletions
diff --git a/lib.c b/lib.c
index 8df9757a..dce7b2c3 100644
--- a/lib.c
+++ b/lib.c
@@ -10882,10 +10882,29 @@ val mismatch(val left, val right, val testfun_in, val keyfun_in)
case CONS:
case LCONS:
return mismatch(right, left, testfun, keyfun);
- case VEC:
- case LIT:
case STR:
+ case LIT:
+ switch (type(left)) {
+ case STR:
+ case LIT:
+ if (keyfun == identity_f && (testfun == equal_f ||
+ testfun == eql_f ||
+ testfun == eq_f))
+ {
+ const wchar_t *lft = c_str(left), *le = lft;
+ const wchar_t *rgt = c_str(right), *ri = rgt;
+
+ while (*le && *ri && *le == *ri)
+ le++, ri++;
+
+ return if3(*le || *ri, num(le - lft), nil);
+ }
+ default:
+ break;
+ }
+ /* fallthrough */
case LSTR:
+ case VEC:
{
val llen = length(left);
val rlen = length(right);
@@ -10913,6 +10932,7 @@ val mismatch(val left, val right, val testfun_in, val keyfun_in)
val rmismatch(val left, val right, val testfun_in, val keyfun_in)
{
+ val self = lit("rmismatch");
val testfun = default_arg(testfun_in, equal_f);
val keyfun = default_arg(keyfun_in, identity_f);
@@ -10980,9 +11000,32 @@ val rmismatch(val left, val right, val testfun_in, val keyfun_in)
case CONS:
case LCONS:
return rmismatch(right, left, testfun, keyfun);
- case VEC:
case LIT:
case STR:
+ switch (type(left)) {
+ case STR:
+ case LIT:
+ if (keyfun == identity_f && (testfun == equal_f ||
+ testfun == eql_f ||
+ testfun == eq_f))
+ {
+ cnum ll = c_num(length(left), self), li = ll - 1;
+ cnum rl = c_num(length(right), self), ri = rl - 1;
+ const wchar_t *lft = c_str(left);
+ const wchar_t *rgt = c_str(right);
+
+ for (; li >= 0 && ri >= 0; li--, ri--) {
+ if (lft[li] != rgt[ri])
+ break;
+ }
+
+ return if2(li >= 0 || ri >= 0, num(li - ll));
+ }
+ default:
+ break;
+ }
+ /* fallthrough */
+ case VEC:
case LSTR:
{
val llen = length(left);