Skip to content

Commit e3cdaab

Browse files
committed
buffer: make indexOf faster
Improve the performance of IndexOf() for Buffer#indexOf() searches.
1 parent 78581c8 commit e3cdaab

File tree

2 files changed

+31
-6
lines changed

2 files changed

+31
-6
lines changed

src/node_buffer.cc

+30-6
Original file line numberDiff line numberDiff line change
@@ -607,13 +607,37 @@ int32_t IndexOf(const char* haystack,
607607
const char* needle,
608608
size_t n_length) {
609609
CHECK_GE(h_length, n_length);
610-
// TODO(trevnorris): Implement Boyer-Moore string search algorithm.
611-
for (size_t i = 0; i < h_length - n_length + 1; i++) {
612-
if (haystack[i] == needle[0]) {
613-
if (memcmp(haystack + i, needle, n_length) == 0)
614-
return i;
615-
}
610+
611+
const char* inc;
612+
const void* ptr;
613+
614+
// Guard against zero length searches
615+
if (h_length == 0 || n_length == 0 || n_length > h_length)
616+
return -1;
617+
618+
// Single character case
619+
if (n_length == 1) {
620+
ptr = memchr(haystack, *needle, h_length);
621+
if (ptr == nullptr)
622+
return -1;
623+
return static_cast<int32_t>(static_cast<const char*>(ptr) - haystack);
616624
}
625+
626+
// Do multicharacter string match
627+
inc = haystack;
628+
do {
629+
ptr = memchr(inc, *needle, h_length - static_cast<int32_t>(inc - haystack));
630+
if (ptr != nullptr) {
631+
const void* ptr2 = memmem(inc,
632+
h_length - static_cast<int32_t>(inc - haystack),
633+
needle,
634+
n_length);
635+
if (ptr2 != nullptr)
636+
return static_cast<int32_t>(static_cast<const char*>(ptr2) - haystack);
637+
inc = static_cast<const char*>(ptr);
638+
}
639+
} while (ptr != nullptr);
640+
617641
return -1;
618642
}
619643

test/parallel/test-buffer-indexof.js

+1
Original file line numberDiff line numberDiff line change
@@ -26,6 +26,7 @@ assert.equal(b.indexOf('bc', -5), 1);
2626
assert.equal(b.indexOf('bc', NaN), 1);
2727
assert.equal(b.indexOf('bc', -Infinity), 1);
2828
assert.equal(b.indexOf('bc', Infinity), -1);
29+
assert.equal(b.indexOf('cd'), 2);
2930
assert.equal(b.indexOf('f'), b.length - 1);
3031
assert.equal(b.indexOf('z'), -1);
3132
assert.equal(b.indexOf(''), -1);

0 commit comments

Comments
 (0)