From af7480d0190cb5dcf279a7ddfab320ff084a3471 Mon Sep 17 00:00:00 2001 From: jvoisin Date: Thu, 22 Jun 2023 17:58:58 +0200 Subject: Add tests for stpcpy --- include/string.h | 13 ++++++++++--- tests/Makefile | 7 +++++-- tests/test_stpcpy_overwrite_over.c | 15 +++++++++++++++ tests/test_stpcpy_overwrite_under.c | 15 +++++++++++++++ tests/test_stpcpy_static_write.c | 16 ++++++++++++++++ 5 files changed, 61 insertions(+), 5 deletions(-) create mode 100644 tests/test_stpcpy_overwrite_over.c create mode 100644 tests/test_stpcpy_overwrite_under.c create mode 100644 tests/test_stpcpy_static_write.c diff --git a/include/string.h b/include/string.h index f416a51..f738901 100644 --- a/include/string.h +++ b/include/string.h @@ -85,9 +85,16 @@ __access(write_only, 1) __access(read_only, 2) _FORTIFY_FN(stpcpy) char *stpcpy(char *__d, const char *__s) { - size_t __b = __bos(__d, 0); + size_t __n = strlen(__s) + 1; - if (strlen(__s) + 1 > __b) + /* trap if pointers are overlapping but not if dst == src. + * gcc seems to like to generate code that relies on dst == src */ + if ((__d < __s && __d + __n > __s) || + (__s < __d && __s + __n > __d)) + __builtin_trap(); + + size_t __b = __bos(__d, 0); + if (__n > __b) __builtin_trap(); return __orig_stpcpy(__d, __s); } @@ -129,7 +136,7 @@ _FORTIFY_FN(strcpy) char *strcpy(char *__d, const char *__s) __builtin_trap(); size_t __b = __bos(__d, 0); - if (strlen(__s) + 1 > __b) + if (__n > __b) __builtin_trap(); return __orig_strcpy(__d, __s); } diff --git a/tests/Makefile b/tests/Makefile index 80a4626..6722add 100644 --- a/tests/Makefile +++ b/tests/Makefile @@ -12,13 +12,16 @@ TARGETS=test_memcpy_static_write \ test_memmove_dynamic_read \ test_memset_static_write \ test_memset_dynamic_write \ - test_strcpy_static_write \ + test_stpcpy_overwrite_over \ + test_stpcpy_overwrite_under \ + test_stpcpy_static_write \ test_strcat_static_write \ test_strcpy_overwrite_over \ test_strcpy_overwrite_under \ - test_strncpy_static_write \ + test_strcpy_static_write \ test_strncpy_overwrite_over \ test_strncpy_overwrite_under \ + test_strncpy_static_write \ test_getcwd \ .SILENT: diff --git a/tests/test_stpcpy_overwrite_over.c b/tests/test_stpcpy_overwrite_over.c new file mode 100644 index 0000000..b7cbedc --- /dev/null +++ b/tests/test_stpcpy_overwrite_over.c @@ -0,0 +1,15 @@ +#include "common.h" + +#include + +int main(int argc, char** argv) { + char buffer[9] = {'A', 'A', 'A', 'A', 'B', 'B', 'B', 'B', '\0'}; + puts(buffer); + + CHK_FAIL_START + stpcpy(buffer+1, buffer); + CHK_FAIL_END + + puts(buffer); + return ret; +} diff --git a/tests/test_stpcpy_overwrite_under.c b/tests/test_stpcpy_overwrite_under.c new file mode 100644 index 0000000..2759e5d --- /dev/null +++ b/tests/test_stpcpy_overwrite_under.c @@ -0,0 +1,15 @@ +#include "common.h" + +#include + +int main(int argc, char** argv) { + char buffer[9] = {'A', 'A', 'A', 'A', 'B', 'B', 'B', 'B', '\0'}; + puts(buffer); + + CHK_FAIL_START + stpcpy(buffer-1, buffer); + CHK_FAIL_END + + puts(buffer); + return ret; +} diff --git a/tests/test_stpcpy_static_write.c b/tests/test_stpcpy_static_write.c new file mode 100644 index 0000000..4aa4aee --- /dev/null +++ b/tests/test_stpcpy_static_write.c @@ -0,0 +1,16 @@ +#include "common.h" + +#include + +int main(int argc, char** argv) { + char buffer[8] = {0}; + strcpy(buffer, "1234567"); + puts(buffer); + + CHK_FAIL_START + stpcpy(buffer, "1234567890"); + CHK_FAIL_END + + puts(buffer); + return ret; +} -- cgit v1.3