blob: a98b38405bcc87bcaa8a01a0976824a50d8936dc [file] [log] [blame]
// Copyright 2025 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "chrome/browser/tab/tab_state_storage_service_android.h"
#include <memory>
#include "base/android/callback_android.h"
#include "base/android/jni_android.h"
#include "base/android/jni_array.h"
#include "base/android/jni_bytebuffer.h"
#include "base/android/jni_string.h"
#include "base/android/token_android.h"
#include "base/functional/callback_forward.h"
#include "base/logging.h"
#include "base/memory/ptr_util.h"
#include "base/memory/raw_ptr.h"
#include "base/token.h"
#include "chrome/browser/tab/tab_state_storage_backend.h"
#include "chrome/browser/tab/tab_state_storage_service.h"
// Must come after all headers that specialize FromJniType() / ToJniType().
#include "chrome/browser/tab/jni_headers/TabStateStorageService_jni.h"
namespace tabs {
namespace {
void RunJavaCallbackLoadAllTabs(
JNIEnv* env,
const base::android::JavaRef<jobject>& j_callback,
std::vector<tabs_pb::TabState> tab_states) {
std::vector<base::android::ScopedJavaLocalRef<jobject>> j_tab_state_vector;
for (tabs_pb::TabState& tab_state : tab_states) {
base::android::ScopedJavaLocalRef<jobject> j_web_contents_state_buffer;
if (tab_state.has_web_contents_state_bytes()) {
// TODO(https://crbug.com/427255040): This is probably leaking memory and
// should be fixed. No path back from Java when the owning object is
// destroyed/cleaned/gc'd, and Java currently has no way to tell the
// backing implementation of the owning object.
raw_ptr<std::string> web_contents_state_bytes_ptr =
tab_state.release_web_contents_state_bytes();
j_web_contents_state_buffer =
base::android::ScopedJavaLocalRef<jobject>::Adopt(
env, env->NewDirectByteBuffer(
static_cast<void*>(web_contents_state_bytes_ptr->data()),
web_contents_state_bytes_ptr->size()));
}
base::Token tab_group_token(tab_state.tab_group_id_high(),
tab_state.tab_group_id_low());
base::android::ScopedJavaLocalRef<jobject> j_tab_group_id =
base::android::TokenAndroid::Create(env, tab_group_token);
base::android::ScopedJavaLocalRef<jobject> j_tab_state =
Java_TabStateStorageService_createTabState(
env, tab_state.parent_id(), tab_state.root_id(),
tab_state.timestamp_millis(), j_web_contents_state_buffer,
tab_state.opener_app_id(), tab_state.theme_color(),
tab_state.launch_type_at_creation(), tab_state.user_agent(),
tab_state.last_navigation_committed_timestamp_millis(),
j_tab_group_id, tab_state.tab_has_sensitive_content(),
tab_state.is_pinned());
j_tab_state_vector.push_back(j_tab_state);
}
base::android::ScopedJavaLocalRef<jclass> type =
base::android::GetClass(env, "org/chromium/chrome/browser/tab/TabState");
base::android::ScopedJavaLocalRef<jobjectArray> j_tab_state_array =
base::android::ToTypedJavaArrayOfObjects(env, j_tab_state_vector,
type.obj());
base::android::RunObjectCallbackAndroid(j_callback, j_tab_state_array);
}
} // namespace
TabStateStorageServiceAndroid::TabStateStorageServiceAndroid(
TabStateStorageService* tab_state_storage_service)
: tab_state_storage_service_(tab_state_storage_service) {
JNIEnv* env = base::android::AttachCurrentThread();
java_obj_.Reset(Java_TabStateStorageService_create(
env, reinterpret_cast<intptr_t>(this)));
}
TabStateStorageServiceAndroid::~TabStateStorageServiceAndroid() = default;
void TabStateStorageServiceAndroid::SaveTab(
JNIEnv* env,
int id,
int parent_tab_id,
int root_id,
long timestamp_millis,
const jni_zero::JavaParamRef<jobject>& web_contents_state_buffer,
std::string opener_app_id,
int theme_color,
int launch_type_at_creation,
int user_agent,
long last_navigation_committed_timestamp_millis,
const jni_zero::JavaParamRef<jobject>& j_tab_group_id,
bool tab_has_sensitive_content,
bool is_pinned) {
std::unique_ptr<std::string> web_content_state_string;
if (web_contents_state_buffer) {
base::span<const uint8_t> span =
base::android::JavaByteBufferToSpan(env, web_contents_state_buffer);
web_content_state_string =
std::make_unique<std::string>(span.begin(), span.end());
}
std::unique_ptr<base::Token> tab_group_id;
if (j_tab_group_id) {
tab_group_id = std::make_unique<base::Token>(
base::android::TokenAndroid::FromJavaToken(env, j_tab_group_id));
}
tab_state_storage_service_->SaveTab(
id, parent_tab_id, root_id, timestamp_millis,
web_content_state_string.get(), opener_app_id, theme_color,
launch_type_at_creation, user_agent,
last_navigation_committed_timestamp_millis, tab_group_id.get(),
tab_has_sensitive_content, is_pinned);
}
void TabStateStorageServiceAndroid::LoadAllTabs(
JNIEnv* env,
const jni_zero::JavaParamRef<jobject>& j_callback) {
auto load_all_tabs_callback =
base::BindOnce(&RunJavaCallbackLoadAllTabs, env,
jni_zero::ScopedJavaGlobalRef<jobject>(j_callback));
tab_state_storage_service_->LoadAllTabs(std::move(load_all_tabs_callback));
}
base::android::ScopedJavaLocalRef<jobject>
TabStateStorageServiceAndroid::GetJavaObject() {
return base::android::ScopedJavaLocalRef<jobject>(java_obj_);
}
// This function is declared in tab_state_storage_service.h and
// should be linked in to any binary using
// TabStateStorageService::GetJavaObject.
// static
base::android::ScopedJavaLocalRef<jobject>
TabStateStorageService::GetJavaObject(
TabStateStorageService* tab_state_storage_service) {
TabStateStorageServiceAndroid* service_android =
static_cast<TabStateStorageServiceAndroid*>(
tab_state_storage_service->GetUserData(
kTabStateStorageServiceAndroidKey));
if (!service_android) {
service_android =
new TabStateStorageServiceAndroid(tab_state_storage_service);
tab_state_storage_service->SetUserData(kTabStateStorageServiceAndroidKey,
base::WrapUnique(service_android));
}
return service_android->GetJavaObject();
}
} // namespace tabs