Skip to content

Commit

Permalink
Add embedder texture initial code for impeller
Browse files Browse the repository at this point in the history
Add registerTextureWithType interface to proc table
  • Loading branch information
xiaowei-guan authored and JSUYA committed Jun 11, 2024
1 parent f967dfc commit 9ed9b65
Show file tree
Hide file tree
Showing 8 changed files with 314 additions and 27 deletions.
22 changes: 22 additions & 0 deletions shell/platform/embedder/embedder.cc
Original file line number Diff line number Diff line change
Expand Up @@ -2712,6 +2712,26 @@ FlutterEngineResult FlutterEngineRegisterExternalTexture(
return kSuccess;
}

FlutterEngineResult FlutterEngineRegisterExternalTextureWithType(
FLUTTER_API_SYMBOL(FlutterEngine) engine,
int64_t texture_identifier,
FlutterTextureType type) {
if (engine == nullptr) {
return LOG_EMBEDDER_ERROR(kInvalidArguments, "Engine handle was invalid.");
}

if (texture_identifier == 0) {
return LOG_EMBEDDER_ERROR(kInvalidArguments,
"Texture identifier was invalid.");
}
if (!reinterpret_cast<flutter::EmbedderEngine*>(engine)->RegisterTexture(
texture_identifier, type)) {
return LOG_EMBEDDER_ERROR(kInternalInconsistency,
"Could not register the specified texture.");
}
return kSuccess;
}

FlutterEngineResult FlutterEngineUnregisterExternalTexture(
FLUTTER_API_SYMBOL(FlutterEngine) engine,
int64_t texture_identifier) {
Expand Down Expand Up @@ -3287,6 +3307,8 @@ FlutterEngineResult FlutterEngineGetProcAddresses(
SET_PROC(SendPlatformMessageResponse,
FlutterEngineSendPlatformMessageResponse);
SET_PROC(RegisterExternalTexture, FlutterEngineRegisterExternalTexture);
SET_PROC(RegisterExternalTextureWithType,
FlutterEngineRegisterExternalTextureWithType);
SET_PROC(UnregisterExternalTexture, FlutterEngineUnregisterExternalTexture);
SET_PROC(MarkExternalTextureFrameAvailable,
FlutterEngineMarkExternalTextureFrameAvailable);
Expand Down
50 changes: 49 additions & 1 deletion shell/platform/embedder/embedder.h
Original file line number Diff line number Diff line change
Expand Up @@ -294,6 +294,7 @@ typedef struct {
} FlutterTransformation;

typedef void (*VoidCallback)(void* /* user data */);
typedef bool (*BoolCallback)(void* /* user data */);

typedef enum {
/// Specifies an OpenGL texture target type. Textures are specified using
Expand Down Expand Up @@ -371,6 +372,8 @@ typedef struct {
uint32_t format;
/// User data to be returned on the invocation of the destruction callback.
void* user_data;
/// Callback invoked that texture start binding.
BoolCallback bind_callback;
/// Callback invoked (on an engine managed thread) that asks the embedder to
/// collect the texture.
VoidCallback destruction_callback;
Expand All @@ -382,6 +385,10 @@ typedef struct {
size_t width;
/// Height of the texture.
size_t height;
/// The pixel data buffer.
const uint8_t* buffer;
/// The size of buffer.
size_t buffer_size;
} FlutterOpenGLTexture;

typedef struct {
Expand All @@ -401,7 +408,15 @@ typedef struct {
VoidCallback destruction_callback;
} FlutterOpenGLFramebuffer;

typedef bool (*BoolCallback)(void* /* user data */);
// Possible values for the type specified in FlutterDesktopTextureInfo.
// Additional types may be added in the future.
typedef enum {
// A Pixel buffer-based texture.
kFlutterPixelBufferTexture,
// A platform-specific GPU surface-backed texture.
kFlutterGpuSurfaceTexture
} FlutterTextureType;

typedef FlutterTransformation (*TransformationCallback)(void* /* user data */);
typedef uint32_t (*UIntCallback)(void* /* user data */);
typedef bool (*SoftwareSurfacePresentCallback)(void* /* user data */,
Expand Down Expand Up @@ -2679,6 +2694,32 @@ FlutterEngineResult FlutterEngineRegisterExternalTexture(
FLUTTER_API_SYMBOL(FlutterEngine) engine,
int64_t texture_identifier);

//------------------------------------------------------------------------------
/// @brief Register an external texture with a unique (per engine)
/// identifier. Only rendering backends that support external
/// textures accept external texture registrations. After the
/// external texture is registered, the application can mark that a
/// frame is available by calling
/// `FlutterEngineMarkExternalTextureFrameAvailable`.
///
/// @see FlutterEngineUnregisterExternalTexture()
/// @see FlutterEngineMarkExternalTextureFrameAvailable()
///
/// @param[in] engine A running engine instance.
/// @param[in] texture_identifier The identifier of the texture to register
/// with the engine. The embedder may supply new
/// frames to this texture using the same
/// identifier.
/// @param[in] type The type of the texture.
///
/// @return The result of the call.
///
FLUTTER_EXPORT
FlutterEngineResult FlutterEngineRegisterExternalTextureWithType(
FLUTTER_API_SYMBOL(FlutterEngine) engine,
int64_t texture_identifier,
FlutterTextureType type);

//------------------------------------------------------------------------------
/// @brief Unregister a previous texture registration.
///
Expand Down Expand Up @@ -3131,6 +3172,11 @@ typedef FlutterEngineResult (*FlutterEngineSendPlatformMessageResponseFnPtr)(
typedef FlutterEngineResult (*FlutterEngineRegisterExternalTextureFnPtr)(
FLUTTER_API_SYMBOL(FlutterEngine) engine,
int64_t texture_identifier);
typedef FlutterEngineResult (
*FlutterEngineRegisterExternalTextureWithTypeFnPtr)(
FLUTTER_API_SYMBOL(FlutterEngine) engine,
int64_t texture_identifier,
FlutterTextureType type);
typedef FlutterEngineResult (*FlutterEngineUnregisterExternalTextureFnPtr)(
FLUTTER_API_SYMBOL(FlutterEngine) engine,
int64_t texture_identifier);
Expand Down Expand Up @@ -3217,6 +3263,8 @@ typedef struct {
PlatformMessageReleaseResponseHandle;
FlutterEngineSendPlatformMessageResponseFnPtr SendPlatformMessageResponse;
FlutterEngineRegisterExternalTextureFnPtr RegisterExternalTexture;
FlutterEngineRegisterExternalTextureWithTypeFnPtr
RegisterExternalTextureWithType;
FlutterEngineUnregisterExternalTextureFnPtr UnregisterExternalTexture;
FlutterEngineMarkExternalTextureFrameAvailableFnPtr
MarkExternalTextureFrameAvailable;
Expand Down
9 changes: 9 additions & 0 deletions shell/platform/embedder/embedder_engine.cc
Original file line number Diff line number Diff line change
Expand Up @@ -153,6 +153,15 @@ bool EmbedderEngine::RegisterTexture(int64_t texture) {
return true;
}

bool EmbedderEngine::RegisterTexture(int64_t texture, FlutterTextureType type) {
if (!IsValid()) {
return false;
}
shell_->GetPlatformView()->RegisterTexture(
external_texture_resolver_->ResolveExternalTexture(texture, type));
return true;
}

bool EmbedderEngine::UnregisterTexture(int64_t texture) {
if (!IsValid()) {
return false;
Expand Down
2 changes: 2 additions & 0 deletions shell/platform/embedder/embedder_engine.h
Original file line number Diff line number Diff line change
Expand Up @@ -58,6 +58,8 @@ class EmbedderEngine {

bool RegisterTexture(int64_t texture);

bool RegisterTexture(int64_t texture, FlutterTextureType type);

bool UnregisterTexture(int64_t texture);

bool MarkTextureFrameAvailable(int64_t texture);
Expand Down
156 changes: 140 additions & 16 deletions shell/platform/embedder/embedder_external_texture_gl.cc
Original file line number Diff line number Diff line change
Expand Up @@ -5,6 +5,11 @@
#include "flutter/shell/platform/embedder/embedder_external_texture_gl.h"

#include "flutter/fml/logging.h"
#include "flutter/impeller/display_list/dl_image_impeller.h"
#include "flutter/impeller/renderer/backend/gles/context_gles.h"
#include "flutter/impeller/renderer/backend/gles/texture_gles.h"
#include "impeller/aiks/aiks_context.h"
#include "impeller/renderer/backend/gles/gles.h"
#include "include/core/SkCanvas.h"
#include "include/core/SkPaint.h"
#include "third_party/skia/include/core/SkAlphaType.h"
Expand Down Expand Up @@ -37,7 +42,7 @@ void EmbedderExternalTextureGL::Paint(PaintContext& context,
if (last_image_ == nullptr) {
last_image_ =
ResolveTexture(Id(), //
context.gr_context, //
context, //
SkISize::Make(bounds.width(), bounds.height()) //
);
}
Expand All @@ -55,12 +60,34 @@ void EmbedderExternalTextureGL::Paint(PaintContext& context,
}
}

sk_sp<DlImage> EmbedderExternalTextureGL::ResolveTexture(
// |flutter::Texture|
void EmbedderExternalTextureGL::OnGrContextCreated() {}

// |flutter::Texture|
void EmbedderExternalTextureGL::OnGrContextDestroyed() {}

// |flutter::Texture|
void EmbedderExternalTextureGL::MarkNewFrameAvailable() {
last_image_ = nullptr;
}

// |flutter::Texture|
void EmbedderExternalTextureGL::OnTextureUnregistered() {}

EmbedderExternalTextureSkiaGL::EmbedderExternalTextureSkiaGL(
int64_t texture_identifier,
const ExternalTextureCallback& callback)
: EmbedderExternalTextureGL(texture_identifier, callback) {}

EmbedderExternalTextureSkiaGL::~EmbedderExternalTextureSkiaGL() = default;

sk_sp<DlImage> EmbedderExternalTextureSkiaGL::ResolveTexture(
int64_t texture_id,
GrDirectContext* context,
PaintContext& context,
const SkISize& size) {
context->flushAndSubmit();
context->resetContext(kAll_GrBackendState);
GrDirectContext* gr_context = context.gr_context;
gr_context->flushAndSubmit();
gr_context->resetContext(kAll_GrBackendState);
std::unique_ptr<FlutterOpenGLTexture> texture =
external_texture_callback_(texture_id, size.width(), size.height());

Expand All @@ -83,7 +110,7 @@ sk_sp<DlImage> EmbedderExternalTextureGL::ResolveTexture(
width, height, skgpu::Mipmapped::kNo, gr_texture_info);
SkImages::TextureReleaseProc release_proc = texture->destruction_callback;
auto image =
SkImages::BorrowTextureFrom(context, // context
SkImages::BorrowTextureFrom(gr_context, // context
gr_backend_texture, // texture handle
kTopLeft_GrSurfaceOrigin, // origin
kRGBA_8888_SkColorType, // color type
Expand All @@ -99,26 +126,123 @@ sk_sp<DlImage> EmbedderExternalTextureGL::ResolveTexture(
if (release_proc) {
release_proc(texture->user_data);
}
FML_LOG(ERROR) << "Could not create external texture->";
return nullptr;
}

// This image should not escape local use by EmbedderExternalTextureGL
return DlImage::Make(std::move(image));
}

// |flutter::Texture|
void EmbedderExternalTextureGL::OnGrContextCreated() {}
EmbedderExternalTextureGLImpellerPixelBuffer::
EmbedderExternalTextureGLImpellerPixelBuffer(
int64_t texture_identifier,
const ExternalTextureCallback& callback)
: EmbedderExternalTextureGL(texture_identifier, callback) {}

// |flutter::Texture|
void EmbedderExternalTextureGL::OnGrContextDestroyed() {}
sk_sp<DlImage> EmbedderExternalTextureGLImpellerPixelBuffer::ResolveTexture(
int64_t texture_id,
PaintContext& context,
const SkISize& size) {
std::unique_ptr<FlutterOpenGLTexture> texture =
external_texture_callback_(texture_id, size.width(), size.height());

// |flutter::Texture|
void EmbedderExternalTextureGL::MarkNewFrameAvailable() {
last_image_ = nullptr;
if (!texture) {
return nullptr;
}

size_t width = size.width();
size_t height = size.height();

if (texture->width != 0 && texture->height != 0) {
width = texture->width;
height = texture->height;
}

impeller::TextureDescriptor desc;
desc.type = impeller::TextureType::kTexture2D;
impeller::AiksContext* aiks_context = context.aiks_context;
const auto& gl_context =
impeller::ContextGLES::Cast(*aiks_context->GetContext());
desc.storage_mode = impeller::StorageMode::kDevicePrivate;
desc.format = impeller::PixelFormat::kR8G8B8A8UNormInt;
desc.size = {static_cast<int>(width), static_cast<int>(height)};
desc.mip_count = 1;
auto textureGLES =
std::make_shared<impeller::TextureGLES>(gl_context.GetReactor(), desc);
if (!textureGLES->SetContents(texture->buffer, texture->buffer_size)) {
if (texture->destruction_callback) {
texture->destruction_callback(texture->user_data);
}
return nullptr;
}
if (texture->destruction_callback) {
texture->destruction_callback(texture->user_data);
}
return impeller::DlImageImpeller::Make(textureGLES);
}

// |flutter::Texture|
void EmbedderExternalTextureGL::OnTextureUnregistered() {}
EmbedderExternalTextureGLImpellerPixelBuffer::
~EmbedderExternalTextureGLImpellerPixelBuffer() = default;

EmbedderExternalTextureGLImpellerSurface::
EmbedderExternalTextureGLImpellerSurface(
int64_t texture_identifier,
const ExternalTextureCallback& callback)
: EmbedderExternalTextureGL(texture_identifier, callback) {}

sk_sp<DlImage> EmbedderExternalTextureGLImpellerSurface::ResolveTexture(
int64_t texture_id,
PaintContext& context,
const SkISize& size) {
std::unique_ptr<FlutterOpenGLTexture> texture =
external_texture_callback_(texture_id, size.width(), size.height());

if (!texture) {
return nullptr;
}
size_t width = size.width();
size_t height = size.height();

if (texture->width != 0 && texture->height != 0) {
width = texture->width;
height = texture->height;
}

impeller::TextureDescriptor desc;
desc.type = impeller::TextureType::kTextureExternalOES;
impeller::AiksContext* aiks_context = context.aiks_context;
const auto& gl_context =
impeller::ContextGLES::Cast(*aiks_context->GetContext());
desc.storage_mode = impeller::StorageMode::kDevicePrivate;
desc.format = impeller::PixelFormat::kR8G8B8A8UNormInt;
desc.size = {static_cast<int>(width), static_cast<int>(height)};
desc.mip_count = 1;
auto textureGLES = std::make_shared<impeller::TextureGLES>(
gl_context.GetReactor(), desc,
impeller::TextureGLES::IsWrapped::kWrapped);
textureGLES->SetCoordinateSystem(
impeller::TextureCoordinateSystem::kUploadFromHost);
if (!textureGLES->Bind()) {
if (texture->destruction_callback) {
texture->destruction_callback(texture->user_data);
}
return nullptr;
}

if (!texture->bind_callback(texture->user_data)) {
if (texture->destruction_callback) {
texture->destruction_callback(texture->user_data);
}
return nullptr;
}

if (texture->destruction_callback) {
texture->destruction_callback(texture->user_data);
}

return impeller::DlImageImpeller::Make(textureGLES);
}

EmbedderExternalTextureGLImpellerSurface::
~EmbedderExternalTextureGLImpellerSurface() = default;
} // namespace flutter
Loading

0 comments on commit 9ed9b65

Please sign in to comment.