MiniStdLib.cpp 2.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105
  1. /*
  2. * Copyright (c) 2018-2020, Andreas Kling <kling@serenityos.org>
  3. *
  4. * SPDX-License-Identifier: BSD-2-Clause
  5. */
  6. #include <Kernel/StdLib.h>
  7. extern "C" {
  8. void* memcpy(void* dest_ptr, const void* src_ptr, size_t n)
  9. {
  10. #if ARCH(I386) || ARCH(X86_64)
  11. size_t dest = (size_t)dest_ptr;
  12. size_t src = (size_t)src_ptr;
  13. // FIXME: Support starting at an unaligned address.
  14. if (!(dest & 0x3) && !(src & 0x3) && n >= 12) {
  15. size_t size_ts = n / sizeof(size_t);
  16. # if ARCH(I386)
  17. asm volatile(
  18. "rep movsl\n"
  19. : "=S"(src), "=D"(dest)
  20. : "S"(src), "D"(dest), "c"(size_ts)
  21. : "memory");
  22. # else
  23. asm volatile(
  24. "rep movsq\n"
  25. : "=S"(src), "=D"(dest)
  26. : "S"(src), "D"(dest), "c"(size_ts)
  27. : "memory");
  28. # endif
  29. n -= size_ts * sizeof(size_t);
  30. if (n == 0)
  31. return dest_ptr;
  32. }
  33. asm volatile(
  34. "rep movsb\n" ::"S"(src), "D"(dest), "c"(n)
  35. : "memory");
  36. #else
  37. u8* pd = (u8*)dest_ptr;
  38. u8 const* ps = (u8 const*)src_ptr;
  39. for (; n--;)
  40. *pd++ = *ps++;
  41. #endif
  42. return dest_ptr;
  43. }
  44. void* memmove(void* dest, const void* src, size_t n)
  45. {
  46. if (dest < src)
  47. return memcpy(dest, src, n);
  48. u8* pd = (u8*)dest;
  49. const u8* ps = (const u8*)src;
  50. for (pd += n, ps += n; n--;)
  51. *--pd = *--ps;
  52. return dest;
  53. }
  54. void* memset(void* dest_ptr, int c, size_t n)
  55. {
  56. #if ARCH(I386) || ARCH(X86_64)
  57. size_t dest = (size_t)dest_ptr;
  58. // FIXME: Support starting at an unaligned address.
  59. if (!(dest & 0x3) && n >= 12) {
  60. size_t size_ts = n / sizeof(size_t);
  61. size_t expanded_c = explode_byte((u8)c);
  62. # if ARCH(I386)
  63. asm volatile(
  64. "rep stosl\n"
  65. : "=D"(dest)
  66. : "D"(dest), "c"(size_ts), "a"(expanded_c)
  67. : "memory");
  68. # else
  69. asm volatile(
  70. "rep stosq\n"
  71. : "=D"(dest)
  72. : "D"(dest), "c"(size_ts), "a"(expanded_c)
  73. : "memory");
  74. # endif
  75. n -= size_ts * sizeof(size_t);
  76. if (n == 0)
  77. return dest_ptr;
  78. }
  79. asm volatile(
  80. "rep stosb\n"
  81. : "=D"(dest), "=c"(n)
  82. : "0"(dest), "1"(n), "a"(c)
  83. : "memory");
  84. #else
  85. u8* pd = (u8*)dest_ptr;
  86. for (; n--;)
  87. *pd++ = c;
  88. #endif
  89. return dest_ptr;
  90. }
  91. size_t strlen(const char* str)
  92. {
  93. size_t len = 0;
  94. while (*(str++))
  95. ++len;
  96. return len;
  97. }
  98. }