am 29f06943: Merge "Add GNU-compatible strerror_r."
				
					
				
			* commit '29f06943a19ef2271a08f4223f597a3383f0d9c0': Add GNU-compatible strerror_r.
This commit is contained in:
		@@ -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) {
 | 
				
			||||||
 
 | 
				
			|||||||
		Reference in New Issue
	
	Block a user