luau/VM/src
Arseny Kapoulkine f5303b3dd7
Make table.concat faster (#1243)
table.concat is idiomatic and should be the fastest way to concatenate
all table array elements together, but apparently you can beat it by
using `string.format`, `string.rep` and `table.unpack`:

```lua
string.format(string.rep("%*", #t), table.unpack(t))
```

... this just won't do, so we should fix table.concat performance.

The deficit comes from two places:

- rawgeti overhead followed by other stack accesses, all to extract a
string from what is almost always an in-bounds array lookup
- addlstring overhead in case separator is empty (extra function calls)

This change fixes this by using a fast path for in-bounds array lookup
for a string. Note that `table.concat` also supports numbers (these need
to be converted to strings which is a little cumbersome and has innate
overhead), and out-of-bounds accesses*. In these cases we fall back to
the old implementation.

To trigger out-of-bounds accesses, you need to skip the past-array-end
element (which is nil per array invariant), but this is achievable
because table.concat supports offset+length arguments. This should
almost never come up in practice but the per-element branches et al are
fairly cheap compared to the eventual string copy/alloc anyway.

This change makes table.concat ~2x faster when the separator is empty;
the table.concat benchmark shows +40% gains but it uses a variety of
string separators of different lengths so it doesn't get the full
benefit from this change.

---------

Co-authored-by: vegorov-rbx <75688451+vegorov-rbx@users.noreply.github.com>
2024-04-29 05:19:01 -07:00
..
lapi.cpp Add tagged lightuserdata (#1087) 2023-12-14 15:05:51 -08:00
lapi.h Sync to upstream/release/501 (#20) 2021-11-01 14:52:34 -07:00
laux.cpp Sync to upstream/release/600 (#1076) 2023-10-20 18:10:30 -07:00
lbaselib.cpp Sync to upstream/release/539 (#625) 2022-08-04 15:35:33 -07:00
lbitlib.cpp Sync to upstream/release/608 (#1145) 2024-01-12 14:25:27 -08:00
lbuffer.cpp Sync to upstream/release/600 (#1076) 2023-10-20 18:10:30 -07:00
lbuffer.h Sync to upstream/release/600 (#1076) 2023-10-20 18:10:30 -07:00
lbuflib.cpp Sync to upstream/release/608 (#1145) 2024-01-12 14:25:27 -08:00
lbuiltins.cpp Sync to upstream/release/602 (#1089) 2023-11-03 16:45:04 -07:00
lbuiltins.h Sync to upstream/release/551 (#727) 2022-10-28 03:37:29 -07:00
lbytecode.h Sync to upstream/release/529 (#505) 2022-05-26 15:08:16 -07:00
lcommon.h Sync to upstream/release/539 (#625) 2022-08-04 15:35:33 -07:00
lcorolib.cpp Sync to upstream/release/544 (#669) 2022-09-08 15:14:25 -07:00
ldblib.cpp Sync to upstream/release/621 (#1229) 2024-04-12 10:18:49 -07:00
ldebug.cpp Sync to upstream/release/619 (#1218) 2024-03-30 16:14:44 -07:00
ldebug.h Sync to upstream/release/589 (#1000) 2023-08-04 12:18:54 -07:00
ldo.cpp Sync to upstream/release/617 (#1204) 2024-03-15 16:37:39 -07:00
ldo.h Sync to upstream/release/577 (#934) 2023-05-19 12:37:30 -07:00
lfunc.cpp Sync to upstream/release/623 (#1236) 2024-04-25 15:26:09 -07:00
lfunc.h Sync to upstream/release/545 (#674) 2022-09-15 15:38:17 -07:00
lgc.cpp Sync to upstream/release/623 (#1236) 2024-04-25 15:26:09 -07:00
lgc.h Sync to upstream/release/600 (#1076) 2023-10-20 18:10:30 -07:00
lgcdebug.cpp Sync to upstream/release/621 (#1229) 2024-04-12 10:18:49 -07:00
linit.cpp Sync to upstream/release/611 (#1160) 2024-02-02 13:32:42 -08:00
lmathlib.cpp Sync to upstream/release/595 (#1044) 2023-09-15 10:26:59 -07:00
lmem.cpp Sync to upstream/release/622 (#1232) 2024-04-19 14:48:02 -07:00
lmem.h Sync to upstream/release/622 (#1232) 2024-04-19 14:48:02 -07:00
lnumprint.cpp Sync to upstream/release/615 (#1175) 2024-03-01 10:45:26 -08:00
lnumutils.h Sync to upstream/release/593 (#1024) 2023-09-01 10:58:27 -07:00
lobject.cpp Sync to upstream/release/615 (#1175) 2024-03-01 10:45:26 -08:00
lobject.h Sync to upstream/release/623 (#1236) 2024-04-25 15:26:09 -07:00
loslib.cpp Sync to upstream/release/603 (#1097) 2023-11-10 13:10:07 -08:00
lperf.cpp Sync to upstream/release/609 (#1150) 2024-01-19 10:04:46 -08:00
lstate.cpp Sync to upstream/release/609 (#1150) 2024-01-19 10:04:46 -08:00
lstate.h Sync to upstream/release/617 (#1204) 2024-03-15 16:37:39 -07:00
lstring.cpp Sync to upstream/release/611 (#1160) 2024-02-02 13:32:42 -08:00
lstring.h Sync to upstream/release/539 (#625) 2022-08-04 15:35:33 -07:00
lstrlib.cpp Sync to upstream/release/615 (#1175) 2024-03-01 10:45:26 -08:00
ltable.cpp Sync to upstream/release/572 (#899) 2023-04-14 11:06:22 -07:00
ltable.h Sync to upstream/release/538 (#616) 2022-07-28 21:24:07 -07:00
ltablib.cpp Make table.concat faster (#1243) 2024-04-29 05:19:01 -07:00
ltm.cpp Sync to upstream/release/615 (#1175) 2024-03-01 10:45:26 -08:00
ltm.h Sync to upstream/release/593 (#1024) 2023-09-01 10:58:27 -07:00
ludata.cpp Add lua_getuserdatadtor (#870) 2023-04-11 12:46:55 -07:00
ludata.h Sync to upstream/release/539 (#625) 2022-08-04 15:35:33 -07:00
lutf8lib.cpp Sync to upstream/release/608 (#1145) 2024-01-12 14:25:27 -08:00
lvm.h Sync to upstream/release/581 (#958) 2023-06-16 10:35:18 -07:00
lvmexecute.cpp Sync to upstream/release/615 (#1175) 2024-03-01 10:45:26 -08:00
lvmload.cpp Sync to upstream/release/623 (#1236) 2024-04-25 15:26:09 -07:00
lvmutils.cpp Sync to upstream/release/615 (#1175) 2024-03-01 10:45:26 -08:00