mirror of
git://git.code.sf.net/p/zsh/code
synced 2025-01-19 11:31:26 +01:00
38973: Optimize indexing array parameters.
% () { for 1 in $prefix/zsh/bin/zsh Src/zsh; do $1 -f -c 'a=( {1..1000000} ); repeat 3 time ( repeat 300 : $a[1] )'; done } ( repeat 300; do; : $a[1]; done; ) 1.68s user 0.01s system 98% cpu 1.718 total ( repeat 300; do; : $a[1]; done; ) 1.69s user 0.01s system 99% cpu 1.710 total ( repeat 300; do; : $a[1]; done; ) 1.69s user 0.01s system 99% cpu 1.714 total ( repeat 300; do; : $a[1]; done; ) 0.00s user 0.01s system 72% cpu 0.022 total ( repeat 300; do; : $a[1]; done; ) 0.00s user 0.01s system 72% cpu 0.022 total ( repeat 300; do; : $a[1]; done; ) 0.01s user 0.01s system 69% cpu 0.023 total
This commit is contained in:
parent
faa163cd5f
commit
1a368bf31f
4 changed files with 55 additions and 5 deletions
|
@ -1,5 +1,8 @@
|
|||
2016-08-01 Daniel Shahaf <d.s@daniel.shahaf.name>
|
||||
|
||||
* 38973: Src/params.c, Src/subst.c, Src/utils.c: Optimize
|
||||
indexing array parameters.
|
||||
|
||||
* 38964: Completion/Unix/Command/_git: _git-config: Complete
|
||||
option names present in the config file.
|
||||
|
||||
|
|
|
@ -2026,7 +2026,7 @@ getstrvalue(Value v)
|
|||
else {
|
||||
if (v->start < 0)
|
||||
v->start += arrlen(ss);
|
||||
s = (v->start >= arrlen(ss) || v->start < 0) ?
|
||||
s = (arrlen_le(ss, v->start) || v->start < 0) ?
|
||||
(char *) hcalloc(1) : ss[v->start];
|
||||
}
|
||||
return s;
|
||||
|
|
15
Src/subst.c
15
Src/subst.c
|
@ -2548,12 +2548,19 @@ paramsubst(LinkList l, LinkNode n, char **str, int qt, int pf_flags,
|
|||
* necessary joining of arrays until this point
|
||||
* to avoid the multsub() horror.
|
||||
*/
|
||||
int tmplen = arrlen(v->pm->gsu.a->getfn(v->pm));
|
||||
|
||||
if (v->start < 0)
|
||||
/* arrlen() is expensive, so only compute it if needed. */
|
||||
int tmplen = -1;
|
||||
|
||||
if (v->start < 0) {
|
||||
tmplen = arrlen(v->pm->gsu.a->getfn(v->pm));
|
||||
v->start += tmplen + ((v->flags & VALFLAG_INV) ? 1 : 0);
|
||||
if (!(v->flags & VALFLAG_INV) &&
|
||||
(v->start >= tmplen || v->start < 0))
|
||||
}
|
||||
if (!(v->flags & VALFLAG_INV))
|
||||
if (v->start < 0 ||
|
||||
(tmplen != -1
|
||||
? v->start >= tmplen
|
||||
: arrlen_le(v->pm->gsu.a->getfn(v->pm), v->start)))
|
||||
vunset = 1;
|
||||
}
|
||||
if (!vunset) {
|
||||
|
|
40
Src/utils.c
40
Src/utils.c
|
@ -2280,6 +2280,46 @@ arrlen(char **s)
|
|||
return count;
|
||||
}
|
||||
|
||||
/* Return TRUE iff arrlen(s) >= lower_bound, but more efficiently. */
|
||||
|
||||
/**/
|
||||
mod_export char
|
||||
arrlen_ge(char **s, unsigned lower_bound)
|
||||
{
|
||||
while (lower_bound--)
|
||||
if (!*s++)
|
||||
return 0 /* FALSE */;
|
||||
|
||||
return 1 /* TRUE */;
|
||||
}
|
||||
|
||||
/* Return TRUE iff arrlen(s) > lower_bound, but more efficiently. */
|
||||
|
||||
/**/
|
||||
mod_export char
|
||||
arrlen_gt(char **s, unsigned lower_bound)
|
||||
{
|
||||
return arrlen_ge(s, 1+lower_bound);
|
||||
}
|
||||
|
||||
/* Return TRUE iff arrlen(s) <= upper_bound, but more efficiently. */
|
||||
|
||||
/**/
|
||||
mod_export char
|
||||
arrlen_le(char **s, unsigned upper_bound)
|
||||
{
|
||||
return arrlen_lt(s, 1+upper_bound);
|
||||
}
|
||||
|
||||
/* Return TRUE iff arrlen(s) < upper_bound, but more efficiently. */
|
||||
|
||||
/**/
|
||||
mod_export char
|
||||
arrlen_lt(char **s, unsigned upper_bound)
|
||||
{
|
||||
return !arrlen_ge(s, upper_bound);
|
||||
}
|
||||
|
||||
/* Skip over a balanced pair of parenthesis. */
|
||||
|
||||
/**/
|
||||
|
|
Loading…
Reference in a new issue