Use strerror_r for JNI error messages. (#10463) (#10465)

Motivation:
We previously relied on `strerror`, but this function is unfortunately not thread-safe.

Modification:
The use of `strerror` has been changed to `strerror_r`, which is thread-safe.
This function has a more complicated API, and has portability concerns that needs to be handled.
This accounts for the relatively large increase in lines of code.

Result:
Error messages from JNI are now always generated in a thread-safe way.
This commit is contained in:
Chris Vest 2020-08-12 07:11:20 +02:00 committed by GitHub
parent 58654fff7d
commit 160e7f83d8
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23

View File

@ -29,12 +29,59 @@ static jclass portUnreachableExceptionClass = NULL;
static jclass closedChannelExceptionClass = NULL; static jclass closedChannelExceptionClass = NULL;
static jmethodID closedChannelExceptionMethodId = NULL; static jmethodID closedChannelExceptionMethodId = NULL;
/**
Our `strerror_r` wrapper makes sure that the function is XSI compliant,
even on platforms where the GNU variant is exposed.
Note: `strerrbuf` must be initialized to all zeros prior to calling this function.
XSI or GNU functions do not have such a requirement, but our wrappers do.
*/
#if (_POSIX_C_SOURCE >= 200112L || _XOPEN_SOURCE >= 600 || __APPLE__) && ! _GNU_SOURCE
static inline int strerror_r_xsi(int errnum, char *strerrbuf, size_t buflen) {
return strerror_r(errnum, strerrbuf, buflen);
}
#else
static inline int strerror_r_xsi(int errnum, char *strerrbuf, size_t buflen) {
char* tmp = strerror_r(errnum, strerrbuf, buflen);
if (strerrbuf[0] == '\0') {
// Our output buffer was not used. Copy from tmp.
strncpy(strerrbuf, tmp, buflen - 1); // Use (buflen - 1) to avoid overwriting terminating \0.
}
if (errno != 0) {
return -1;
}
return 0;
}
#endif
/** Notice: every usage of exceptionMessage needs to release the allocated memory for the sequence of char */ /** Notice: every usage of exceptionMessage needs to release the allocated memory for the sequence of char */
static char* exceptionMessage(char* msg, int error) { static char* exceptionMessage(char* msg, int error) {
// strerror is returning a constant, so no need to free anything coming from strerror if (error < 0) {
// error may be negative because some functions return negative values. we should make sure it is always // Error may be negative because some functions return negative values. We should make sure it is always
// positive when passing to standard library functions. // positive when passing to standard library functions.
return netty_unix_util_prepend(msg, strerror(error < 0 ? -error : error)); error = -error;
}
int buflen = 32;
char* strerrbuf = NULL;
int result = 0;
do {
buflen = buflen * 2;
if (buflen >= 2048) {
break; // Limit buffer growth.
}
if (strerrbuf != NULL) {
free(strerrbuf);
}
strerrbuf = calloc(buflen, sizeof(char));
result = strerror_r_xsi(error, strerrbuf, buflen);
if (result == -1) {
result = errno;
}
} while (result == ERANGE);
char* combined = netty_unix_util_prepend(msg, strerrbuf);
free(strerrbuf);
return combined;
} }
// Exported C methods // Exported C methods