Merge "Add GNU-compatible strerror_r."
This commit is contained in:
commit
29f06943a1
@ -75,10 +75,12 @@ struct BufferOutputStream {
|
|||||||
len = strlen(data);
|
len = strlen(data);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
total += len;
|
||||||
|
|
||||||
while (len > 0) {
|
while (len > 0) {
|
||||||
int avail = end_ - pos_;
|
int avail = end_ - pos_;
|
||||||
if (avail == 0) {
|
if (avail == 0) {
|
||||||
break;
|
return;
|
||||||
}
|
}
|
||||||
if (avail > len) {
|
if (avail > len) {
|
||||||
avail = len;
|
avail = len;
|
||||||
@ -87,11 +89,10 @@ struct BufferOutputStream {
|
|||||||
pos_ += avail;
|
pos_ += avail;
|
||||||
pos_[0] = '\0';
|
pos_[0] = '\0';
|
||||||
len -= avail;
|
len -= avail;
|
||||||
total += avail;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
int total;
|
size_t total;
|
||||||
|
|
||||||
private:
|
private:
|
||||||
char* buffer_;
|
char* buffer_;
|
||||||
@ -109,18 +110,19 @@ struct FdOutputStream {
|
|||||||
len = strlen(data);
|
len = strlen(data);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
total += len;
|
||||||
|
|
||||||
while (len > 0) {
|
while (len > 0) {
|
||||||
int rc = TEMP_FAILURE_RETRY(write(fd_, data, len));
|
int rc = TEMP_FAILURE_RETRY(write(fd_, data, len));
|
||||||
if (rc == -1) {
|
if (rc == -1) {
|
||||||
break;
|
return;
|
||||||
}
|
}
|
||||||
data += rc;
|
data += rc;
|
||||||
len -= rc;
|
len -= rc;
|
||||||
total += rc;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
int total;
|
size_t total;
|
||||||
|
|
||||||
private:
|
private:
|
||||||
int fd_;
|
int fd_;
|
||||||
|
@ -1,11 +1,16 @@
|
|||||||
/* $OpenBSD: strerror_r.c,v 1.6 2005/08/08 08:05:37 espie Exp $ */
|
/* $OpenBSD: strerror_r.c,v 1.6 2005/08/08 08:05:37 espie Exp $ */
|
||||||
/* Public Domain <marc@snafu.org> */
|
/* Public Domain <marc@snafu.org> */
|
||||||
|
|
||||||
|
// G++ automatically defines _GNU_SOURCE, which then means that <string.h>
|
||||||
|
// gives us the GNU variant.
|
||||||
|
#undef _GNU_SOURCE
|
||||||
|
|
||||||
|
#include <string.h>
|
||||||
|
|
||||||
#include <errno.h>
|
#include <errno.h>
|
||||||
#include <limits.h>
|
#include <limits.h>
|
||||||
#include <signal.h>
|
#include <signal.h>
|
||||||
#include <stdio.h>
|
#include <stdio.h>
|
||||||
#include <string.h>
|
|
||||||
|
|
||||||
#include "private/ErrnoRestorer.h"
|
#include "private/ErrnoRestorer.h"
|
||||||
#include "private/libc_logging.h"
|
#include "private/libc_logging.h"
|
||||||
@ -62,6 +67,12 @@ int strerror_r(int error_number, char* buf, size_t buf_len) {
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
extern "C" char* __gnu_strerror_r(int error_number, char* buf, size_t buf_len) {
|
||||||
|
ErrnoRestorer errno_restorer; // The glibc strerror_r doesn't set errno if it truncates...
|
||||||
|
strerror_r(error_number, buf, buf_len);
|
||||||
|
return buf; // ...and just returns whatever fit.
|
||||||
|
}
|
||||||
|
|
||||||
extern "C" __LIBC_HIDDEN__ const char* __strsignal(int signal_number, char* buf, size_t buf_len) {
|
extern "C" __LIBC_HIDDEN__ const char* __strsignal(int signal_number, char* buf, size_t buf_len) {
|
||||||
const char* signal_name = __strsignal_lookup(signal_number);
|
const char* signal_name = __strsignal_lookup(signal_number);
|
||||||
if (signal_name != NULL) {
|
if (signal_name != NULL) {
|
||||||
|
@ -68,7 +68,11 @@ extern char* strtok(char* __restrict, const char* __restrict);
|
|||||||
extern char* strtok_r(char* __restrict, const char* __restrict, char** __restrict);
|
extern char* strtok_r(char* __restrict, const char* __restrict, char** __restrict);
|
||||||
|
|
||||||
extern char* strerror(int);
|
extern char* strerror(int);
|
||||||
extern int strerror_r(int errnum, char *buf, size_t n);
|
#if defined(__USE_GNU)
|
||||||
|
extern char* strerror_r(int, char*, size_t) __RENAME(__gnu_strerror_r);
|
||||||
|
#else /* POSIX */
|
||||||
|
extern int strerror_r(int, char*, size_t);
|
||||||
|
#endif
|
||||||
|
|
||||||
extern size_t strnlen(const char *, size_t) __purefunc;
|
extern size_t strnlen(const char *, size_t) __purefunc;
|
||||||
extern char* strncat(char* __restrict, const char* __restrict, size_t);
|
extern char* strncat(char* __restrict, const char* __restrict, size_t);
|
||||||
|
@ -95,6 +95,7 @@ libBionicStandardTests_src_files := \
|
|||||||
stdio_ext_test.cpp \
|
stdio_ext_test.cpp \
|
||||||
stdlib_test.cpp \
|
stdlib_test.cpp \
|
||||||
string_test.cpp \
|
string_test.cpp \
|
||||||
|
string_posix_strerror_r_test.cpp \
|
||||||
strings_test.cpp \
|
strings_test.cpp \
|
||||||
stubs_test.cpp \
|
stubs_test.cpp \
|
||||||
sstream_test.cpp \
|
sstream_test.cpp \
|
||||||
|
@ -176,3 +176,15 @@ TEST(libc_logging, lld_LLONG_MIN) {
|
|||||||
GTEST_LOG_(INFO) << "This test does nothing.\n";
|
GTEST_LOG_(INFO) << "This test does nothing.\n";
|
||||||
#endif // __BIONIC__
|
#endif // __BIONIC__
|
||||||
}
|
}
|
||||||
|
|
||||||
|
TEST(libc_logging, buffer_overrun) {
|
||||||
|
#if defined(__BIONIC__)
|
||||||
|
char buf[BUFSIZ];
|
||||||
|
ASSERT_EQ(11, __libc_format_buffer(buf, sizeof(buf), "hello %s", "world"));
|
||||||
|
EXPECT_STREQ("hello world", buf);
|
||||||
|
ASSERT_EQ(11, __libc_format_buffer(buf, 8, "hello %s", "world"));
|
||||||
|
EXPECT_STREQ("hello w", buf);
|
||||||
|
#else // __BIONIC__
|
||||||
|
GTEST_LOG_(INFO) << "This test does nothing.\n";
|
||||||
|
#endif // __BIONIC__
|
||||||
|
}
|
||||||
|
57
tests/string_posix_strerror_r_test.cpp
Normal file
57
tests/string_posix_strerror_r_test.cpp
Normal file
@ -0,0 +1,57 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (C) 2012 The Android Open Source Project
|
||||||
|
*
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
* you may not use this file except in compliance with the License.
|
||||||
|
* You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#undef _GNU_SOURCE
|
||||||
|
|
||||||
|
// Old versions of glibc (like our current host prebuilt sysroot one) have
|
||||||
|
// headers that don't work if you #undef _GNU_SOURCE, which makes it
|
||||||
|
// impossible to build this test.
|
||||||
|
#include <features.h>
|
||||||
|
|
||||||
|
#if !defined(__GLIBC__)
|
||||||
|
#include <string.h>
|
||||||
|
|
||||||
|
#include <errno.h>
|
||||||
|
#include <gtest/gtest.h>
|
||||||
|
|
||||||
|
TEST(string, posix_strerror_r) {
|
||||||
|
char buf[256];
|
||||||
|
|
||||||
|
// Valid.
|
||||||
|
ASSERT_EQ(0, strerror_r(0, buf, sizeof(buf)));
|
||||||
|
ASSERT_STREQ("Success", buf);
|
||||||
|
ASSERT_EQ(0, strerror_r(1, buf, sizeof(buf)));
|
||||||
|
ASSERT_STREQ("Operation not permitted", buf);
|
||||||
|
|
||||||
|
// Invalid.
|
||||||
|
ASSERT_EQ(0, strerror_r(-1, buf, sizeof(buf)));
|
||||||
|
ASSERT_STREQ("Unknown error -1", buf);
|
||||||
|
ASSERT_EQ(0, strerror_r(1234, buf, sizeof(buf)));
|
||||||
|
ASSERT_STREQ("Unknown error 1234", buf);
|
||||||
|
|
||||||
|
// Buffer too small.
|
||||||
|
errno = 0;
|
||||||
|
memset(buf, 0, sizeof(buf));
|
||||||
|
ASSERT_EQ(-1, strerror_r(4567, buf, 2));
|
||||||
|
ASSERT_STREQ("U", buf);
|
||||||
|
// The POSIX strerror_r sets errno to ERANGE (the GNU one doesn't).
|
||||||
|
ASSERT_EQ(ERANGE, errno);
|
||||||
|
}
|
||||||
|
#else
|
||||||
|
# if __GLIBC_PREREQ(2, 15)
|
||||||
|
# error this test should work now
|
||||||
|
# endif
|
||||||
|
#endif
|
@ -14,6 +14,8 @@
|
|||||||
* limitations under the License.
|
* limitations under the License.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
#define _GNU_SOURCE 1
|
||||||
|
|
||||||
#include <string.h>
|
#include <string.h>
|
||||||
|
|
||||||
#include <errno.h>
|
#include <errno.h>
|
||||||
@ -72,28 +74,34 @@ TEST(string, strerror_concurrent) {
|
|||||||
#endif // __BIONIC__
|
#endif // __BIONIC__
|
||||||
}
|
}
|
||||||
|
|
||||||
TEST(string, strerror_r) {
|
TEST(string, gnu_strerror_r) {
|
||||||
#if defined(__BIONIC__) // glibc's strerror_r doesn't even have the same signature as the POSIX one.
|
|
||||||
char buf[256];
|
char buf[256];
|
||||||
|
|
||||||
|
// Note that glibc doesn't necessarily write into the buffer.
|
||||||
|
|
||||||
// Valid.
|
// Valid.
|
||||||
ASSERT_EQ(0, strerror_r(0, buf, sizeof(buf)));
|
ASSERT_STREQ("Success", strerror_r(0, buf, sizeof(buf)));
|
||||||
|
#if defined(__BIONIC__)
|
||||||
ASSERT_STREQ("Success", buf);
|
ASSERT_STREQ("Success", buf);
|
||||||
ASSERT_EQ(0, strerror_r(1, buf, sizeof(buf)));
|
#endif
|
||||||
|
ASSERT_STREQ("Operation not permitted", strerror_r(1, buf, sizeof(buf)));
|
||||||
|
#if defined(__BIONIC__)
|
||||||
ASSERT_STREQ("Operation not permitted", buf);
|
ASSERT_STREQ("Operation not permitted", buf);
|
||||||
|
#endif
|
||||||
|
|
||||||
// Invalid.
|
// Invalid.
|
||||||
ASSERT_EQ(0, strerror_r(-1, buf, sizeof(buf)));
|
ASSERT_STREQ("Unknown error -1", strerror_r(-1, buf, sizeof(buf)));
|
||||||
ASSERT_STREQ("Unknown error -1", buf);
|
ASSERT_STREQ("Unknown error -1", buf);
|
||||||
ASSERT_EQ(0, strerror_r(1234, buf, sizeof(buf)));
|
ASSERT_STREQ("Unknown error 1234", strerror_r(1234, buf, sizeof(buf)));
|
||||||
ASSERT_STREQ("Unknown error 1234", buf);
|
ASSERT_STREQ("Unknown error 1234", buf);
|
||||||
|
|
||||||
// Buffer too small.
|
// Buffer too small.
|
||||||
ASSERT_EQ(-1, strerror_r(0, buf, 2));
|
errno = 0;
|
||||||
ASSERT_EQ(ERANGE, errno);
|
memset(buf, 0, sizeof(buf));
|
||||||
#else // __BIONIC__
|
ASSERT_EQ(buf, strerror_r(4567, buf, 2));
|
||||||
GTEST_LOG_(INFO) << "This test does nothing.\n";
|
ASSERT_STREQ("U", buf);
|
||||||
#endif // __BIONIC__
|
// The GNU strerror_r doesn't set errno (the POSIX one sets it to ERANGE).
|
||||||
|
ASSERT_EQ(0, errno);
|
||||||
}
|
}
|
||||||
|
|
||||||
TEST(string, strsignal) {
|
TEST(string, strsignal) {
|
||||||
|
Loading…
x
Reference in New Issue
Block a user