/* * Copyright 2019 Simon Arlott * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ #include #include #include #include NativeConsole Serial; /* millis() on C++ native could be for non-Arduino millis(): #include auto start = std::chrono::system_clock::now(); auto end = std::chrono::system_clock::now(); diff = std::chrono::duration_cast(end - start).count(); */ static unsigned long __millis = 0; static bool __output_pins[256]; static int __output_level[256]; int main(int argc __attribute__((unused)), char * argv[] __attribute__((unused))) { memset(__output_pins, 0, sizeof(__output_pins)); memset(__output_level, 0, sizeof(__output_level)); setup(); // loop(); // run once static unsigned long __cycles = 0; while (millis() <= 10 * 1000 && __cycles++ <= 10 * 1000) { loop(); } return 0; } unsigned long millis() { return __millis; } void delay(unsigned long millis) { __millis += millis; } void yield(void) { } int snprintf_P(char * str, size_t size, const char * format, ...) { va_list ap; va_start(ap, format); int ret = vsnprintf_P(str, size, format, ap); va_end(ap); return ret; } int vsnprintf_P(char * str, size_t size, const char * format, va_list ap) { std::string native_format; char previous = 0; for (size_t i = 0; i < strlen(format); i++) { char c = format[i]; // This would be a lot easier if the ESP8266 platform // simply read all strings with 32-bit accesses instead // of repurposing %S (wchar_t). if (previous == '%' && c == 'S') { c = 's'; } native_format += c; previous = c; } return vsnprintf(str, size, native_format.c_str(), ap); } void pinMode(uint8_t pin, uint8_t mode) { __output_pins[pin] = (mode == OUTPUT); } void digitalWrite(uint8_t pin, uint8_t value) { __output_level[pin] = (value == HIGH) ? HIGH : LOW; } int digitalRead(uint8_t pin) { if (__output_pins[pin]) { return __output_level[pin]; } else if (pin & 1) { return HIGH; } else { return LOW; } } /* * Copy string src to buffer dst of size dsize. At most dsize-1 * chars will be copied. Always NUL terminates (unless dsize == 0). * Returns strlen(src); if retval >= dsize, truncation occurred. * * https://github.com/freebsd/freebsd/blob/master/sys/libkern/strlcpy.c */ size_t strlcpy(char * __restrict dst, const char * __restrict src, size_t dsize) { const char * osrc = src; size_t nleft = dsize; /* Copy as many bytes as will fit. */ if (nleft != 0) { while (--nleft != 0) { if ((*dst++ = *src++) == '\0') break; } } /* Not enough room in dst, add NUL and traverse rest of src. */ if (nleft == 0) { if (dsize != 0) *dst = '\0'; /* NUL-terminate dst */ while (*src++) ; } return (src - osrc - 1); /* count does not include NUL */ } /* * Appends src to string dst of size siz (unlike strncat, siz is the * full size of dst, not space left). At most siz-1 characters * will be copied. Always NUL terminates (unless siz <= strlen(dst)). * Returns strlen(src) + MIN(siz, strlen(initial dst)). * If retval >= siz, truncation occurred. * * https://github.com/freebsd/freebsd/blob/master/sys/libkern/strlcat.c */ size_t strlcat(char * dst, const char * src, size_t siz) { char * d = dst; const char * s = src; size_t n = siz; size_t dlen; /* Find the end of dst and adjust bytes left but don't go past end */ while (n-- != 0 && *d != '\0') d++; dlen = d - dst; n = siz - dlen; if (n == 0) return (dlen + strlen(s)); while (*s != '\0') { if (n != 1) { *d++ = *s; n--; } s++; } *d = '\0'; return (dlen + (s - src)); /* count does not include NUL */ }