summaryrefslogtreecommitdiffstatshomepage
path: root/lib
diff options
context:
space:
mode:
authorPaul Sokolovsky <pfalcon@users.sourceforge.net>2015-04-05 21:44:20 +0300
committerPaul Sokolovsky <pfalcon@users.sourceforge.net>2015-04-05 21:57:55 +0300
commit3a84c8b58d7c87785bf1ae48eb41c91ecde9fc1c (patch)
tree62129d5f488f0680cb6a165a4006c391da18195e /lib
parente5c4362a98cf31d909476d9f06825228f05a2c82 (diff)
downloadmicropython-3a84c8b58d7c87785bf1ae48eb41c91ecde9fc1c.tar.gz
micropython-3a84c8b58d7c87785bf1ae48eb41c91ecde9fc1c.zip
string0.c: Move from stmhal/ to lib/.
Diffstat (limited to 'lib')
-rw-r--r--lib/libc/string0.c205
1 files changed, 205 insertions, 0 deletions
diff --git a/lib/libc/string0.c b/lib/libc/string0.c
new file mode 100644
index 0000000000..b73a5e9ea9
--- /dev/null
+++ b/lib/libc/string0.c
@@ -0,0 +1,205 @@
+/*
+ * This file is part of the Micro Python project, http://micropython.org/
+ *
+ * The MIT License (MIT)
+ *
+ * Copyright (c) 2013, 2014 Damien P. George
+ *
+ * Permission is hereby granted, free of charge, to any person obtaining a copy
+ * of this software and associated documentation files (the "Software"), to deal
+ * in the Software without restriction, including without limitation the rights
+ * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
+ * copies of the Software, and to permit persons to whom the Software is
+ * furnished to do so, subject to the following conditions:
+ *
+ * The above copyright notice and this permission notice shall be included in
+ * all copies or substantial portions of the Software.
+ *
+ * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
+ * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
+ * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
+ * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
+ * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
+ * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
+ * THE SOFTWARE.
+ */
+
+#include <stdint.h>
+#include "std.h"
+
+#define likely(x) __builtin_expect((x), 1)
+
+void *memcpy(void *dst, const void *src, size_t n) {
+ if (likely(!(((uint32_t)dst) & 3) && !(((uint32_t)src) & 3))) {
+ // pointers aligned
+ uint32_t *d = dst;
+ const uint32_t *s = src;
+
+ // copy words first
+ for (size_t i = (n >> 2); i; i--) {
+ *d++ = *s++;
+ }
+
+ if (n & 2) {
+ // copy half-word
+ *(uint16_t*)d = *(const uint16_t*)s;
+ d = (uint32_t*)((uint16_t*)d + 1);
+ s = (const uint32_t*)((const uint16_t*)s + 1);
+ }
+
+ if (n & 1) {
+ // copy byte
+ *((uint8_t*)d) = *((const uint8_t*)s);
+ }
+ } else {
+ // unaligned access, copy bytes
+ uint8_t *d = dst;
+ const uint8_t *s = src;
+
+ for (; n; n--) {
+ *d++ = *s++;
+ }
+ }
+
+ return dst;
+}
+
+void *memmove(void *dest, const void *src, size_t n) {
+ if (src < dest && (uint8_t*)dest < (const uint8_t*)src + n) {
+ // need to copy backwards
+ uint8_t *d = (uint8_t*)dest + n - 1;
+ const uint8_t *s = (const uint8_t*)src + n - 1;
+ for (; n > 0; n--) {
+ *d-- = *s--;
+ }
+ return dest;
+ } else {
+ // can use normal memcpy
+ return memcpy(dest, src, n);
+ }
+}
+
+void *memset(void *s, int c, size_t n) {
+ if (c == 0 && ((uint32_t)s & 3) == 0) {
+ // aligned store of 0
+ uint32_t *s32 = s;
+ for (size_t i = n >> 2; i > 0; i--) {
+ *s32++ = 0;
+ }
+ if (n & 2) {
+ *((uint16_t*)s32) = 0;
+ s32 = (uint32_t*)((uint16_t*)s32 + 1);
+ }
+ if (n & 1) {
+ *((uint8_t*)s32) = 0;
+ }
+ } else {
+ uint8_t *s2 = s;
+ for (; n > 0; n--) {
+ *s2++ = c;
+ }
+ }
+ return s;
+}
+
+int memcmp(const char *s1, const char *s2, size_t n) {
+ while (n--) {
+ char c1 = *s1++;
+ char c2 = *s2++;
+ if (c1 < c2) return -1;
+ else if (c1 > c2) return 1;
+ }
+ return 0;
+}
+
+size_t strlen(const char *str) {
+ int len = 0;
+ for (const char *s = str; *s; s++) {
+ len += 1;
+ }
+ return len;
+}
+
+int strcmp(const char *s1, const char *s2) {
+ while (*s1 && *s2) {
+ char c1 = *s1++; // XXX UTF8 get char, next char
+ char c2 = *s2++; // XXX UTF8 get char, next char
+ if (c1 < c2) return -1;
+ else if (c1 > c2) return 1;
+ }
+ if (*s2) return -1;
+ else if (*s1) return 1;
+ else return 0;
+}
+
+int strncmp(const char *s1, const char *s2, size_t n) {
+ while (*s1 && *s2 && n > 0) {
+ char c1 = *s1++; // XXX UTF8 get char, next char
+ char c2 = *s2++; // XXX UTF8 get char, next char
+ n--;
+ if (c1 < c2) return -1;
+ else if (c1 > c2) return 1;
+ }
+ if (n == 0) return 0;
+ else if (*s2) return -1;
+ else if (*s1) return 1;
+ else return 0;
+}
+
+char *strcpy(char *dest, const char *src) {
+ char *d = dest;
+ while (*src) {
+ *d++ = *src++;
+ }
+ *d = '\0';
+ return dest;
+}
+
+// needed because gcc optimises strcpy + strcat to this
+char *stpcpy(char *dest, const char *src) {
+ while (*src) {
+ *dest++ = *src++;
+ }
+ *dest = '\0';
+ return dest;
+}
+
+char *strcat(char *dest, const char *src) {
+ char *d = dest;
+ while (*d) {
+ d++;
+ }
+ while (*src) {
+ *d++ = *src++;
+ }
+ *d = '\0';
+ return dest;
+}
+
+// Public Domain implementation of strchr from:
+// http://en.wikibooks.org/wiki/C_Programming/Strings#The_strchr_function
+char *strchr(const char *s, int c)
+{
+ /* Scan s for the character. When this loop is finished,
+ s will either point to the end of the string or the
+ character we were looking for. */
+ while (*s != '\0' && *s != (char)c)
+ s++;
+ return ((*s == c) ? (char *) s : 0);
+}
+
+
+// Public Domain implementation of strstr from:
+// http://en.wikibooks.org/wiki/C_Programming/Strings#The_strstr_function
+char *strstr(const char *haystack, const char *needle)
+{
+ size_t needlelen;
+ /* Check for the null needle case. */
+ if (*needle == '\0')
+ return (char *) haystack;
+ needlelen = strlen(needle);
+ for (; (haystack = strchr(haystack, *needle)) != 0; haystack++)
+ if (strncmp(haystack, needle, needlelen) == 0)
+ return (char *) haystack;
+ return 0;
+}