Bug 1962204 - Prepare the crash helper sockets before sending them to the Android service process r=afranchuk,geckoview-reviewers,ohall
The additional native methods are contained by the libcrashhelper shared library instead of within libxul because they're needed before we load libxul. Differential Revision: https://phabricator.services.mozilla.com/D246521
This commit is contained in:
@@ -6,6 +6,7 @@ package org.mozilla.gecko.crashhelper;
|
|||||||
|
|
||||||
import android.app.Service;
|
import android.app.Service;
|
||||||
import android.content.ComponentName;
|
import android.content.ComponentName;
|
||||||
|
import android.content.Context;
|
||||||
import android.content.Intent;
|
import android.content.Intent;
|
||||||
import android.content.ServiceConnection;
|
import android.content.ServiceConnection;
|
||||||
import android.os.Binder;
|
import android.os.Binder;
|
||||||
@@ -129,7 +130,13 @@ public final class CrashHelper extends Service {
|
|||||||
throws IOException {
|
throws IOException {
|
||||||
mBreakpadClient = ParcelFileDescriptor.dup(breakpadClientFd);
|
mBreakpadClient = ParcelFileDescriptor.dup(breakpadClientFd);
|
||||||
mBreakpadServer = ParcelFileDescriptor.dup(breakpadServerFd);
|
mBreakpadServer = ParcelFileDescriptor.dup(breakpadServerFd);
|
||||||
|
if (!CrashHelper.set_breakpad_opts(mBreakpadServer.getFd())) {
|
||||||
|
throw new IOException("Could not set the proper options on the Breakpad socket");
|
||||||
|
}
|
||||||
mListener = ParcelFileDescriptor.dup(listenerFd);
|
mListener = ParcelFileDescriptor.dup(listenerFd);
|
||||||
|
if (!CrashHelper.bind_and_listen(mListener.getFd())) {
|
||||||
|
throw new IOException("Could not listen on incoming connections");
|
||||||
|
}
|
||||||
mClient = ParcelFileDescriptor.dup(clientFd);
|
mClient = ParcelFileDescriptor.dup(clientFd);
|
||||||
mServer = ParcelFileDescriptor.dup(serverFd);
|
mServer = ParcelFileDescriptor.dup(serverFd);
|
||||||
}
|
}
|
||||||
@@ -141,13 +148,12 @@ public final class CrashHelper extends Service {
|
|||||||
// google_breakpad::CrashGenerationServer::CreateReportChannel(), so we can
|
// google_breakpad::CrashGenerationServer::CreateReportChannel(), so we can
|
||||||
// use them Breakpad's crash generation server & clients. The rest are
|
// use them Breakpad's crash generation server & clients. The rest are
|
||||||
// specific to the crash helper process.
|
// specific to the crash helper process.
|
||||||
public static Pipes createCrashHelperPipes() {
|
public static Pipes createCrashHelperPipes(final Context context) {
|
||||||
// Ideally we'd set all the required options for the server socket, by using
|
// We can't set the required socket options for the Breakpad server socket
|
||||||
// Os.setsockoptInt() to enable the SO_PASSCRED socket option and then a
|
// or our own listener from here, so we delegate those parts to native
|
||||||
// couple of calls to Os.fcntlInt() to read the socket file flags and then
|
// functions in crashhelper_android.cpp.
|
||||||
// set it in non-blocking mode by setting O_NONBLOCK. Unfortunately both
|
GeckoLoader.doLoadLibrary(context, "crashhelper");
|
||||||
// calls require Android API version more recent than we support, so this
|
|
||||||
// job is delegated to crashhelper_android.cpp after receiving the socket.
|
|
||||||
try {
|
try {
|
||||||
final FileDescriptor breakpad_client_fd = new FileDescriptor();
|
final FileDescriptor breakpad_client_fd = new FileDescriptor();
|
||||||
final FileDescriptor breakpad_server_fd = new FileDescriptor();
|
final FileDescriptor breakpad_server_fd = new FileDescriptor();
|
||||||
@@ -188,4 +194,8 @@ public final class CrashHelper extends Service {
|
|||||||
|
|
||||||
protected static native void crash_generator(
|
protected static native void crash_generator(
|
||||||
int clientPid, int breakpadFd, String minidumpPath, int listenFd, int serverFd);
|
int clientPid, int breakpadFd, String minidumpPath, int listenFd, int serverFd);
|
||||||
|
|
||||||
|
protected static native boolean set_breakpad_opts(int breakpadFd);
|
||||||
|
|
||||||
|
protected static native boolean bind_and_listen(int listenFd);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -399,14 +399,14 @@ public final class GeckoRuntime implements Parcelable {
|
|||||||
}
|
}
|
||||||
|
|
||||||
private int[] startCrashHelper() {
|
private int[] startCrashHelper() {
|
||||||
final CrashHelper.Pipes pipes = CrashHelper.createCrashHelperPipes();
|
final Context context = GeckoAppShell.getApplicationContext();
|
||||||
|
final CrashHelper.Pipes pipes = CrashHelper.createCrashHelperPipes(context);
|
||||||
|
|
||||||
if (pipes == null) {
|
if (pipes == null) {
|
||||||
Log.e(LOGTAG, "Could not create the crash reporter IPC pipes");
|
Log.e(LOGTAG, "Could not create the crash reporter IPC pipes");
|
||||||
return new int[] {-1, -1};
|
return new int[] {-1, -1};
|
||||||
}
|
}
|
||||||
|
|
||||||
final Context context = GeckoAppShell.getApplicationContext();
|
|
||||||
try {
|
try {
|
||||||
@SuppressWarnings("unchecked")
|
@SuppressWarnings("unchecked")
|
||||||
final Class<? extends Service> cls =
|
final Class<? extends Service> cls =
|
||||||
|
|||||||
@@ -7,6 +7,7 @@
|
|||||||
#include <android/log.h>
|
#include <android/log.h>
|
||||||
#include <fcntl.h>
|
#include <fcntl.h>
|
||||||
#include <sys/socket.h>
|
#include <sys/socket.h>
|
||||||
|
#include <sys/un.h>
|
||||||
|
|
||||||
// For DirectAuxvDumpInfo
|
// For DirectAuxvDumpInfo
|
||||||
#include "mozilla/toolkit/crashreporter/rust_minidump_writer_linux_ffi_generated.h"
|
#include "mozilla/toolkit/crashreporter/rust_minidump_writer_linux_ffi_generated.h"
|
||||||
@@ -14,21 +15,54 @@
|
|||||||
|
|
||||||
#define CRASH_HELPER_LOGTAG "GeckoCrashHelper"
|
#define CRASH_HELPER_LOGTAG "GeckoCrashHelper"
|
||||||
|
|
||||||
|
extern "C" JNIEXPORT jboolean JNICALL
|
||||||
|
Java_org_mozilla_gecko_crashhelper_CrashHelper_set_1breakpad_1opts(
|
||||||
|
JNIEnv* jenv, jclass, jint breakpad_fd) {
|
||||||
|
// Enable passing credentials on the Breakpad server socket. We'd love to do
|
||||||
|
// it inside CrashHelper.java but the Java methods require an Android API
|
||||||
|
// version that's too recent for us.
|
||||||
|
const int val = 1;
|
||||||
|
int res = setsockopt(breakpad_fd, SOL_SOCKET, SO_PASSCRED, &val, sizeof(val));
|
||||||
|
if (res < 0) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
extern "C" JNIEXPORT jboolean JNICALL
|
||||||
|
Java_org_mozilla_gecko_crashhelper_CrashHelper_bind_1and_1listen(
|
||||||
|
JNIEnv* jenv, jclass, jint listen_fd) {
|
||||||
|
struct sockaddr_un addr = {
|
||||||
|
.sun_family = AF_UNIX,
|
||||||
|
.sun_path = {},
|
||||||
|
};
|
||||||
|
|
||||||
|
// The address' path deliberately starts with a null byte to inform the
|
||||||
|
// kernel that this is an abstract address and not an actual file path.
|
||||||
|
snprintf(addr.sun_path + 1, sizeof(addr.sun_path) - 2,
|
||||||
|
"gecko-crash-helper-pipe.%d", getpid());
|
||||||
|
|
||||||
|
int res = bind(listen_fd, (const struct sockaddr*)&addr, sizeof(addr));
|
||||||
|
if (res < 0) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
res = listen(listen_fd, 1);
|
||||||
|
if (res < 0) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
extern "C" JNIEXPORT void JNICALL
|
extern "C" JNIEXPORT void JNICALL
|
||||||
Java_org_mozilla_gecko_crashhelper_CrashHelper_crash_1generator(
|
Java_org_mozilla_gecko_crashhelper_CrashHelper_crash_1generator(
|
||||||
JNIEnv* jenv, jclass, jint client_pid, jint breakpad_fd,
|
JNIEnv* jenv, jclass, jint client_pid, jint breakpad_fd,
|
||||||
jstring minidump_path, jint listen_fd, jint server_fd) {
|
jstring minidump_path, jint listen_fd, jint server_fd) {
|
||||||
// Enable passing credentials on the server socket and set it in non-blocking
|
// The breakpad server socket needs to be put in non-blocking mode, we do it
|
||||||
// mode. We'd love to do it inside CrashHelper.java but the Java methods
|
// here as the Rust code that picks it up won't touch it anymore and just
|
||||||
// require an Android API version that's too recent for us.
|
// pass it along to Breakpad.
|
||||||
const int val = 1;
|
|
||||||
int res = setsockopt(breakpad_fd, SOL_SOCKET, SO_PASSCRED, &val, sizeof(val));
|
|
||||||
if (res < 0) {
|
|
||||||
__android_log_print(ANDROID_LOG_FATAL, CRASH_HELPER_LOGTAG,
|
|
||||||
"Unable to set the Breakpad pipe socket options");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
int flags = fcntl(breakpad_fd, F_GETFL);
|
int flags = fcntl(breakpad_fd, F_GETFL);
|
||||||
if (flags == -1) {
|
if (flags == -1) {
|
||||||
__android_log_print(ANDROID_LOG_FATAL, CRASH_HELPER_LOGTAG,
|
__android_log_print(ANDROID_LOG_FATAL, CRASH_HELPER_LOGTAG,
|
||||||
@@ -36,7 +70,7 @@ Java_org_mozilla_gecko_crashhelper_CrashHelper_crash_1generator(
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
res = fcntl(breakpad_fd, F_SETFL, flags | O_NONBLOCK);
|
int res = fcntl(breakpad_fd, F_SETFL, flags | O_NONBLOCK);
|
||||||
if (res == -1) {
|
if (res == -1) {
|
||||||
__android_log_print(ANDROID_LOG_FATAL, CRASH_HELPER_LOGTAG,
|
__android_log_print(ANDROID_LOG_FATAL, CRASH_HELPER_LOGTAG,
|
||||||
"Unable to set the Breakpad pipe in non-blocking mode");
|
"Unable to set the Breakpad pipe in non-blocking mode");
|
||||||
|
|||||||
@@ -2,7 +2,10 @@
|
|||||||
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
||||||
* file, You can obtain one at https://mozilla.org/MPL/2.0/. */
|
* file, You can obtain one at https://mozilla.org/MPL/2.0/. */
|
||||||
|
|
||||||
use nix::poll::{poll, PollFd, PollFlags, PollTimeout};
|
use nix::{
|
||||||
|
errno::Errno,
|
||||||
|
poll::{poll, PollFd, PollFlags, PollTimeout},
|
||||||
|
};
|
||||||
|
|
||||||
use crate::{errors::IPCError, ignore_eintr, IPCConnector, IPCEvent, IPCListener};
|
use crate::{errors::IPCError, ignore_eintr, IPCConnector, IPCEvent, IPCListener};
|
||||||
|
|
||||||
@@ -48,8 +51,14 @@ pub fn wait_for_events(
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if revents.contains(PollFlags::POLLHUP) && (index > 0) {
|
if revents.contains(PollFlags::POLLHUP) {
|
||||||
events.push(IPCEvent::Disconnect(index - 1));
|
if index > 0 {
|
||||||
|
events.push(IPCEvent::Disconnect(index - 1));
|
||||||
|
} else {
|
||||||
|
// This should never happen, unless the listener socket was not
|
||||||
|
// set up properly or a failure happened during setup.
|
||||||
|
return Err(IPCError::System(Errno::EFAULT));
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if !revents.is_empty() {
|
if !revents.is_empty() {
|
||||||
|
|||||||
Reference in New Issue
Block a user