return 0;
}
+static __attribute__((unused))
+int strerror_r(int errnum, char *buf, size_t buflen)
+{
+ if (buflen < 18)
+ return ERANGE;
+
+ __builtin_memcpy(buf, "errno=", 6);
+ i64toa_r(errnum, buf + 6);
+ return 0;
+}
+
static __attribute__((unused))
const char *strerror(int errno)
{
- static char buf[18] = "errno=";
+ static char buf[18];
+ char *b = buf;
+
+ /* Force gcc to use 'register offset' to access buf[]. */
+ _NOLIBC_OPTIMIZER_HIDE_VAR(b);
- i64toa_r(errno, &buf[6]);
+ /* Use strerror_r() to avoid having the only .data in small programs. */
+ strerror_r(errno, b, sizeof(buf));
- return buf;
+ return b;
}
#endif /* _NOLIBC_STDIO_H */
return 0;
}
-int test_strerror(void)
-{
- char buf[100];
- ssize_t ret;
-
- memset(buf, 'A', sizeof(buf));
-
- errno = EINVAL;
- ret = snprintf(buf, sizeof(buf), "%m");
- if (is_nolibc) {
- if (ret < 6 || memcmp(buf, "errno=", 6))
- return 1;
- }
-
- return 0;
-}
-
static int test_printf_error(void)
{
int fd, ret, saved_errno;
CASE_TEST(string_width); EXPECT_VFPRINTF(1, " 1", "%10s", "1"); break;
CASE_TEST(number_width); EXPECT_VFPRINTF(1, " 1", "%10d", 1); break;
CASE_TEST(width_trunc); EXPECT_VFPRINTF(1, " 1", "%25d", 1); break;
+ CASE_TEST(errno); errno = 22; EXPECT_VFPRINTF(is_nolibc, "errno=22", "%m"); break;
+ CASE_TEST(errno-neg); errno = -22; EXPECT_VFPRINTF(is_nolibc, " errno=-22", "%12m"); break;
CASE_TEST(scanf); EXPECT_ZR(1, test_scanf()); break;
- CASE_TEST(strerror); EXPECT_ZR(1, test_strerror()); break;
CASE_TEST(printf_error); EXPECT_ZR(1, test_printf_error()); break;
case __LINE__:
return ret; /* must be last */