| // Copyright 2019 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/thumbnail/generator/android/thumbnail_generator.h" | 
 |  | 
 | #include <memory> | 
 |  | 
 | #include "base/android/jni_string.h" | 
 | #include "base/functional/bind.h" | 
 | #include "base/strings/string_util.h" | 
 | #include "base/task/sequenced_task_runner.h" | 
 | #include "base/threading/thread_restrictions.h" | 
 | #include "chrome/browser/thumbnail/generator/android/thumbnail_media_parser.h" | 
 | #include "chrome/browser/thumbnail/generator/thumbnail_util.h" | 
 | #include "content/public/browser/browser_thread.h" | 
 | #include "ui/gfx/android/java_bitmap.h" | 
 |  | 
 | // Must come after all headers that specialize FromJniType() / ToJniType(). | 
 | #include "chrome/browser/thumbnail/generator/jni_headers/ThumbnailGenerator_jni.h" | 
 |  | 
 | class SkBitmap; | 
 |  | 
 | using base::android::JavaParamRef; | 
 | using base::android::ScopedJavaGlobalRef; | 
 |  | 
 | namespace { | 
 |  | 
 | void ForwardJavaCallback(const ScopedJavaGlobalRef<jobject>& java_delegate, | 
 |                          const ScopedJavaGlobalRef<jstring>& content_id, | 
 |                          int icon_size, | 
 |                          const ScopedJavaGlobalRef<jobject>& callback, | 
 |                          SkBitmap thumbnail) { | 
 |   DCHECK_CURRENTLY_ON(content::BrowserThread::UI); | 
 |   JNIEnv* env = base::android::AttachCurrentThread(); | 
 |   Java_ThumbnailGenerator_onThumbnailRetrieved( | 
 |       env, java_delegate, content_id, icon_size, | 
 |       thumbnail.drawsNothing() ? nullptr : gfx::ConvertToJavaBitmap(thumbnail), | 
 |       callback); | 
 | } | 
 |  | 
 | void OnThumbnailScaled(base::OnceCallback<void(SkBitmap)> java_callback, | 
 |                        SkBitmap scaled_thumbnail) { | 
 |   DCHECK_CURRENTLY_ON(content::BrowserThread::UI); | 
 |   std::move(java_callback).Run(std::move(scaled_thumbnail)); | 
 | } | 
 |  | 
 | }  // namespace | 
 |  | 
 | ThumbnailGenerator::ThumbnailGenerator(const JavaParamRef<jobject>& jobj) | 
 |     : java_delegate_(jobj) { | 
 |   DCHECK(!jobj.is_null()); | 
 | } | 
 |  | 
 | ThumbnailGenerator::~ThumbnailGenerator() = default; | 
 |  | 
 | void ThumbnailGenerator::Destroy(JNIEnv* env) { | 
 |   DCHECK_CURRENTLY_ON(content::BrowserThread::UI); | 
 |   delete this; | 
 | } | 
 |  | 
 | void ThumbnailGenerator::OnImageThumbnailRetrieved( | 
 |     base::OnceCallback<void(SkBitmap)> java_callback, | 
 |     const SkBitmap& thumbnail) { | 
 |   DCHECK_CURRENTLY_ON(content::BrowserThread::UI); | 
 |  | 
 |   // Send the bitmap back to Java-land. | 
 |   std::move(java_callback).Run(std::move(thumbnail)); | 
 | } | 
 |  | 
 | void ThumbnailGenerator::OnVideoThumbnailRetrieved( | 
 |     base::OnceCallback<void(SkBitmap)> java_callback, | 
 |     int icon_size, | 
 |     std::unique_ptr<ThumbnailMediaParser> parser, | 
 |     bool success, | 
 |     chrome::mojom::MediaMetadataPtr media_metadata, | 
 |     SkBitmap thumbnail) { | 
 |   DCHECK_CURRENTLY_ON(content::BrowserThread::UI); | 
 |  | 
 |   // Scale the bitmap before sending back to Java. | 
 |   ScaleDownBitmap(icon_size, std::move(thumbnail), | 
 |                   base::BindOnce(&OnThumbnailScaled, std::move(java_callback))); | 
 |  | 
 |   // We want to delete |parser| but can't do it immediately because current | 
 |   // stack contains functions that belong to ThumbnailMediaParser and the | 
 |   // VideoDecoder that the parser owens. This would cause use-after-free. | 
 |   // That's why |parser|'s destruction is postponed till the current task | 
 |   // is completed and the call stack doesn't have frames referencing memory | 
 |   // owned by |parser| | 
 |   base::SequencedTaskRunner::GetCurrentDefault()->DeleteSoon(FROM_HERE, | 
 |                                                              std::move(parser)); | 
 | } | 
 |  | 
 | void ThumbnailGenerator::RetrieveThumbnail( | 
 |     JNIEnv* env, | 
 |     const JavaParamRef<jstring>& jcontent_id, | 
 |     const JavaParamRef<jstring>& jfile_path, | 
 |     const JavaParamRef<jstring>& jmime_type, | 
 |     jint icon_size, | 
 |     const JavaParamRef<jobject>& callback) { | 
 |   DCHECK_CURRENTLY_ON(content::BrowserThread::UI); | 
 |  | 
 |   base::FilePath file_path = base::FilePath::FromUTF8Unsafe( | 
 |       base::android::ConvertJavaStringToUTF8(env, jfile_path)); | 
 |  | 
 |   std::string mime_type = | 
 |       jmime_type.is_null() | 
 |           ? "" | 
 |           : base::android::ConvertJavaStringToUTF8(env, jmime_type); | 
 |  | 
 |   // Bind everything passed back to Java. | 
 |   auto java_callback = | 
 |       base::BindOnce(&ForwardJavaCallback, java_delegate_, | 
 |                      ScopedJavaGlobalRef<jstring>(jcontent_id), icon_size, | 
 |                      ScopedJavaGlobalRef<jobject>(callback)); | 
 |  | 
 |   // Retrieve video thumbnail. | 
 |   if (base::StartsWith(mime_type, "video/", | 
 |                        base::CompareCase::INSENSITIVE_ASCII)) { | 
 |     auto parser = ThumbnailMediaParser::Create(mime_type, file_path); | 
 |     auto* const parser_ptr = parser.get(); | 
 |     parser_ptr->Start( | 
 |         base::BindOnce(&ThumbnailGenerator::OnVideoThumbnailRetrieved, | 
 |                        weak_factory_.GetWeakPtr(), std::move(java_callback), | 
 |                        icon_size, std::move(parser))); | 
 |     return; | 
 |   } | 
 |  | 
 |   // Retrieve image thumbnail. | 
 |   auto request = std::make_unique<ImageThumbnailRequest>( | 
 |       icon_size, | 
 |       base::BindOnce(&ThumbnailGenerator::OnImageThumbnailRetrieved, | 
 |                      weak_factory_.GetWeakPtr(), std::move(java_callback))); | 
 |   request->Start(file_path); | 
 |  | 
 |   // Dropping ownership of |request| here because it will clean itself up once | 
 |   // the started request finishes. | 
 |   request.release(); | 
 | } | 
 |  | 
 | // static | 
 | static jlong JNI_ThumbnailGenerator_Init(JNIEnv* env, | 
 |                                          const JavaParamRef<jobject>& jobj) { | 
 |   return reinterpret_cast<intptr_t>(new ThumbnailGenerator(jobj)); | 
 | } |