#include "components/cronet/android/cronet_library_loader.h"
#include <android/trace.h>
#include <jni.h>
#include <sys/system_properties.h>
#include <memory>
#include <string>
#include <utility>
#include "base/android/android_info.h"
#include "base/android/base_jni_init.h"
#include "base/android/jni_android.h"
#include "base/android/jni_array.h"
#include "base/android/jni_registrar.h"
#include "base/android/jni_string.h"
#include "base/android/jni_utils.h"
#include "base/android/library_loader/library_loader_hooks.h"
#include "base/check_op.h"
#include "base/feature_list.h"
#include "base/logging/logging_settings.h"
#include "base/message_loop/message_pump_type.h"
#include "base/metrics/field_trial_params.h"
#include "base/no_destructor.h"
#include "base/synchronization/waitable_event.h"
#include "base/task/current_thread.h"
#include "base/task/sequenced_task_runner.h"
#include "base/task/single_thread_task_executor.h"
#include "base/task/task_traits.h"
#include "base/task/thread_pool.h"
#include "base/task/thread_pool/thread_pool_instance.h"
#include "base/time/time_delta_from_string.h"
#include "base/trace_event/trace_event.h"
#include "base/tracing/perfetto_platform.h"
#include "build/build_config.h"
#include "components/cronet/android/cronet_base_feature.h"
#include "components/cronet/android/cronet_jni_registration_generated.h"
#include "components/cronet/android/proto/base_feature_overrides.pb.h"
#include "components/cronet/cronet_global_state.h"
#include "components/cronet/version.h"
#include "net/android/network_change_notifier_delegate_android.h"
#include "net/android/network_change_notifier_factory_android.h"
#include "net/base/network_change_notifier.h"
#include "net/log/net_log.h"
#include "net/log/net_log_capture_mode.h"
#include "net/log/trace_net_log_observer.h"
#include "net/proxy_resolution/configured_proxy_resolution_service.h"
#include "net/proxy_resolution/proxy_config_service_android.h"
#include "third_party/perfetto/include/perfetto/tracing/tracing.h"
#include "third_party/zlib/zlib.h"
#include "components/cronet/android/cronet_jni_headers/CronetLibraryLoader_jni.h"
using base::android::JavaParamRef;
using base::android::ScopedJavaLocalRef;
namespace cronet {
namespace {
BASE_FEATURE(kLogMe, "CronetLogMe", base::FEATURE_DISABLED_BY_DEFAULT);
constexpr base::FeatureParam<std::string> kLogMeMessage{&kLogMe, "message", ""};
base::SingleThreadTaskExecutor* g_init_task_executor = nullptr;
std::unique_ptr<net::NetworkChangeNotifier> g_network_change_notifier;
base::WaitableEvent g_init_thread_init_done(
base::WaitableEvent::ResetPolicy::MANUAL,
base::WaitableEvent::InitialState::NOT_SIGNALED);
std::optional<net::NetLogCaptureMode> g_trace_net_log_capture_mode;
::org::chromium::net::httpflags::BaseFeatureOverrides GetBaseFeatureOverrides(
JNIEnv* env) {
const auto serializedProto =
cronet::Java_CronetLibraryLoader_getBaseFeatureOverrides(env);
CHECK(serializedProto);
const auto serializedProtoSize =
base::android::SafeGetArrayLength(env, serializedProto);
::org::chromium::net::httpflags::BaseFeatureOverrides overrides;
void* const serializedProtoArray =
env->GetPrimitiveArrayCritical(serializedProto.obj(), nullptr);
CHECK(serializedProtoArray != nullptr);
CHECK(overrides.ParseFromArray(serializedProtoArray, serializedProtoSize));
env->ReleasePrimitiveArrayCritical(serializedProto.obj(),
serializedProtoArray, JNI_ABORT);
return overrides;
}
void InitializePerfetto() {
static base::NoDestructor<base::tracing::PerfettoPlatform> perfetto_platform(
base::ThreadPool::CreateSequencedTaskRunner(
{base::MayBlock(), base::TaskPriority::USER_BLOCKING}),
base::tracing::PerfettoPlatform::Options{
.process_name_prefix = "cronet-"});
perfetto::TracingInitArgs tracing_init_args;
tracing_init_args.backends = perfetto::kSystemBackend;
tracing_init_args.platform = perfetto_platform.get();
tracing_init_args.enable_system_consumer = false;
perfetto::Tracing::Initialize(tracing_init_args);
base::TrackEvent::Register();
constexpr char trace_delay_system_property_name[] =
"debug.cronet.init_trace_sleep";
if (const prop_info* const trace_delay_prop_info =
__system_property_find(trace_delay_system_property_name);
trace_delay_prop_info != nullptr) {
std::array<char, PROP_VALUE_MAX> trace_delay_buffer;
const std::string_view trace_delay_str(
trace_delay_buffer.data(),
__system_property_read(trace_delay_prop_info, nullptr,
trace_delay_buffer.data()));
const auto trace_delay = base::TimeDeltaFromString(trace_delay_str);
if (!trace_delay.has_value()) {
LOG(WARNING) << "Invalid value for system property "
<< trace_delay_system_property_name;
} else {
ATrace_beginSection(absl::StrFormat("Sleeping %s for Perfetto",
absl::FormatStreamed(*trace_delay))
.c_str());
base::PlatformThread::Sleep(*trace_delay);
ATrace_endSection();
}
}
}
}
static void JNI_CronetLibraryLoader_NativeInit(JNIEnv* env,
jboolean initializePerfetto) {
logging::InitLogging(logging::LoggingSettings());
if (!base::ThreadPoolInstance::Get()) {
base::ThreadPoolInstance::CreateAndStartWithDefaultParams("Cronet");
}
if (initializePerfetto) {
ATrace_beginSection("CronetLibraryLoader_NativeInit initializing Perfetto");
InitializePerfetto();
ATrace_endSection();
}
ApplyBaseFeatureOverrides(GetBaseFeatureOverrides(env));
if (base::FeatureList::IsEnabled(kLogMe)) {
LOG( INFO)
<< "CronetLogMe feature flag set, logging as instructed. Message: "
<< kLogMeMessage.Get();
}
}
bool OnInitThread() {
DCHECK(g_init_task_executor);
return g_init_task_executor->task_runner()->RunsTasksInCurrentSequence();
}
jint CronetOnLoad(JavaVM* vm, void* reserved) {
base::android::InitVM(vm);
JNIEnv* env = base::android::AttachCurrentThread();
if (!RegisterNatives(env)) {
return -1;
}
if (!base::android::OnJNIOnLoadInit())
return -1;
return JNI_VERSION_1_6;
}
void CronetOnUnLoad(JavaVM* jvm, void* reserved) {
if (base::ThreadPoolInstance::Get())
base::ThreadPoolInstance::Get()->Shutdown();
base::android::LibraryLoaderExitHook();
}
static void JNI_CronetLibraryLoader_CronetInitOnInitThread(
JNIEnv* env,
net::NetLogCaptureMode trace_net_log_capture_mode) {
DCHECK(!base::CurrentThread::IsSet());
DCHECK(!g_init_task_executor);
g_init_task_executor =
new base::SingleThreadTaskExecutor(base::MessagePumpType::JAVA);
static base::NoDestructor<net::TraceNetLogObserver> trace_net_log_observer(
net::TraceNetLogObserver::Options{
.capture_mode = trace_net_log_capture_mode,
.use_sensitive_category = trace_net_log_capture_mode !=
net::NetLogCaptureMode::kHeavilyRedacted,
.root_track_name = "Cronet NetLog",
.verbose = true,
});
CHECK(!g_trace_net_log_capture_mode.has_value());
g_trace_net_log_capture_mode = trace_net_log_capture_mode;
trace_net_log_observer->WatchForTraceStart(net::NetLog::Get());
DCHECK(!g_network_change_notifier);
if (!net::NetworkChangeNotifier::GetFactory()) {
net::NetworkChangeNotifier::SetFactory(
new net::NetworkChangeNotifierFactoryAndroid(
net::NetworkChangeNotifierDelegateAndroid::ForceUpdateNetworkState::
kDisabled));
}
g_network_change_notifier = net::NetworkChangeNotifier::CreateIfNeeded();
DCHECK(g_network_change_notifier);
g_init_thread_init_done.Signal();
}
static net::NetLogCaptureMode
JNI_CronetLibraryLoader_GetTraceNetLogCaptureModeForTesting(JNIEnv* env) {
return g_trace_net_log_capture_mode.value();
}
static ScopedJavaLocalRef<jstring> JNI_CronetLibraryLoader_GetCronetVersion(
JNIEnv* env) {
#if defined(ARCH_CPU_ARM64)
if (base::android::android_info::sdk_int() ==
base::android::android_info::SDK_VERSION_MARSHMALLOW) {
crc32(0, Z_NULL, 0);
}
#endif
return base::android::ConvertUTF8ToJavaString(env, CRONET_VERSION);
}
static void JNI_CronetLibraryLoader_SetMinLogLevel(JNIEnv* env,
jint jlog_level) {
logging::SetMinLogLevel(jlog_level);
}
void PostTaskToInitThread(const base::Location& posted_from,
base::OnceClosure task) {
g_init_thread_init_done.Wait();
g_init_task_executor->task_runner()->PostTask(posted_from, std::move(task));
}
void EnsureInitialized() {
if (g_init_task_executor) {
g_init_thread_init_done.Wait();
return;
}
static class RunOnce {
public:
RunOnce() {
JNIEnv* env = base::android::AttachCurrentThread();
cronet::Java_CronetLibraryLoader_ensureInitializedFromNative(env);
}
} s_run_once;
}
std::unique_ptr<net::ProxyConfigService> CreateProxyConfigService(
const scoped_refptr<base::SequencedTaskRunner>& io_task_runner) {
std::unique_ptr<net::ProxyConfigService> service =
net::ProxyConfigService::CreateSystemProxyConfigService(io_task_runner);
net::ProxyConfigServiceAndroid* android_proxy_config_service =
static_cast<net::ProxyConfigServiceAndroid*>(service.get());
android_proxy_config_service->set_exclude_pac_url(true);
return service;
}
std::unique_ptr<net::ProxyResolutionService> CreateProxyResolutionService(
std::unique_ptr<net::ProxyConfigService> proxy_config_service,
net::NetLog* net_log) {
return net::ConfiguredProxyResolutionService::CreateWithoutProxyResolver(
std::move(proxy_config_service),
nullptr, net_log);
}
std::string CreateDefaultUserAgent(const std::string& partial_user_agent) {
cronet::EnsureInitialized();
JNIEnv* env = base::android::AttachCurrentThread();
std::string user_agent = base::android::ConvertJavaStringToUTF8(
cronet::Java_CronetLibraryLoader_getDefaultUserAgent(env));
if (!partial_user_agent.empty())
user_agent.insert(user_agent.size() - 1, "; " + partial_user_agent);
return user_agent;
}
void SetNetworkThreadPriorityOnNetworkThread(double priority) {
int priority_int = priority;
DCHECK_LE(priority_int, 19);
DCHECK_GE(priority_int, -20);
if (priority_int >= -20 && priority_int <= 19) {
JNIEnv* env = base::android::AttachCurrentThread();
cronet::Java_CronetLibraryLoader_setNetworkThreadPriorityOnNetworkThread(
env, priority_int);
}
}
}
DEFINE_JNI(CronetLibraryLoader)