diff options
author | Toni <matzeton@googlemail.com> | 2024-05-08 11:38:53 +0200 |
---|---|---|
committer | GitHub <noreply@github.com> | 2024-05-08 11:38:53 +0200 |
commit | e9dc035c5ca8e8e9012c76d5e2bdb9c085d6664a (patch) | |
tree | a348f5ceafba2ef93f609f89b2f0c4d47695cd00 /src | |
parent | 2b4e2f9c9a5befc011c5743fee29dafa273ad6cb (diff) |
Added optimized memmem/strlcpy version (#2424)
* credits goes to Vladimir Gavrilov
Signed-off-by: Toni Uhlig <matzeton@googlemail.com>
Diffstat (limited to 'src')
-rw-r--r-- | src/include/ndpi_api.h | 45 | ||||
-rw-r--r-- | src/lib/ndpi_main.c | 49 |
2 files changed, 54 insertions, 40 deletions
diff --git a/src/include/ndpi_api.h b/src/include/ndpi_api.h index 5fbd064fc..3132e139b 100644 --- a/src/include/ndpi_api.h +++ b/src/include/ndpi_api.h @@ -2235,30 +2235,41 @@ extern "C" { const char *ndpi_lru_cache_idx_to_name(lru_cache_type idx); /** - * Searches for a subsequence ('needle') within a block of memory ('haystack'). + * @brief Finds the first occurrence of the sequence `needle` in the array + * `haystack`. * - * @par haystack = pointer to the block of memory where the search is performed - * @par haystack_len = length of the haystack block of memory - * @par needle = pointer to the memory block representing the needle to search for - * @par needle_len = length of the needle memory block - * - * @return Pointer to the beginning of the needle within the haystack if found; - * otherwise, NULL. + * This function searches for the first occurrence of the sequence `needle` of + * length `needle_len` in the array `haystack` of length `haystack_len`. If + * `haystack` or `needle` is `NULL`, or `haystack_len` is less than + * `needle_len`, or `needle_len` is 0, the function returns `NULL`. + * + * For optimization, if `needle_len` is 1, the `memchr` function is used. + * + * @param haystack Pointer to the array in which the search is performed. + * @param haystack_len Length of the `haystack` array. + * @param needle Pointer to the array to be searched for in `haystack`. + * @param needle_len Length of the `needle` array. + * @return Pointer to the first occurrence of `needle` in `haystack` or `NULL` + * if `needle` is not found. */ void* ndpi_memmem(const void* haystack, size_t haystack_len, const void* needle, size_t needle_len); /** - * Copies up to -dst_len - 1- characters from -src- to -dst-, ensuring the result is - * null-terminated. Measures -src- length and handles potential null pointers. - * - * @par dst = destination buffer - * @par src = source string - * @par dst_len = size of the destination buffer, includes the null terminator - * - * @return Length of `src`. If greater or equal to -dst_len-, -dst- has been truncated. + * @brief Copies src string to dst buffer with length limit + * + * Copies the string src into dst buffer, limiting the copy length by dst_len. + * Handles both null-terminated and non null-terminated strings based on + * src_len. Ensures null-termination in dst if dst_len > 0. + * + * @param dst Destination buffer + * @param src Source string + * @param dst_len Size of dst buffer + * @param src_len Length of src string + * + * @return Length of src string */ - size_t ndpi_strlcpy(char* dst, const char* src, size_t dst_len); + size_t ndpi_strlcpy(char* dst, const char* src, size_t dst_len, size_t src_len); #ifdef __cplusplus } diff --git a/src/lib/ndpi_main.c b/src/lib/ndpi_main.c index c41bb1e02..ca7c5bfc9 100644 --- a/src/lib/ndpi_main.c +++ b/src/lib/ndpi_main.c @@ -11376,45 +11376,48 @@ char *ndpi_dump_config(struct ndpi_detection_module_struct *ndpi_str, void* ndpi_memmem(const void* haystack, size_t haystack_len, const void* needle, size_t needle_len) { - if (!haystack || !needle) { + if (!haystack || !needle || haystack_len < needle_len || needle_len == 0) { return NULL; } - if (needle_len > haystack_len) { - return NULL; + if (needle_len == 1) { + return (void *)memchr(haystack, *(const u_int8_t *)needle, haystack_len); } - if ((size_t)(const u_int8_t*)haystack+haystack_len < haystack_len) { - return NULL; - } + const u_int8_t *current = (const u_int8_t *)haystack; + const u_int8_t *haystack_end = (const u_int8_t *)haystack + haystack_len; - if (needle_len == 1) { - return memchr(haystack, *(int*)needle, haystack_len); - } + while (current <= haystack_end - needle_len) { + /* Find the first occurrence of the first character from the needle */ + current = (const u_int8_t *)memchr(current, *(const u_int8_t *)needle, + haystack_end - current); + + if (!current) { + return NULL; + } + + /* Check the rest of the needle for a match */ + if ((current + needle_len <= haystack_end) && + (memcmp(current, needle, needle_len) == 0)) { + return (void *)current; + } - const u_int8_t* h = NULL; - const u_int8_t* n = (const u_int8_t*)needle; - for (h = haystack; h <= (const u_int8_t*)haystack+haystack_len-needle_len; ++h) { - if (*h == n[0] && !memcmp(h, needle, needle_len)) - return (void*)h; + /* Shift one character to the right for the next search */ + current++; } return NULL; } -size_t ndpi_strlcpy(char *dst, const char* src, size_t dst_len) +size_t ndpi_strlcpy(char *dst, const char* src, size_t dst_len, size_t src_len) { - if (!dst || !src) { + if (!dst || !src || dst_len == 0) { return 0; } - size_t src_len = strlen(src); - - if (dst_len != 0) { - size_t len = (src_len < dst_len - 1) ? src_len : dst_len - 1; - memcpy(dst, src, len); - dst[len] = '\0'; - } + size_t copy_len = ndpi_min(src_len, dst_len - 1); + memmove(dst, src, copy_len); + dst[copy_len] = '\0'; return src_len; } |