summaryrefslogtreecommitdiff
path: root/Src/params.c
diff options
context:
space:
mode:
authorPeter Stephenson <p.w.stephenson@ntlworld.com>2016-11-20 19:41:52 +0000
committerPeter Stephenson <p.w.stephenson@ntlworld.com>2016-11-20 19:41:52 +0000
commit368884a3aacd9852ae1310346695dbf2240e863b (patch)
treea3e717b3466e1a4e365fda288d737733d80ce68a /Src/params.c
parenta2426747dabd72b8bf6fef73e4c1b229c7325385 (diff)
downloadzsh-368884a3aacd9852ae1310346695dbf2240e863b.tar.gz
zsh-368884a3aacd9852ae1310346695dbf2240e863b.zip
39995 (from 39977): Optimise string parameter assignment.
If setter is the standard one and string length is unchnaged we can copy into place.
Diffstat (limited to 'Src/params.c')
-rw-r--r--Src/params.c39
1 files changed, 31 insertions, 8 deletions
diff --git a/Src/params.c b/Src/params.c
index 9d741cb7b..a79debc93 100644
--- a/Src/params.c
+++ b/Src/params.c
@@ -2436,9 +2436,10 @@ assignstrvalue(Value v, char *val, int flags)
v->pm->width = strlen(val);
} else {
char *z, *x;
- int zlen;
+ int zlen, vlen, newsize;
- z = dupstring_glen(v->pm->gsu.s->getfn(v->pm), (unsigned*) &zlen);
+ z = v->pm->gsu.s->getfn(v->pm);
+ zlen = strlen(z);
if ((v->flags & VALFLAG_INV) && unset(KSHARRAYS))
v->start--, v->end--;
@@ -2469,12 +2470,34 @@ assignstrvalue(Value v, char *val, int flags)
}
else if (v->end > zlen)
v->end = zlen;
- x = (char *) zalloc(v->start + strlen(val) + zlen - v->end + 1);
- strncpy(x, z, v->start);
- strcpy(x + v->start, val);
- strcat(x + v->start, z + v->end);
- v->pm->gsu.s->setfn(v->pm, x);
- zsfree(val);
+
+ vlen = strlen(val);
+ /* Characters preceding start index +
+ characters of what is assigned +
+ characters following end index */
+ newsize = v->start + vlen + (zlen - v->end);
+
+ /* Does new size differ? */
+ if (newsize != zlen || v->pm->gsu.s->setfn != strsetfn) {
+ x = (char *) zalloc(newsize + 1);
+ strncpy(x, z, v->start);
+ strcpy(x + v->start, val);
+ strcat(x + v->start, z + v->end);
+ v->pm->gsu.s->setfn(v->pm, x);
+ } else {
+ Param pm = v->pm;
+ /* Size doesn't change, can limit actions to only
+ * overwriting bytes in already allocated string */
+ strncpy(z + v->start, val, vlen);
+ /* Implement remainder of strsetfn */
+ if (!(pm->node.flags & PM_HASHELEM) &&
+ ((pm->node.flags & PM_NAMEDDIR) ||
+ isset(AUTONAMEDIRS))) {
+ pm->node.flags |= PM_NAMEDDIR;
+ adduserdir(pm->node.nam, z, 0, 0);
+ }
+ }
+ zsfree(val);
}
break;
case PM_INTEGER: