Android12 添加开机铃声

系统默认是没有播放开机铃声的功能,MTK有一套自己的开机铃声处理逻辑,代码在/vendor/mediatek/proprietary/operator/frameworks/bootanimation/MtkBootanimation下,但是在10之后MTK就不在维护这部分代码了。直接使用会有很多编译报错,现在把MTK播放铃声的逻辑移植过来。

移植铃声播放功能
  • /frameworks/base/cmds/bootanimation/BootAnimation.cpp
index 958558669ef..3b4eb12ee00 100755
--- a/frameworks/base/cmds/bootanimation/BootAnimation.cpp
+++ b/frameworks/base/cmds/bootanimation/BootAnimation.cpp
@@ -59,10 +59,20 @@#include "BootAnimation.h"+#include <media/IMediaHTTPService.h>
+#include <media/mediaplayer.h>
+#include <media/MediaPlayerInterface.h>
+#define ANIM_PATH_MAX 255#define STR(x)   #x#define STRTO(x) STR(x)+#define PATH_COUNT 2
+static const char* mAudioPath[2][PATH_COUNT] =
+    {{"/system/media/bootaudio.mp3", "/custom/media/bootaudio.mp3"} , /*  bootaudio path  */
+     {"/system/media/shutaudio.mp3", "/custom/media/shutaudio.mp3"} /*  shutaudio path  */
+    };
+namespace android {using ui::DisplayMode;
@@ -123,6 +133,11 @@ BootAnimation::BootAnimation(sp<Callbacks> callbacks)} else {mShuttingDown = true;}
+
+    bPlayMP3 = true;
+    mBootVideoPlayState = MEDIA_NOP;
+    bAudioStarted = false;
+ALOGD("%sAnimationStartTiming start time: %" PRId64 "ms", mShuttingDown ? "Shutdown" : "Boot",elapsedRealtime());}
@@ -136,6 +151,50 @@ BootAnimation::~BootAnimation() {elapsedRealtime());}+BootVideoListener::BootVideoListener(const sp<BootAnimation> &bootanim) {
+    ALOGD("[MtkBootAnimation %s %d]",__FUNCTION__,__LINE__);
+    mBootanim = bootanim;
+}
+BootVideoListener::~BootVideoListener() {
+    ALOGD("[MtkBootAnimation %s %d]",__FUNCTION__,__LINE__);
+}
+void BootVideoListener::notify(int msg, int ext1, int ext2, const Parcel *obj) {
+    ALOGD("[MtkBootAnimation %s %d] msg=%d ext1=%d ext2=%d",__FUNCTION__,__LINE__, msg, ext1, ext2);
+    if(msg == MEDIA_PLAYBACK_COMPLETE || msg == MEDIA_SEEK_COMPLETE) {
+        mBootanim->setBootVideoPlayState(MEDIA_PLAYBACK_COMPLETE);
+        ALOGD("[MtkBootAnimation %s %d] media player complete",__FUNCTION__,__LINE__);
+    }
+    if(msg == MEDIA_ERROR || msg == MEDIA_SKIPPED) {
+        mBootanim->setBootVideoPlayState(MEDIA_ERROR);
+        ALOGD("[MtkBootAnimation %s %d] media player error",__FUNCTION__,__LINE__);
+    }
+    if(msg == MEDIA_STARTED) {
+        mBootanim->setBootAudioStarted();
+        ALOGD("[MtkBootAnimation %s %d] AudioStarted",__FUNCTION__,__LINE__);
+    }
+#ifdef MTK_AOSP_ENHANCEMENT
+    if(msg ==  MEDIA_ERROR_TYPE_NOT_SUPPORTED || msg ==  MEDIA_ERROR_BAD_FILE
+        || msg == MEDIA_ERROR_CANNOT_CONNECT_TO_SERVER) {
+        mBootanim->setBootVideoPlayState(MEDIA_ERROR);
+        ALOGD("[MtkBootAnimation %s %d] media player error",__FUNCTION__,__LINE__);
+    }
+#endif
+    if(obj == NULL){
+        ALOGD("[MtkBootAnimation %s %d]obj is null \n",__FUNCTION__,__LINE__);
+    }
+}
+void BootAnimation::setBootVideoPlayState(int playState){
+    mBootVideoPlayState = playState;
+    ALOGD("[MtkBootAnimation %s %d]mBootVideoPlayState=%d",__FUNCTION__,__LINE__, mBootVideoPlayState);
+}
+
+void BootAnimation::setBootAudioStarted(void)
+{
+    AutoMutex _l(mMyLock);
+    bAudioStarted = true;
+    mCondition.signal();
+}
+void BootAnimation::onFirstRef() {status_t err = mSession->linkToComposerDeath(this);SLOGE_IF(err, "linkToComposerDeath failed (%s) ", strerror(-err));
@@ -606,6 +665,51 @@ void BootAnimation::findBootAnimationFile() {}bool BootAnimation::threadLoop() {
+    sp<MediaPlayer> mediaplayer;
+    const char* resourcePath = initAudioPath();
+    status_t mediastatus = NO_ERROR;
+    if (resourcePath != NULL) {
+        bPlayMP3 = true;
+        ALOGD("sound file path: %s", resourcePath);
+        mediaplayer = new MediaPlayer();
+        mediastatus = mediaplayer->setDataSource(NULL, resourcePath, NULL);
+
+        sp<BootVideoListener> listener = new BootVideoListener(this);
+        mediaplayer->setListener(listener);
+
+        if (mediastatus == NO_ERROR) {
+            ALOGD("mediaplayer is initialized");
+            Parcel* attributes = new Parcel();
+            attributes->writeInt32(AUDIO_USAGE_MEDIA);            //usage
+            attributes->writeInt32(AUDIO_CONTENT_TYPE_MUSIC);     //audio_content_type_t
+            attributes->writeInt32(AUDIO_SOURCE_DEFAULT);         //audio_source_t
+            attributes->writeInt32(0);                            //audio_flags_mask_t
+            attributes->writeInt32(1);                            //kAudioAttributesMarshallTagFlattenTags of mediaplayerservice.cpp
+            attributes->writeString16(String16("BootAnimationAudioTrack")); // tags
+            mediaplayer->setParameter(KEY_PARAMETER_AUDIO_ATTRIBUTES, *attributes);
+            mediaplayer->setAudioStreamType(AUDIO_STREAM_MUSIC);
+            //mediaplayer->setVolume(1.0, 1.0);
+            mediastatus = mediaplayer->prepare();
+        }
+        if (mediastatus == NO_ERROR) {
+            ALOGD("media player is prepared");
+            mediastatus = mediaplayer->start();
+        }
+
+    }else{
+        bPlayMP3 = false;
+    }
+
+    //mediaplayer start is an async API, and it will be ready after mediaserver parsing all
+    // components which will take some time and cause aduio and video not sync.
+    if (bPlayMP3 && mediastatus == NO_ERROR) {
+        AutoMutex _l(mMyLock);
+        if (!bAudioStarted) {
+            status_t err = mCondition.waitRelative(mMyLock, s2ns(40));
+            ALOGD("audio started re %d", err);
+        }
+    }
+bool result;// We have no bootanimation file, so we use the stock android logo// animation.
@@ -615,6 +719,15 @@ bool BootAnimation::threadLoop() {result = movie();}+    if (resourcePath != NULL) {
+        if (mediastatus == NO_ERROR) {
+            ALOGD("mediaplayer was stareted successfully, now it is going to be stoped");
+            mediaplayer->stop();
+            mediaplayer->disconnect();
+            mediaplayer.clear();
+        }
+    }
+mCallbacks->shutdown();eglMakeCurrent(mDisplay, EGL_NO_SURFACE, EGL_NO_SURFACE, EGL_NO_CONTEXT);eglDestroyContext(mDisplay, mContext);
@@ -694,7 +807,13 @@ bool BootAnimation::android() {if (sleepTime > 0)usleep(sleepTime);-        checkExit();
+        if(!bPlayMP3){
+            checkExit();
+        }else{
+            if(mBootVideoPlayState == MEDIA_PLAYBACK_COMPLETE || mBootVideoPlayState == MEDIA_ERROR) {
+               checkExit();
+            }
+        }} while (!exitPending());glDeleteTextures(1, &mAndroid[0].name);
@@ -1363,7 +1482,13 @@ bool BootAnimation::playAnimation(const Animation& animation) {} while (err<0 && errno == EINTR);}-                checkExit();
+                if(!bPlayMP3){
+                    checkExit();
+                }else{
+                    if(mBootVideoPlayState == MEDIA_PLAYBACK_COMPLETE || mBootVideoPlayState == MEDIA_ERROR) {
+                       checkExit();
+                    }
+                }}usleep(part.pause * ns2us(frameDuration));
@@ -1612,6 +1737,23 @@ status_t BootAnimation::TimeCheckThread::readyToRun() {return NO_ERROR;}+const char* BootAnimation::initAudioPath() {
+    int index = 0;
+    if (mShuttingDown) {
+        index = 1;
+    } else {
+        index = 0;
+    }
+
+    for (int i = 0; i < PATH_COUNT; i++) {
+        if (access(mAudioPath[index][i], F_OK) == 0) {
+            ALOGD("initAudioPath: audio path = %s", mAudioPath[index][i]);
+            return mAudioPath[index][i];
+        }
+    }
+    return NULL;
+}
+// ---------------------------------------------------------------------------} // namespace android
  • /frameworks/base/cmds/bootanimation/BootAnimation.h
index f8a31c6d879..2e3c0c2430e 100755
--- a/frameworks/base/cmds/bootanimation/BootAnimation.h
+++ b/frameworks/base/cmds/bootanimation/BootAnimation.h
@@ -33,6 +33,8 @@#include <EGL/egl.h>#include <GLES/gl.h>+#include <media/mediaplayer.h>
+namespace android {class Surface;
@@ -129,6 +131,8 @@ public:explicit BootAnimation(sp<Callbacks> callbacks);virtual ~BootAnimation();
+    void setBootVideoPlayState(int playState);
+    void setBootAudioStarted(void);sp<SurfaceComposerClient> session() const;@@ -193,6 +197,14 @@ private:void handleViewport(nsecs_t timestep);+    const char* initAudioPath();
+
+    bool bPlayMP3;
+    bool bAudioStarted;
+    int mBootVideoPlayState;
+    Mutex  mMyLock;
+    Condition mCondition;
+sp<SurfaceComposerClient>       mSession;AssetManager mAssets;Texture     mAndroid[2];
@@ -220,6 +232,14 @@ private:Animation* mAnimation = nullptr;};+class BootVideoListener: public MediaPlayerListener{
+public:
+                BootVideoListener(const sp<BootAnimation> &player);
+    virtual     ~BootVideoListener();
+    virtual void notify(int msg, int ext1, int ext2, const Parcel *obj);
+    sp<BootAnimation> mBootanim;
+};
+// ---------------------------------------------------------------------------}; // namespace android
添加lib库
  • /frameworks/base/cmds/bootanimation/Android.bp
--- a/frameworks/base/cmds/bootanimation/Android.bp
+++ b/frameworks/base/cmds/bootanimation/Android.bp
@@ -73,5 +73,6 @@ cc_library_shared {"libEGL","libGLESv1_CM","libgui",
+        "libmedia",],}
第一次开机铃声不全
  • /frameworks/av/services/audioflinger/Threads.cpp
--- a/frameworks/av/services/audioflinger/Threads.cpp
+++ b/frameworks/av/services/audioflinger/Threads.cpp
@@ -5734,6 +5734,17 @@ AudioFlinger::PlaybackThread::mixer_state AudioFlinger::MixerThread::prepareTractrack->mHasVolumeController = false;}+            //add begin
+            char mvalue[PROPERTY_VALUE_MAX];
+            property_get("service.bootanim.exit", mvalue, "");
+            if (strcmp(mvalue,"0") == 0){
+                ALOGE("bootanim volume setting");
+                vlf = 0.25f;
+                vrf = 0.25f;
+            }
+            //add end
+// XXX: these things DON'T need to be done each timemAudioMixer->setBufferProvider(trackId, track);mAudioMixer->enable(trackId);
添加权限
  • /device/mediatek/sepolicy/bsp/plat_private/audioserver.te
--- a/device/mediatek/sepolicy/bsp/plat_private/audioserver.te
+++ b/device/mediatek/sepolicy/bsp/plat_private/audioserver.te
@@ -9,3 +9,5 @@ allow audioserver mtk_perf_service:service_manager find;allow audioserver debuglog_data_file:dir { relabelto create_dir_perms };allow audioserver debuglog_data_file:file create_file_perms;
+
+allow audioserver bootanim_system_prop:file { read open getattr map};
复制铃声文件
  • /vendor/audio-logo/products/resource-copy.mk
LOCAL_PATH:= vendor/audio-logo/
#boot animation
zip_files:= $(notdir $(wildcard $(LOCAL_PATH)/animation/*.zip))
PRODUCT_COPY_FILES += $(foreach file, $(zip_files), \$(LOCAL_PATH)/animation/$(file):system/media/$(file))#boot/shut audio
mp3_files:= $(notdir $(wildcard $(LOCAL_PATH)/animation/*.mp3))
PRODUCT_COPY_FILES += $(foreach file, $(mp3_files), \$(LOCAL_PATH)/animation/$(file):system/media/$(file))

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/diannao/72576.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

3.6V-30V宽压输入降压同步IC内置MOS,电流4A/5A/6A,可以满足汽车应急电源,BMS电池,电池组USB口输出等储能应用

今天给大家介绍一下这三款产品&#xff0c;分别是CJ92340,输入电压4.5V-30V&#xff0c;输出可调&#xff0c;电流负载能力可达4A&#xff0c;频率350KHZ。CJ92350,输入电压3.6V-30V&#xff0c;输出可调&#xff0c;频率可调&#xff0c;带载能力达5A。CJ92360,输入电压3.6V-3…

代码随想录算法训练营第35天 | 01背包问题二维、01背包问题一维、416. 分割等和子集

一、01背包问题二维 二维数组&#xff0c;一维为物品&#xff0c;二维为背包重量 import java.util.Scanner;public class Main{public static void main(String[] args){Scanner scanner new Scanner(System.in);int n scanner.nextInt();int bag scanner.nextInt();int[…

010---基于Verilog HDL的分频器设计

文章目录 摘要一、时序图二、程序设计2.1 rtl2.2 tb 三、仿真分析四、实用性 摘要 文章为学习记录。绘制时序图&#xff0c;编码。通过修改分频值参数&#xff0c;实现一定范围分频值内的任意分频器设计。 一、时序图 二、程序设计 2.1 rtl module divider #(parameter D…

维度建模事实表技术基础解析(以电商场景为例)

维度建模事实表技术基础解析(以电商场景为例) 1. 事实表结构 定义:事实表是维度建模的核心,由外键(关联维度表)、度量值(可量化的业务指标)及退化维度(冗余的维度属性)组成。其本质是记录业务过程中的度量事件,例如电商订单金额、商品库存量等。 场景识别:适用于…

Redis 主从复制、哨兵与集群的关系及工作原理详解

一、核心概念与关系 Redis 的 主从复制、哨兵&#xff08;Sentinel&#xff09; 和 集群&#xff08;Cluster&#xff09; 是逐步演进的高可用与分布式解决方案&#xff0c;三者关系如下&#xff1a; 主从复制&#xff1a;数据冗余与读写分离的基础。 哨兵&#xff1a;在主从…

确认机制的分类及其区别与联系探讨

在传输控制中&#xff0c;确认机制&#xff08;ACK 机制&#xff09;作为反馈模块在实现拥塞控制、丢包恢复和状态监测等功能中起到了至关重要的作用。今天我将基于之前发表的论文研究成果&#xff0c;对确认机制的分类进行系统梳理&#xff0c;并讨论各类机制之间的区别与联系…

115 道 MySQL 面试题,从简单到深入!

1. 什么是数据库事务&#xff1f; 数据库事务是一个作为单个逻辑工作单元执行的一系列操作。事务具有ACID属性&#xff0c;即原子性&#xff08;Atomicity&#xff09;、一致性&#xff08;Consistency&#xff09;、隔离性&#xff08;Isolation&#xff09;和持久性&#xf…

Linux - 网络套接字

一、网络编程 1&#xff09;地址结构 1. IP地址结构 struct in_addr&#xff1a;是用于表示 IPv4 地址 的结构体&#xff0c;定义在头文件 <netinet/in.h> 中。它的主要作用是存储一个 32 位的 IPv4 地址&#xff0c;通常与 struct sockaddr_in 一起使用。 struct in_a…

程序员学商务英语之Visiting the Factory

Dialogue-1 Arranging a Visit安排参观 I was wondering if you would / could lend me a million bucks, you know, I’m trying to start / run my own business. 我想知道你是否能够借给我一百万美金&#xff0c;你知道&#xff0c;我正在创业。 Take off your tie befor…

机器视觉运动控制一体机在天地盖同步跟随贴合解决方案

市场应用背景 纸盒天地盖是一种包装形式&#xff0c;广泛应用于消费电子、食品礼盒、奢侈品及化妆品等领域。其采用高强度纸板&#xff0c;经过预组装处理&#xff0c;结构坚固稳定&#xff0c;能有效保护产品并提升品牌形象。随着包装行业快速发展&#xff0c;市场对天地盖的…

【智能体Agent】ReAct智能体的实现思路和关键技术

基于ReAct&#xff08;Reasoning Acting&#xff09;框架的自主智能体 import re from typing import List, Tuplefrom langchain_community.chat_message_histories.in_memory import ChatMessageHistory from langchain_core.language_models.chat_models import BaseChatM…

Electron打包工具对比

在 Electron 生态中&#xff0c;打包工具的选择直接影响开发效率、配置复杂度和最终应用的性能。以下是主流的 Electron 打包工具及其优劣分析&#xff0c;结合你的 Vue 项目需求&#xff0c;我会在最后给出推荐方案&#xff1a; 一、主流 Electron 打包工具对比 1. Electron …

云原生系列之本地k8s环境搭建

前置条件 Windows 11 家庭中文版&#xff0c;版本号 23H2 云原生环境搭建 操作系统启用wsl(windows subsystem for linux) 开启wsl功能&#xff0c;如下图 安装并开启github加速器 FastGithub 2.1 下载地址&#xff1a;点击下载 2.2 解压安装文件fastgithub_win-x64.zip 2…

【计算机网络入门】TCP拥塞控制

目录 1. TCP拥塞控制和TCP流量控制的区别 2. 检测到拥塞该怎么办 2.1 如何判断网络拥塞&#xff1f; 3. 慢开始算法 拥塞避免算法 4.快重传事件->快恢复算法 5. 总结 1. TCP拥塞控制和TCP流量控制的区别 TCP流量控制是控制端对端的数据发送量。是局部的概念。 TCP拥…

Spring Boot 整合 JMS-ActiveMQ,并安装 ActiveMQ

1. 安装 ActiveMQ 1.1 下载 ActiveMQ 访问 ActiveMQ 官方下载页面&#xff0c;根据你的操作系统选择合适的版本进行下载。这里以 Linux 系统&#xff0c;Java环境1.8版本为例&#xff0c;下载 apache-activemq-5.16.7-bin.tar.gz。 1.2 解压文件 将下载的压缩包解压到指定目…

《几何原本》命题I.13

《几何原本》命题I.13 两条直线相交&#xff0c;邻角是两个直角或者相加等于 18 0 ∘ 180^{\circ} 180∘。 若两角相等&#xff0c;则根据定义&#xff0c;两角为直角。 两角若不相等&#xff0c;如图&#xff0c;则 ( ∠ 1 ∠ 2 ) ∠ 3 ∠ 1 ( ∠ 2 ∠ 3 ) 9 0 ∘ …

优先级队列:通过堆的形式实现

描述: 大顶堆: 小顶堆: 索引位置查找: 代码实现: package com.zy.queue_code.deque;/*** @Author: zy* @Date: 2025-03-05-15:51* @Description:*/ public interface Priority

《OpenCV》—— dlib库

文章目录 dlib库是什么&#xff1f;OpenCV库与dlib库对比dlib库安装dlib——人脸应用实例——人脸检测dlib——人脸应用实例——人脸关键点定位dlib——人脸应用实例——人脸轮廓绘制 dlib库是什么&#xff1f; OpenCV库与dlib库对比 dlib库安装 dlib——人脸应用实例——人脸检…

蓝桥与力扣刷题(蓝桥 旋转)

题目&#xff1a;图片旋转是对图片最简单的处理方式之一&#xff0c;在本题中&#xff0c;你需要对图片顺时针旋转 90 度。 我们用一个 nm的二维数组来表示一个图片&#xff0c;例如下面给出一个 34 的 图片的例子&#xff1a; 1 3 5 7 9 8 7 6 3 5 9 7 这个图片顺时针旋转…

随机播放音乐 伪随机

import java.util.*;/*** https://cloud.tencent.com.cn/developer/news/1045747* 伪随机播放音乐*/ public class MusicPlayer {private List<String> allSongs; // 所有歌曲列表private List<String> playedSongs; // 已经播放过的歌曲列表private Map<String…