openharmony bootanimation分析及常见错误
系统版本:Openharmony 4.1
硬件soc:rk3588
代码路径:
foundation/graphic/graphic_2d/frameworks/bootanimation/src/main.cpp
main的函数很简单,直接调用接口Run,接口位置在boot_animation.cpp中。
/*
* Copyright (c) 2023 Huawei Device Co., Ltd.
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
#include "boot_animation.h"
#include "boot_animation_utils.h"
#include "event_handler.h"
#include "rs_trace.h"
#include "transaction/rs_render_service_client.h"
#include "transaction/rs_interfaces.h"
#include "transaction/rs_transaction.h"
#ifdef NEW_RENDER_CONTEXT
#include "render_context_factory.h"
#include "rs_surface_factory.h"
#endif
#include "parameter.h"
#include "platform/common/rs_system_properties.h"
using namespace OHOS;
constexpr float MAX_ZORDER = 100000.0f;
constexpr float ROATION_DEGREE_Z = -89.9f;
constexpr int HALF_NUM = 2;
void BootAnimation::OnDraw(Rosen::Drawing::CoreCanvas* canvas, int32_t curNo)
{
if (canvas == nullptr) {
LOGE("OnDraw canvas is nullptr");
return;
}
if (curNo > (imgVecSize_ - 1) || curNo < 0) {
return;
}
//读取图片然后渲染
std::shared_ptr<ImageStruct> imgstruct = imageVector_[curNo];
std::shared_ptr<Rosen::Drawing::Image> image = imgstruct->imageData;
ROSEN_TRACE_BEGIN(HITRACE_TAG_GRAPHIC_AGP, "BootAnimation::OnDraw in drawRect");
Rosen::Drawing::Brush brush;
brush.SetColor(Rosen::Drawing::Color::COLOR_BLACK);
Rosen::Drawing::Rect bgRect(0.0, 0.0, windowWidth_, windowHeight_);
canvas->AttachBrush(brush);
canvas->DrawRect(bgRect);
canvas->DetachBrush();
ROSEN_TRACE_END(HITRACE_TAG_GRAPHIC_AGP);
ROSEN_TRACE_BEGIN(HITRACE_TAG_GRAPHIC_AGP, "BootAnimation::OnDraw in drawImageRect");
Rosen::Drawing::Rect rect(pointX_, pointY_, pointX_ + realWidth_, pointY_ + realHeight_);
Rosen::Drawing::SamplingOptions samplingOptions;
canvas->DrawImageRect(*image, rect, samplingOptions);
ROSEN_TRACE_END(HITRACE_TAG_GRAPHIC_AGP);
}
void BootAnimation::Draw()
{
if (picCurNo_ < (imgVecSize_ - 1)) {
picCurNo_ = picCurNo_ + 1;
} else {
CheckExitAnimation();
return;
}
ROSEN_TRACE_BEGIN(HITRACE_TAG_GRAPHIC_AGP, "BootAnimation::Draw RequestFrame");
auto frame = rsSurface_->RequestFrame(windowWidth_, windowHeight_);
if (frame == nullptr) {
LOGE("Draw frame is nullptr");
return;
}
ROSEN_TRACE_END(HITRACE_TAG_GRAPHIC_AGP);
#ifdef NEW_RENDER_CONTEXT
if (rsSurface_ == nullptr) {
LOGE("rsSurface is nullptr");
return;
}
auto canvas = rsSurface_->GetCanvas();
OnDraw(canvas, picCurNo_);
ROSEN_TRACE_BEGIN(HITRACE_TAG_GRAPHIC_AGP, "BootAnimation::Draw FlushFrame");
rsSurface_->FlushFrame();
ROSEN_TRACE_END(HITRACE_TAG_GRAPHIC_AGP);
#else
framePtr_ = std::move(frame);
auto canvas = framePtr_->GetCanvas();
OnDraw(canvas, picCurNo_);
ROSEN_TRACE_BEGIN(HITRACE_TAG_GRAPHIC_AGP, "BootAnimation::Draw FlushFrame");
rsSurface_->FlushFrame(framePtr_);
ROSEN_TRACE_END(HITRACE_TAG_GRAPHIC_AGP);
#endif
}
void BootAnimation::Init(Rosen::ScreenId defaultId, int32_t width, int32_t height)
{
defaultId_ = defaultId;
windowWidth_ = width;
windowHeight_ = height;
LOGI("Init enter, width: %{public}d, height: %{public}d", width, height);
InitPicCoordinates();
InitRsDisplayNode();
InitRsSurfaceNode();
#ifdef PLAYER_FRAMEWORK_ENABLE
if (animationConfig_.IsBootVideoEnabled()) {
LOGI("Init end");
return;
}
#endif
LOGI("Playing boot animation using sequence frames.");
system::SetParameter("bootevent.bootanimation.started", "true");
auto& rsClient = OHOS::Rosen::RSInterfaces::GetInstance();
while (receiver_ == nullptr) {
receiver_ = rsClient.CreateVSyncReceiver("BootAnimation", mainHandler_);
}
VsyncError ret = receiver_->Init();
if (ret) {
LOGE("vsync receiver init failed: %{public}d", ret);
PostTask(std::bind(&AppExecFwk::EventRunner::Stop, runner_));
return;
}
InitRsSurface();
ROSEN_TRACE_BEGIN(HITRACE_TAG_GRAPHIC_AGP, "BootAnimation::preload");
//解压 imageVector_ 图片的向量在OnDraw里面进行渲染
if (animationConfig_.ReadPicZipFile(imageVector_, freq_)) {
imgVecSize_ = imageVector_.size();
} else {
LOGE("Read PicZipFile failed");
PostTask(std::bind(&AppExecFwk::EventRunner::Stop, runner_));
return;
}
ROSEN_TRACE_END(HITRACE_TAG_GRAPHIC_AGP);
OHOS::Rosen::VSyncReceiver::FrameCallback fcb = {
.userData_ = this,
//OnVsync事件,当vsync信号接收到后,调用draw进行图片渲染
.callback_ = std::bind(&BootAnimation::OnVsync, this),
};
int32_t changefreq = static_cast<int32_t>((1000.0 / freq_) / 16);
//设置vsync频率,具体是软硬件的vsync需要看底层
ret = receiver_->SetVSyncRate(fcb, changefreq);
if (ret) {
PostTask(std::bind(&AppExecFwk::EventRunner::Stop, runner_));
LOGE("SetVSyncRate failed: %{public}d %{public}d %{public}d", ret, freq_, changefreq);
return;
} else {
LOGI("SetVSyncRate success: %{public}d %{public}d", freq_, changefreq);
}
}
void BootAnimation::Run(Rosen::ScreenId id, int screenWidth, int screenHeight)
{
LOGI("Run enter");
//解析配置文件,目前没使用配置文件,所以直接使用代码中声明的默认位置
animationConfig_.ParserCustomCfgFile();
Rosen::RSInterfaces& interface = Rosen::RSInterfaces::GetInstance();
if (animationConfig_.GetRotateScreenId() >= 0) {
id = interface.GetActiveScreenId();
LOGI("GetActiveScreenId: " BPUBU64 "", id);
Rosen::RSScreenModeInfo modeinfo = interface.GetScreenActiveMode(id);
screenWidth = modeinfo.GetScreenWidth();
screenHeight = modeinfo.GetScreenHeight();
LOGI("screenWidth: %{public}d, screenHeight: %{public}d", screenWidth, screenHeight);
if (id > 0) {
LOGI("SetScreenPowerStatus POWER_STATUS_OFF_FAKE: 0");
interface.SetScreenPowerStatus(0, Rosen::ScreenPowerStatus::POWER_STATUS_OFF_FAKE);
LOGI("SetScreenPowerStatus POWER_STATUS_ON: " BPUBU64 "", id);
interface.SetScreenPowerStatus(id, Rosen::ScreenPowerStatus::POWER_STATUS_ON);
}
} else if (interface.GetScreenPowerStatus(id) != Rosen::ScreenPowerStatus::POWER_STATUS_ON) {
interface.SetScreenPowerStatus(id, Rosen::ScreenPowerStatus::POWER_STATUS_ON);
}
//创建一个事件接收器
runner_ = AppExecFwk::EventRunner::Create(false);
//接收的handler对象创建runner_
mainHandler_ = std::make_shared<AppExecFwk::EventHandler>(runner_);
//发送事件启动init
mainHandler_->PostTask(std::bind(&BootAnimation::Init, this, id, screenWidth, screenHeight));
LOGI("PostTask Init");
#ifdef PLAYER_FRAMEWORK_ENABLE
if (animationConfig_.IsBootVideoEnabled()) {
mainHandler_->PostTask(std::bind(&BootAnimation::PlayVideo, this));
mainHandler_->PostTask(std::bind(&BootAnimation::PlaySound, this));
LOGI("PostTask PlayVideo");
} else {
mainHandler_->PostTask(std::bind(&BootAnimation::PlaySound, this));
LOGI("PostTask PlaySound");
}
#else
LOGI("player_framework part is not enabled.");
#endif
runner_->Run();
}
void BootAnimation::InitRsSurfaceNode()
{
LOGI("Init RsSurfaceNode enter");
struct Rosen::RSSurfaceNodeConfig rsSurfaceNodeConfig;
rsSurfaceNodeConfig.SurfaceNodeName = "BootAnimationNode";
rsSurfaceNodeConfig.isSync = true;
Rosen::RSSurfaceNodeType rsSurfaceNodeType = Rosen::RSSurfaceNodeType::SELF_DRAWING_WINDOW_NODE;
rsSurfaceNode_ = Rosen::RSSurfaceNode::Create(rsSurfaceNodeConfig, rsSurfaceNodeType);
if (!rsSurfaceNode_) {
LOGE("rsSurfaceNode_ is nullptr");
return;
}
rsSurfaceNode_->SetRotation(0, 0, ROATION_DEGREE_Z);
rsSurfaceNode_->SetPositionZ(MAX_ZORDER);
rsSurfaceNode_->SetBounds({0, 0, windowWidth_, windowHeight_});
rsSurfaceNode_->SetBackgroundColor(0xFF000000);
rsSurfaceNode_->SetFrameGravity(Rosen::Gravity::RESIZE_ASPECT);
rsSurfaceNode_->SetBootAnimation(true);
OHOS::Rosen::RSTransaction::FlushImplicitTransaction();
rsSurfaceNode_->AttachToDisplay(defaultId_);
OHOS::Rosen::RSTransaction::FlushImplicitTransaction();
system::SetParameter("bootevent.bootanimation.ready", "true");
LOGI("Set bootevent.bootanimation.ready true");
}
void BootAnimation::InitRsSurface()
{
LOGI("InitRsSurface");
#if defined(NEW_RENDER_CONTEXT)
renderContext_ = Rosen::RenderContextBaseFactory::CreateRenderContext();
if (renderContext_ == nullptr) {
LOGE("Create Render Context failed");
return;
}
renderContext_->Init();
std::shared_ptr<Rosen::DrawingContext> drawingContext = std::make_shared<Rosen::DrawingContext>(
renderContext_->GetRenderType());
sptr<Surface> surface = rsSurfaceNode_->GetSurface();
drawingContext->SetUpDrawingContext();
rsSurface_ = Rosen::RSSurfaceFactory::CreateRSSurface(Rosen::PlatformName::OHOS, surface, drawingContext);
rsSurface_->SetRenderContext(renderContext_);
#else
rsSurface_ = OHOS::Rosen::RSSurfaceExtractor::ExtractRSSurface(rsSurfaceNode_);
if (rsSurface_ == nullptr) {
LOGE("rsSurface is nullptr");
return;
}
#ifdef ACE_ENABLE_GL
if (Rosen::RSSystemProperties::GetGpuApiType() == OHOS::Rosen::GpuApiType::OPENGL) {
rc_ = OHOS::Rosen::RenderContextFactory::GetInstance().CreateEngine();
if (rc_ == nullptr) {
LOGE("InitilizeEglContext failed");
return;
} else {
LOGI("init egl context");
rc_->InitializeEglContext();
rsSurface_->SetRenderContext(rc_);
}
}
#endif // ACE_ENABLE_GL
if (rc_ == nullptr) {
LOGI("rc is nullptr, use cpu");
}
#endif
}
BootAnimation::~BootAnimation()
{
if (rsSurfaceNode_) {
rsSurfaceNode_->DetachToDisplay(defaultId_);
}
if (rsDisplayNode_) {
rsDisplayNode_->RemoveFromTree();
}
OHOS::Rosen::RSTransaction::FlushImplicitTransaction();
}
void BootAnimation::InitPicCoordinates()
{
if (windowWidth_ >= windowHeight_) {
realHeight_ = windowHeight_;
realWidth_ = realHeight_;
pointX_ = (windowWidth_ - realWidth_) / HALF_NUM;
} else {
realWidth_ = windowWidth_;
realHeight_ = realWidth_;
pointY_ = (windowHeight_ - realHeight_) / HALF_NUM;
}
}
void BootAnimation::OnVsync()
{
PostTask(std::bind(&BootAnimation::Draw, this));
}
bool BootAnimation::CheckExitAnimation()
{
if (!isAnimationEnd_) {
LOGI("Boot animation is end");
system::SetParameter("bootevent.bootanimation.finished", "true");
isAnimationEnd_ = true;
}
std::string bootEventCompleted = system::GetParameter("bootevent.boot.completed", "false");
if (bootEventCompleted == "true") {
mainHandler_->PostTask(std::bind(&AppExecFwk::EventRunner::Stop, runner_));
LOGI("Read bootevent.boot.completed is true");
return true;
}
return false;
}
#ifdef PLAYER_FRAMEWORK_ENABLE
void BootAnimation::PlaySound()
{
LOGI("PlaySound start");
bool bootSoundEnabled = BootAnimationUtils::GetBootAnimationSoundEnabled();
if (bootSoundEnabled == true) {
LOGI("PlaySound read bootSoundEnabled is true");
if (soundPlayer_ == nullptr) {
soundPlayer_ = Media::PlayerFactory::CreatePlayer();
}
const int maxFailCount = 3;
for (int i = 0; i < maxFailCount; i++) {
if (soundPlayer_ != nullptr) {
break;
}
soundPlayer_ = Media::PlayerFactory::CreatePlayer();
usleep(SLEEP_TIME_US);
}
if (soundPlayer_ == nullptr) {
LOGE("CreatePlayer fail, soundPlayer_ is nullptr");
return;
}
std::string uri = animationConfig_.GetSoundUrl();
soundPlayer_->SetSource(uri);
soundPlayer_->SetLooping(false);
soundPlayer_->PrepareAsync();
soundPlayer_->Play();
}
LOGI("PlaySound end");
}
#endif
#ifdef PLAYER_FRAMEWORK_ENABLE
void BootAnimation::PlayVideo()
{
LOGI("PlayVideo start w:%{public}d h:%{public}d", windowWidth_, windowHeight_);
if (!animationConfig_.IsBootVideoEnabled()) {
return;
}
fcb_ = {
.userData_ = this,
.callback_ = std::bind(&BootAnimation::CloseVideoPlayer, this),
};
LOGI("PlayVideo setVideo screenId:%{public}d", (int32_t)defaultId_);
bootVideoPlayer_ = std::make_shared<BootVideoPlayer>();
bootVideoPlayer_->SetVideoPath(
defaultId_ == 0 ? animationConfig_.GetBootVideoPath() : animationConfig_.GetBootExtraVideoPath());
bootVideoPlayer_->SetPlayerSurface(rsSurfaceNode_ ? rsSurfaceNode_->GetSurface() : nullptr);
bootVideoPlayer_->SetCallback(&fcb_);
if (!bootVideoPlayer_->PlayVideo()) {
LOGE("Play video failed.");
CloseVideoPlayer();
}
}
#endif
#ifdef PLAYER_FRAMEWORK_ENABLE
void BootAnimation::CloseVideoPlayer()
{
LOGI("Close video player.");
while (!CheckExitAnimation()) {
usleep(SLEEP_TIME_US);
}
LOGI("Check Exit Animation end.");
}
#endif
void BootAnimation::InitRsDisplayNode()
{
OHOS::Rosen::RSDisplayNodeConfig config = {defaultId_, false, 0};
rsDisplayNode_ = OHOS::Rosen::RSDisplayNode::Create(config);
if (rsDisplayNode_ == nullptr) {
LOGE("Failed to init display node!");
return;
}
rsDisplayNode_->SetDisplayOffset(0, 0);
rsDisplayNode_->SetFrame(0, 0, windowWidth_, windowHeight_);
rsDisplayNode_->SetBounds(0, 0, windowWidth_, windowHeight_);
rsDisplayNode_->SetBootAnimation(true);
// flush transaction
auto transactionProxy = OHOS::Rosen::RSTransactionProxy::GetInstance();
if (transactionProxy != nullptr) {
transactionProxy->FlushImplicitTransaction();
}
LOGD("InitRsDisplayNode success");
}