MediaScannerService研究
侯 亮
(本文以Android 5.1为准)
1 概述
MediaScannerService是Android平台提供的一个用于扫描手机中多媒体文件的应用级service。它并不是系统服务。MediaScannerService和MediaProvider有着非常紧密的关系,因为扫描出的结果总需要存储到某个地方来展现给用户。那么它们具体是如何结合的呢?本文将逐步加以阐述。
我们先来初步了解一下MediaScannerService,它在AndroidManifest.xml文件里的相关信息如下:
【packages/providers/mediaprovider/AndroidManifest.xml】
<service android:name="MediaScannerService" android:exported="true">
<intent-filter>
<action android:name="android.media.IMediaScannerService" />
</intent-filter>
</service>
MediaScannerService本身继承于Service,而且还实现了Runnable接口。其定义截选如下:
【packages/providers/mediaprovider/src/com/android/providers/media/MediaScannerService.java】
public class MediaScannerService extends Service implements Runnable
{
private static final String TAG = "MediaScannerService";
private volatile Looper mServiceLooper;
private volatile ServiceHandler mServiceHandler;
private PowerManager.WakeLock mWakeLock;
private String[] mExternalStoragePaths;
. . . . . .
private final IMediaScannerService.Stub mBinder = new IMediaScannerService.Stub()
. . . . . .
. . . . . .
}
1.1 在onCreate()中启动工作线程
MediaScannerService的onCreate()函数如下:
【packages/providers/mediaprovider/src/com/android/providers/media/MediaScannerService.java】
@Override
public void onCreate()
{
PowerManager pm = (PowerManager)getSystemService(Context.POWER_SERVICE);
mWakeLock = pm.newWakeLock(PowerManager.PARTIAL_WAKE_LOCK, TAG);
StorageManager storageManager =
(StorageManager)getSystemService(Context.STORAGE_SERVICE);
mExternalStoragePaths = storageManager.getVolumePaths();
// 启动最重要的工作线程,该线程也是个消息泵线程
Thread thr = new Thread(null, this, "MediaScannerService");
thr.start();
}
可以看到,onCreate()里会启动最重要的工作线程,该线程也是个消息泵线程。每当用户需要扫描媒体文件时,基本上都是在向这个消息泵里发送Message,并在处理Message时完成真正的scan动作。请注意,创建Thread时传入的第二个参数就是MediaScannerService自身,也就是说线程的主要行为其实就是MediaScannerService的run()函数,该函数的代码如下:
public void run()
{
Process.setThreadPriority(Process.THREAD_PRIORITY_BACKGROUND +
Process.THREAD_PRIORITY_LESS_FAVORABLE);
Looper.prepare();
mServiceLooper = Looper.myLooper(); // 消息looper
mServiceHandler = new ServiceHandler(); // 发送消息的handler
Looper.loop();
}
后续就是通过上面那个mServiceHandler向消息队列发送Message的。
1.2 向工作线程发送Message
比较常见的向消息泵发送Message的做法是调用startService(),并在MediaScannerService的onStartCommand()函数里sendMessage()。比如,和MediaScannerService配套提供的MediaScannerReceiver,当它收到类似ACTION_BOOT_COMPLETED这样的系统广播时,就会调用自己的scan()或scanFile()函数。而scan()函数的代码如下:
【packages/providers/mediaprovider/src/com/android/providers/media/MediaScannerReceiver.java】
private void scan(Context context, String volume) {
Bundle args = new Bundle();
args.putString("volume", volume);
context.startService( new Intent(context,
MediaScannerService.class).putExtras(args));
}
startService()动作会导致走到service的onStartCommand(),并进一步发送消息,其函数截选如下:
@Override
public int onStartCommand(Intent intent, int flags, int startId)
{
. . . . . .
. . . . . .
Message msg = mServiceHandler.obtainMessage();
msg.arg1 = startId;
msg.obj = intent.getExtras();
mServiceHandler.sendMessage(msg); // 发送消息!
// Try again later if we are killed before we can finish scanning.
return Service.START_REDELIVER_INTENT;
}
另外一种比较常见的发送Message的做法是先直接或间接bindService(),绑定成功后会得到一个IMediaScannerService接口,而后外界再通过该接口向MediaScannerService发起命令,请求其扫描特定文件或目录。
IMediaScannerService接口只提供了两个接口函数:
- void requestScanFile(String path, String mimeType, in IMediaScannerListener listener);
- void scanFile(String path, String mimeType);
处理这两种请求的实体是服务内部的mBinder对象,参考代码如下:
【packages/providers/mediaprovider/src/com/android/providers/media/MediaScannerService.java】
private final IMediaScannerService.Stub mBinder = new IMediaScannerService.Stub() {
public void requestScanFile(String path, String mimeType,
IMediaScannerListener listener)
{
Bundle args = new Bundle();
args.putString("filepath", path);
args.putString("mimetype", mimeType);
if (listener != null) {
args.putIBinder("listener", listener.asBinder());
}
startService(new Intent(MediaScannerService.this,
MediaScannerService.class).putExtras(args));
}
public void scanFile(String path, String mimeType) {
requestScanFile(path, mimeType, null);
}
};
说到底还是在调用startService()。
具体处理消息泵线程里的消息时,执行的是ServiceHandler的handleMessage()函数:
private final class ServiceHandler extends Handler
{
@Override
public void handleMessage(Message msg)
{
Bundle arguments = (Bundle) msg.obj;
String filePath = arguments.getString("filepath");
. . . . . .
if (filePath != null) {
. . . . . .
uri = scanFile(filePath, arguments.getString("mimetype"));
. . . . . .
} else {
. . . . . .
scan(directories, volume);
. . . . . .
}
. . . . . .
stopSelf(msg.arg1);
}
};
此时调用的scanFile()或scan()函数才是实际进行扫描动作的地方。扫描动作中主要借助的是辅助类MediaScanner,这个类非常重要,它是打通Java层和C++层的关键,扫描动作最终会调用到MediaScanner的某个native函数,于是程序流程开始走到C++层。
现在,我们可以画一张示意图:
2 运作细节
2.1 发起扫描动作
现在我们已经了解了,要发起扫描动作,大体上只有两种方式:
1)用广播来发起扫描动作;
2)绑定服务来发起扫描动作;
下面我们细说一下这两种方式。
2.1.1 用广播来发起扫描动作
扫描服务的配套receiver是MediaScannerReceiver,它在AndroidManifest.xml里的描述如下:
<receiver android:name="MediaScannerReceiver">
<intent-filter>
<action android:name="android.intent.action.BOOT_COMPLETED" />
</intent-filter>
<intent-filter>
<action android:name="android.intent.action.MEDIA_MOUNTED" />
<data android:scheme="file" />
</intent-filter>
<intent-filter>
<action android:name="android.intent.action.MEDIA_UNMOUNTED" />
<data android:scheme="file" />
</intent-filter>
<intent-filter>
<action android:name="android.intent.action.MEDIA_SCANNER_SCAN_FILE" />
<data android:scheme="file" />
</intent-filter>
</receiver>
MediaScannerReceiver的onReceive()代码如下:
public void onReceive(Context context, Intent intent) {
final String action = intent.getAction();
final Uri uri = intent.getData();
if (Intent.ACTION_BOOT_COMPLETED.equals(action)) {
// Scan both internal and external storage
scan(context, MediaProvider.INTERNAL_VOLUME); // INTERNAL_VOLUME = "internal"
scan(context, MediaProvider.EXTERNAL_VOLUME); // EXTERNAL_VOLUME = "external"
} else {
if (uri.getScheme().equals("file")) {
// handle intents related to external storage
. . . . . .
Log.d(TAG, "action: " + action + " path: " + path);
if (Intent.ACTION_MEDIA_MOUNTED.equals(action)) {
// scan whenever any volume is mounted
scan(context, MediaProvider.EXTERNAL_VOLUME);
} else if (Intent.ACTION_MEDIA_SCANNER_SCAN_FILE.equals(action) &&
path != null && path.startsWith(externalStoragePath + "/")) {
scanFile(context, path);
}
}
}
}
- 当系统刚刚启动时,收到ACTION_BOOT_COMPLETED广播,此时会把内部卷标(“internal”)和外部卷标(“external”)都扫描一下;
- 如果收到ACTION_MEDIA_MOUNTED广播,则只扫描外部卷标;
- 如果收到的是ACTION_MEDIA_SCANNER_SCAN_FILE广播,则扫描具体的文件路径。
当用户插入了扩展介质(一般指SD卡),并且该介质已经被系统正确识别、安装,系统就会发出ACTION_MEDIA_MOUNTED广播。从Android 4.4开始,ACTION_MEDIA_MOUNTED广播只能由系统(系统服务MountService)发出,普通用户是无权发送的。
另外,我们可以通过发送ACTION_MEDIA_SCANNER_SCAN_FILE广播,要求MediaScannerService扫描一下具体的文件。比如说在ExternalStorageProvider的openDocument()函数里,就会设置监听器监听用户是不是在读写模式下close了某个文件,因为close一般表示写入动作已经完成了,那么此时就需要“踢一下”MediaScannerService,让它更新一下自己的数据。这段代码截选如下:
【frameworks/base/packages/externalstorageprovider/src/com/android/externalstorage/ExternalStorageProvider.java】
@Override
public ParcelFileDescriptor openDocument(String documentId, String mode,
CancellationSignal signal)
throws FileNotFoundException
{
. . . . . .
// When finished writing, kick off media scanner
return ParcelFileDescriptor.open(file, pfdMode, mHandler,
new OnCloseListener() {
@Override
public void onClose(IOException e) {
final Intent intent = new Intent(Intent.ACTION_MEDIA_SCANNER_SCAN_FILE);
intent.setData(Uri.fromFile(file));
getContext().sendBroadcast(intent); // 用广播来发起扫描动作
}
});
. . . . . .
}
2.1.2 用MediaScannerConnection来发起扫描动作
除了利用类似ACTION_MEDIA_SCANNER_SCAN_FILE这样的广播,系统中还有一种办法可以发起扫描动作,那就是先利用bindService机制得到的IMediaScannerService代理接口,而后再通过调用该接口的requestScanFile()或scanFile(),同样可以向MediaScannerService发出扫描语义。
不过,我们一般并不直白地去bindService,而是通过一种封装好的辅助类:MediaScannerConnection。该类的定义截选如下:
【frameworks/base/media/java/android/media/MediaScannerConnection.java】
public class MediaScannerConnection implements ServiceConnection {
private static final String TAG = "MediaScannerConnection";
private Context mContext;
private MediaScannerConnectionClient mClient;
private IMediaScannerService mService;
private boolean mConnected; // true if connect() has been called since last disconnect()
private final IMediaScannerListener.Stub mListener = new IMediaScannerListener.Stub()
. . . . . .
请注意那个mService成员,它就是为了绑定service而设计的。
MediaScannerConnection里设计了两个scanFile()函数,一个动态的,一个静态的。大家不要搞混了。
2.1.2.1 动态形式scanFile()
动态形式scanFile()的代码截选:
public void scanFile(String path, String mimeType) {
. . . . . .
mService.requestScanFile(path, mimeType, mListener);
. . . . . .
}
对于动态形式的scanFile()而言,它只能在MediaScannerConnection成功绑定到MediaScannerService之后调用,此时它简单地调用mService.requestScanFile()将语义传递给MediaScannerService,再由MediaScannerService通过startService()向自己的消息泵线程打入消息。
mService.requestScanFile()的最后一个参数mListener的定义如下:
private final IMediaScannerListener.Stub mListener = new IMediaScannerListener.Stub() {
public void scanCompleted(String path, Uri uri) {
MediaScannerConnectionClient client = mClient;
if (client != null) {
client.onScanCompleted(path, uri);
}
}
};
它是个简单的binder实体。每当MediaScannerService扫描完所指定的一个文件后,就会回调到该实体的scanCompleted()。此时一般会经由client.onScanCompleted()一句间接调用下一次scanFile()的动作,从而使扫描多个文件的动作连贯起来。
2.1.2.2 静态形式scanFile()
静态形式scanFile()的代码截选:
public static void scanFile(Context context, String[] paths, String[] mimeTypes,
OnScanCompletedListener callback) {
ClientProxy client = new ClientProxy(paths, mimeTypes, callback);
MediaScannerConnection connection = new MediaScannerConnection(context, client);
client.mConnection = connection;
connection.connect(); // 内部主要是bindService动作
}
对于静态形式的scanFile()而言,会重新创建一个MediaScannerConnection对象,并通过connect()动作和MediaScannerService联系起来。
请大家注意创建MediaScannerConnection时传入的第二个参数client,它必须实现MediaScannerConnectionClient接口。说穿了是为了监听两种事情:
1)和MediaScannerService之间的连接是否建立好了;
2)MediaScannerService中扫描某文件的动作是否执行完了;
MediaScannerConnectionClient接口的定义如下:
【frameworks/base/media/java/android/media/MediaScannerConnection.java】
public interface MediaScannerConnectionClient extends OnScanCompletedListener {
public void onMediaScannerConnected();
public void onScanCompleted(String path, Uri uri);
}
在静态形式的scanFile()中,实现MediaScannerConnectionClient接口的类是ClientProxy,它是这样实现onMediaScannerConnected()和onScanCompleted()的:
【frameworks/base/media/java/android/media/MediaScannerConnection.java】
public void onMediaScannerConnected() {
scanNextPath();
}
public void onScanCompleted(String path, Uri uri) {
if (mClient != null) {
mClient.onScanCompleted(path, uri);
}
scanNextPath();
}
可以看到一旦连接建立成功或者某个文件扫描完毕,就会调用scanNextPath(),进一步扫描接下来的内容,直到把调用静态scanFile()时传入的paths数组遍历完毕。
void scanNextPath() {
if (mNextPath >= mPaths.length) {
mConnection.disconnect();
return;
}
String mimeType = mMimeTypes != null ? mMimeTypes[mNextPath] : null;
mConnection.scanFile(mPaths[mNextPath], mimeType);
mNextPath++;
}
实际上,MediaScannerConnection的connect()动作就是在bindService(),它的代码如下:
【frameworks/base/media/java/android/media/MediaScannerConnection.java】
public void connect() {
synchronized (this) {
if (!mConnected) {
Intent intent = new Intent(IMediaScannerService.class.getName());
intent.setComponent( new ComponentName("com.android.providers.media",
"com.android.providers.media.MediaScannerService"));
mContext.bindService(intent, this, Context.BIND_AUTO_CREATE);
mConnected = true;
}
}
}
因为bindService()动作本身是异步的,初始时mService的值还是null,所以我们不能直接在这里执行类似mService.requestScanFile()这样的操作。我们必须等到bind动作成功完成,系统回调到MediaScannerConnection的onServiceConnected(),才会给mService赋值:
public void onServiceConnected(ComponentName className, IBinder service) {
. . . . . .
synchronized (this) {
mService = IMediaScannerService.Stub.asInterface(service);
if (mService != null && mClient != null) {
mClient.onMediaScannerConnected();
}
}
}
如果bind动作是成功的,而且用户在构造MediaScannerConnection对象时传入了client参数。那么此时就会回调mClient的onMediaScannerConnected()函数。
请注意,静态的scanFile()方法最终并没有直接执行requestScanFile(),它先建立了和MediaScannerService的绑定关系,然后在onServiceConnected()中感知到绑定已经成功之后,才会经由ClientProxy间接转过头调用到自己的scanFile()函数,从而执行到requestScanFile()。
ClientProxy、MediaScannerConnection、MediaScannerService三者之间的关系如下图所示:
以MediaScannerConnection对象为桥梁:
1)其mService“指向”MediaScannerService的mBinder;
2)其mClient指向ClientProxy对象;
当然,在看懂上图后,我们也可以不使用默认的ClientProxy,而添加我们自定义的client对象,只要这个client对象实现了MediaScannerConnectionClient接口即可。比如在MediaProvider中,就定义了另一个类ScannerClient类,代码截选如下:
【packages/providers/mediaprovider/src/com/android/providers/media/MediaProvider.java】
private static final class ScannerClient implements MediaScannerConnectionClient {
String mPath = null;
MediaScannerConnection mScannerConnection;
SQLiteDatabase mDb;
public ScannerClient(Context context, SQLiteDatabase db, String path) {
mDb = db;
mPath = path;
mScannerConnection = new MediaScannerConnection(context, this);
mScannerConnection.connect();
}
@Override
public void onMediaScannerConnected() {
. . . . . .
}
@Override
public void onScanCompleted(String path, Uri uri) {
}
}
这么看来,MediaScannerConnection还真是起连接作用的“connection”,它将发起扫描请求的client和最终执行扫描动作的MediaScannerService连接起来了。我们把上面那张图简化一下,可以看到如下示意图:
以上介绍的就是发起scan动作的方法,接下来我们来看看到底有哪些地方在使用这些方法。
2.2 谁会发起扫描动作
2.2.1 发起者列表
发出ACTION_MEDIA_SCANNER_SCAN_FILE广播的地方:
发起方
相关代码位置
说明
ExternalStorageProvider
openDocument()注册OnCloseListener的地方
ComposeMessageActivity
MMS里copyPart()函数中
saveRingtone()、
copyMedia()中都会调用copyPart()。
DownloadProvider
openFile()注册OnCloseListener的地方
EmlAttachmentProvider
copyAttachment(),将附件拷到外部下载目录(一般是SD卡)时
provider在update()中处理ATTACHMENT的地方
SoundRecorder
addToMediaDB()
录制sample后,要添加进多媒体数据库
利用MediaScannerConnection的地方:
发起方
相关代码位置
说明
AttachmentUtilities
saveAttachment()
代码截选见下文
BeamTransferManager
processFiles()
NFC方面,
finishTransfer()、handleMessage()处理MSG_NEXT_TRANSFER_TIMER时,都会调用processFiles()。
BluetoothOppService
MediaScannerNotifier
没有直接使用MediaScannerConnection.scanFile(),而是编写了自己的MediaScannerNotifier
CalendarDebugActivity
doInBackground()
DumpDbTask的doInBackground(),将数据库文件存成calendar.db.zip之后,调用MediaScannerConnection.scanFile()
DownloadScanner
DownloadScanner
没有直接使用MediaScannerConnection.scanFile(),而是编写了自己的DownloadScanner
FmRecorder
addRecordingToDatabase()
MediaScannerConnection.scanFile(context,
new String[] { mRecordFile.getPath() },
null, null);
IngestService
ScannerClient
没有直接使用MediaScannerConnection.scanFile(),而是编写了自己的ScannerClient
MediaProvider
ScannerClient
没有直接使用MediaScannerConnection.scanFile(),而是编写了自己的ScannerClient
VCardService
CustomeMediaScannerConnectionClient
没有直接使用MediaScannerConnection.scanFile(),而是编写了自己的CustomeMediaScannerConnectionClient
2.2.2 saveAttachment()中的示例代码
我们举一个实际的例子。在Email模块中,如果附件存入了外部存储器,那么就有必要扫描一次媒体文件了,这样才能够立即将相关文件体现到Gallery、Music中。所以在saveAttachment()函数里,就会调用MediaScannerConnection.scanFile():
【packages/apps/email/emailcommon/src/com/android/emailcommon/utility/AttachmentUtilities.java】
public static void saveAttachment(Context context, InputStream in, Attachment attachment) {
. . . . . .
ContentResolver resolver = context.getContentResolver();
if (attachment.mUiDestination == UIProvider.AttachmentDestination.CACHE) {
. . . . . .
} else if (Utility.isExternalStorageMounted()) {
. . . . . .
File file = Utility.createUniqueFile(downloads, attachment.mFileName);
size = copyFile(in, new FileOutputStream(file));
String absolutePath = file.getAbsolutePath();
// 尽管下载管理器会扫描媒体文件,但只会在用户运行download APP并点击相关按钮后,
// 才会进行扫描。所以,我们自己运行一下media scanner,以便把附件立即添加进gallery / music。
MediaScannerConnection.scanFile(context, new String[] {absolutePath},
null, null);
. . . . . .
DownloadManager dm = (DownloadManager)
context.getSystemService(Context.DOWNLOAD_SERVICE);
long id = dm.addCompletedDownload(attachment.mFileName,
attachment.mFileName,
false /* do not use media scanner */,
mimeType, absolutePath, size,
true /* show notification */);
contentUri = dm.getUriForDownloadedFile(id).toString();
. . . . . .
} else {
. . . . . .
throw new IOException();
}
. . . . . .
context.getContentResolver().update(uri, cv, null, null);
}
2.3 说说实际的扫描动作
前文介绍MediaScannerService的消息泵线程时已经说过,最终ServiceHandler的handleMessage()会调用scanFile()或scan()来完成扫描。现在我们来看看scanFile()、scan()的细节。
2.3.1 scanFile()动作
MediaScannerService的scanFile()定义如下:
【packages/providers/mediaprovider/src/com/android/providers/media/MediaScannerService.java】
private Uri scanFile(String path, String mimeType) {
String volumeName = MediaProvider.EXTERNAL_VOLUME;
openDatabase(volumeName);
MediaScanner scanner = createMediaScanner();
try {
String canonicalPath = new File(path).getCanonicalPath();
return scanner.scanSingleFile(canonicalPath, volumeName, mimeType);
} catch (Exception e) {
Log.e(TAG, "bad path " + path + " in scanFile()", e);
return null;
}
}
可以看到,scanFile()函数内部借助了辅助类MediaScanner,调用了该类的scanSingleFile()。这个MediaScanner才是重头戏,它的scanSingleFile()代码截选如下:
【frameworks/base/media/java/android/media/MediaScanner.java】
public Uri scanSingleFile(String path, String volumeName, String mimeType) {
. . . . . .
initialize(volumeName);
prescan(path, true);
File file = new File(path);
. . . . . .
// always scan the file, so we can return the content://media Uri for existing files
return mClient.doScanFile(path, mimeType, lastModifiedSeconds, file.length(),
false, true, MediaScanner.isNoMediaPath(path));
. . . . . .
}
借助了mClient.doScanFile()。
此处的mClient类型为MyMediaScannerClient,mClient的定义是:
private final MyMediaScannerClient mClient = new MyMediaScannerClient();
MyMediaScannerClient类的doScanFile()的代码截选如下:
【frameworks/base/media/java/android/media/MediaScanner.java】
public Uri doScanFile(String path, String mimeType, long lastModified,
long fileSize, boolean isDirectory, boolean scanAlways, boolean noMedia) {
. . . . . .
FileEntry entry = beginFile(path, mimeType, lastModified,
fileSize, isDirectory, noMedia);
. . . . . .
if (entry != null && (entry.mLastModifiedChanged || scanAlways)) {
if (noMedia) {
result = endFile(entry, false, false, false, false, false);
} else {
. . . . . .
. . . . . .
// we only extract metadata for audio and video files
if (isaudio || isvideo) {
processFile(path, mimeType, this);
}
if (isimage) {
processImageFile(path);
}
result = endFile(entry, ringtones, notifications, alarms, music, podcasts);
}
}
. . . . . .
return result;
}
因为MyMediaScannerClient是MediaScanner的内嵌类,所以它可以直接调用MediaScanner的processFile()。
现在我们画一张scanFile()的调用关系图:
2.3.2 scan()动作
与scanFile()动作类似,MediaScannerService中扫描目录的动作是scan():
【packages/providers/mediaprovider/src/com/android/providers/media/MediaScannerService.java】
private void scan(String[] directories, String volumeName) {
. . . . . .
values.put(MediaStore.MEDIA_SCANNER_VOLUME, volumeName);
Uri scanUri = getContentResolver().insert(MediaStore.getMediaScannerUri(), values);
. . . . . .
MediaScanner scanner = createMediaScanner();
scanner.scanDirectories(directories, volumeName);
. . . . . .
sendBroadcast(new Intent(Intent.ACTION_MEDIA_SCANNER_FINISHED, uri));
. . . . . .
}
同样是借助了辅助类MediaScanner,调用了该类的scanDirectories()。
scanDirectories()的代码截选如下:
【frameworks/base/media/java/android/media/MediaScanner.java】
public void scanDirectories(String[] directories, String volumeName) {
. . . . . .
for (int i = 0; i < directories.length; i++) {
processDirectory(directories[i], mClient);
}
. . . . . .
}
我们画一张scan()的调用关系图:
2.3.3 MediaScanner
顾名思义,MediaScanner就是个“媒体文件扫描器”。它必须打通java层次和C++层次。请大家注意它的两个native函数:native_init()和native_setup(),以及两个重要成员变量:一个是上文刚刚提到的mClient成员,另一个是mNativeContext。
MediaScanner的相关代码截选如下:
【frameworks/base/media/java/android/media/MediaScanner.java】
public class MediaScanner
{
static {
System.loadLibrary("media_jni");
native_init(); // 将java层和c++层联系起来
}
. . . . . .
private long mNativeContext;
. . . . . .
public MediaScanner(Context c) {
native_setup();
. . . . . .
}
. . . . . .
// 一开始就具有明确的mClient对象
private final MyMediaScannerClient mClient = new MyMediaScannerClient();
. . . . . .
}
MediaScanner类加载之时,就会同时加载动态链接库“media_jni”,并调用native_init()将java层和c++层联系起来。而且MediaScanner对象一开始就具有明确的mClient对象,类型为MyMediaScannerClient。
经过分析代码,我们发现在C++层会有个与MediaScanner相对应的类,叫作StagefrightMediaScanner。当java层创建MediaScanner对象时,MediaScanner的构造函数就调用了native_setup(),该函数对应到C++层就是android_media_MediaScanner_native_setup(),其代码如下:
【frameworks/base/media/jni/android_media_MediaScanner.cpp】
static void
android_media_MediaScanner_native_setup(JNIEnv *env, jobject thiz)
{
ALOGV("native_setup");
MediaScanner *mp = new StagefrightMediaScanner;
if (mp == NULL) {
jniThrowException(env, kRunTimeException, "Out of memory");
return;
}
env->SetLongField(thiz, fields.context, (jlong)mp);
}
最后一句env->SetLongField()其实就是在为java层MediaScanner的mNativeContext域赋值。
后续我们会看到,每当C++层执行扫描动作时,还会再创建一个MyMediaScannerClient对象,这个对象和Java层的同名类对应。我们画一张图来说明:
2.3.4 调用到C++层次
不管是扫描文件,还是扫描目录,总之MediaScannerService已经把工作委托给MediaScanner的scanSingleFile()和scanDirectories()了,而这两个函数到头来都是调用MediaScanner自己的native函数,即processFile()和processDirectory()。其声明如下:
【frameworks/base/media/java/android/media/MediaScanner.java】
private native void processDirectory(String path, MediaScannerClient client);
private native void processFile(String path, String mimeType, MediaScannerClient client);
MediaScanner中调用的processFile()对应于C++层的android_media_MediaScanner_processFile()。代码截选如下:
【frameworks/base/media/jni/android_media_MediaScanner.cpp】
static void android_media_MediaScanner_processFile(
JNIEnv *env, jobject thiz, jstring path,
jstring mimeType, jobject client)
{
. . . . . .
MediaScanner *mp = getNativeScanner_l(env, thiz);
. . . . . .
const char *mimeTypeStr =
(mimeType ? env->GetStringUTFChars(mimeType, NULL) : NULL);
if (mimeType && mimeTypeStr == NULL) { // Out of memory
// ReleaseStringUTFChars can be called with an exception pending.
env->ReleaseStringUTFChars(path, pathStr);
return;
}
MyMediaScannerClient myClient(env, client); // 构造一个临时的myClient
MediaScanResult result = mp->processFile(pathStr, mimeTypeStr, myClient);
if (result == MEDIA_SCAN_RESULT_ERROR) {
ALOGE("An error occurred while scanning file '%s'.", pathStr);
}
. . . . . .
}
注意这里构造了一个局部的(C++层次)MyMediaScannerClient对象,构造myClient时传入的client参数来自于Java层调用processFile()时传入的那个(Java层次)MyMediaScannerClient对象。这个对象会记录在C++层MyMediaScannerClient的mClient域中,这个在前面的示意图中已有表示。
相应的,processDirectory()对应于C++层的android_media_MediaScanner_processDirectory()。代码截选如下:
【frameworks/base/media/jni/android_media_MediaScanner.cpp】
static void android_media_MediaScanner_processDirectory(
JNIEnv *env, jobject thiz, jstring path, jobject client)
{
. . . . . .
MediaScanner *mp = getNativeScanner_l(env, thiz);
. . . . . .
MyMediaScannerClient myClient(env, client);
MediaScanResult result = mp->processDirectory(pathStr, myClient);
. . . . . .
}
2.3.4.1 processFile()
android_media_MediaScanner_processFile()函数中的那个mp是经由下面这句得到的:
MediaScanner *mp = getNativeScanner_l(env, thiz);
它指向的其实就是StagefrightMediaScanner,所以这里调用的processFile就是:
【frameworks/av/media/libstagefright/StagefrightMediaScanner.cpp】
MediaScanResult StagefrightMediaScanner::processFile(
const char *path, const char *mimeType,
MediaScannerClient &client) {
ALOGV("processFile '%s'.", path);
client.setLocale(locale());
client.beginFile();
MediaScanResult result = processFileInternal(path, mimeType, client);
client.endFile();
return result;
}
主要行为在processFileInternal()里:
【frameworks/av/media/libstagefright/StagefrightMediaScanner.cpp】
MediaScanResult StagefrightMediaScanner::processFileInternal(
const char *path, const char * /* mimeType */,
MediaScannerClient &client) {
const char *extension = strrchr(path, '.');
. . . . . .
if (!FileHasAcceptableExtension(extension)) {
return MEDIA_SCAN_RESULT_SKIPPED;
}
if (!strcasecmp(extension, ".mid")
|| !strcasecmp(extension, ".smf")
|| !strcasecmp(extension, ".imy")
. . . . . .
return HandleMIDI(path, &client);
}
sp<MediaMetadataRetriever> mRetriever(new MediaMetadataRetriever);
int fd = open(path, O_RDONLY | O_LARGEFILE);
. . . . . .
status = mRetriever->setDataSource(fd, 0, 0x7ffffffffffffffL);
close(fd);
. . . . . .
const char *value;
if ((value = mRetriever->extractMetadata(
METADATA_KEY_MIMETYPE)) != NULL) {
status = client.setMimeType(value);
. . . . . .
}
struct KeyMap {
const char *tag;
int key;
};
static const KeyMap kKeyMap[] = {
{ "tracknumber", METADATA_KEY_CD_TRACK_NUMBER },
{ "discnumber", METADATA_KEY_DISC_NUMBER },
{ "album", METADATA_KEY_ALBUM },
{ "artist", METADATA_KEY_ARTIST },
. . . . . .
};
static const size_t kNumEntries = sizeof(kKeyMap) / sizeof(kKeyMap[0]);
for (size_t i = 0; i < kNumEntries; ++i) {
const char *value;
if ((value = mRetriever->extractMetadata(kKeyMap[i].key)) != NULL) {
status = client.addStringTag(kKeyMap[i].tag, value);
. . . . . .
}
}
return MEDIA_SCAN_RESULT_OK;
}
可以看到,processFileInternal()里扫描具体文件的大体流程,无非是先获取多媒体文件的元数据,然后再通过MyMediaScannerClient将元数据信息从C++层传递到Java层。
processFileInternal()里的主要细节有:
1)调用FileHasAcceptableExtension()函数,看看文件的扩展名是不是属于多媒体文件扩展名,合适的扩展名有:
【frameworks/av/media/libstagefright/StagefrightMediaScanner.cpp】
static bool FileHasAcceptableExtension(const char *extension) {
static const char *kValidExtensions[] = {
".mp3", ".mp4", ".m4a", ".3gp", ".3gpp", ".3g2", ".3gpp2",
".mpeg", ".ogg", ".mid", ".smf", ".imy", ".wma", ".aac",
".wav", ".amr", ".midi", ".xmf", ".rtttl", ".rtx", ".ota",
".mkv", ".mka", ".webm", ".ts", ".fl", ".flac", ".mxmf",
".avi", ".mpeg", ".mpg", ".awb", ".mpga"
};
. . . . . .
}
如果扩展名不合适,则直接return MEDIA_SCAN_RESULT_SKIPPED。
2)看看文件是不是midi文件,如果是midi文件,则以HandleMIDI()来处理。
【frameworks/av/media/libstagefright/StagefrightMediaScanner.cpp】
if (!strcasecmp(extension, ".mid")
|| !strcasecmp(extension, ".smf")
|| !strcasecmp(extension, ".imy")
|| !strcasecmp(extension, ".midi")
|| !strcasecmp(extension, ".xmf")
|| !strcasecmp(extension, ".rtttl")
|| !strcasecmp(extension, ".rtx")
|| !strcasecmp(extension, ".ota")
|| !strcasecmp(extension, ".mxmf")) {
return HandleMIDI(path, &client);
}
从HandleMIDI()的代码看,要解析并提取midi文件的元数据,需要用到一种EAS引擎,利用EAS_ParseMetaData()解析出时长信息。并调用MyMediaScannerClient的addStringTag()。
3)如果是其他支持的多媒体文件,则利用工具类MediaMetadataRetriever来获取文件的元数据,并将得到的元数据传递给MyMediaScannerClient。
其实MediaMetadataRetriever内部是利用系统服务“media.player”来解析多媒体文件的,这个系统服务对应的代理接口是IMediaPlayerService,它有个成员函数createMetadataRetriever()可以用于获取IMediaMetadataRetriever接口,而后就可以调用该接口的setDataSource()和extractMetadata()了。
processFileInternal()里主要通过两个函数,向Java层的MyMediaScannerClient传递数据,一个是setMimeType(),另一个是addStringTag()。以C++层的setMimeType()为例,其代码如下:
【frameworks/base/media/jni/android_media_MediaScanner.cpp】
virtual status_t setMimeType(const char* mimeType)
{
ALOGV("setMimeType: %s", mimeType);
jstring mimeTypeStr;
if ((mimeTypeStr = mEnv->NewStringUTF(mimeType)) == NULL) {
mEnv->ExceptionClear();
return NO_MEMORY;
}
mEnv->CallVoidMethod(mClient, mSetMimeTypeMethodID, mimeTypeStr);
mEnv->DeleteLocalRef(mimeTypeStr);
return checkAndClearExceptionFromCallback(mEnv, "setMimeType");
}
基本上只是通过JNI技术,调用到Java层的setMimeType()而已。
现在我们画一张关于扫描文件的简单示意图,来整理一下思路。大家顺着箭头看图就可以了。
2.3.4.2 processDirectory()
按理说,和processFile()类似,processDirectory()最终对应的代码也应该在StagefrightMediaScanner里,但是StagefrightMediaScanner并没有编写这个函数,又因为StagefrightMediaScanner继承于MediaScanner(C++层次),所以实际上使用的是MediaScanner的ProcessDirectory()
【frameworks/av/media/libmedia/MediaScanner.cpp】
MediaScanResult MediaScanner::processDirectory(
const char *path, MediaScannerClient &client) {
int pathLength = strlen(path);
. . . . . .
char* pathBuffer = (char *)malloc(PATH_MAX + 1);
. . . . . .
strcpy(pathBuffer, path);
. . . . . .
client.setLocale(locale());
MediaScanResult result = doProcessDirectory(pathBuffer, pathRemaining, client, false);
free(pathBuffer);
return result;
}
【frameworks/av/media/libmedia/MediaScanner.cpp】
MediaScanResult MediaScanner::doProcessDirectory(char *path, int pathRemaining,
MediaScannerClient &client, bool noMedia) {
char* fileSpot = path + strlen(path);
struct dirent* entry;
if (shouldSkipDirectory(path)) {
. . . . . .
return MEDIA_SCAN_RESULT_OK;
}
// Treat all files as non-media in directories that contain a ".nomedia" file
if (pathRemaining >= 8 /* strlen(".nomedia") */ ) {
strcpy(fileSpot, ".nomedia");
if (access(path, F_OK) == 0) {
ALOGV("found .nomedia, setting noMedia flag");
noMedia = true;
}
. . . . . .
}
DIR* dir = opendir(path);
. . . . . .
MediaScanResult result = MEDIA_SCAN_RESULT_OK;
while ((entry = readdir(dir))) {
if (doProcessDirectoryEntry(path, pathRemaining, client, noMedia, entry, fileSpot)
== MEDIA_SCAN_RESULT_ERROR) {
result = MEDIA_SCAN_RESULT_ERROR;
break;
}
}
closedir(dir);
return result;
}
doProcessDirectory()先判断需要扫描的目录是不是应该“跳过”的目录,如果是的话,则直接return MEDIA_SCAN_RESULT_OK。判断函数shouldSkipDirectory()的代码如下:
【frameworks/av/media/libmedia/MediaScanner.cpp】
bool MediaScanner::shouldSkipDirectory(char *path) {
if (path && mSkipList && mSkipIndex) {
int len = strlen(path);
int idx = 0;
int startPos = 0;
while (mSkipIndex[idx] != -1) {
if ((len == mSkipIndex[idx])
&& (strncmp(path, &mSkipList[startPos], len) == 0)) {
return true;
}
startPos += mSkipIndex[idx] + 1; // extra char for the delimiter
idx++;
}
}
return false;
}
其实就是比对一下“需要扫描的目录”是否存在于mSkipList列表中。这个列表的内容其实来自于“testing.mediascanner.skiplist”属性,该属性可以记录若干目录名,目录名之间以逗号分隔。在C++层的MediaScanner构造函数中,会调用loadSkipList()来读取这个属性,解析属性中记录的所有目录名并写入mSkipList列表。
接着doProcessDirectory()用一个while循环多次调用doProcessDirectoryEntry(),其内部在必要时候,会再次调用doProcessDirectory()分析子目录。while语句的循环判断部分用到了readdir()函数,readdir()是linux上返回所指目录中“下一个进入点”(next entry)的函数,我们常常在一个while循环中调用它,以便遍历出目录中的所有内容。
doProcessDirectoryEntry()函数的定义截选如下:
【frameworks/av/media/libmedia/MediaScanner.cpp】
MediaScanResult MediaScanner::doProcessDirectoryEntry(
char *path, int pathRemaining, MediaScannerClient &client, bool noMedia,
struct dirent* entry, char* fileSpot) {
struct stat statbuf;
const char* name = entry->d_name;
. . . . . .
int type = entry->d_type;
. . . . . .
if (type == DT_DIR) { // 普通目录
. . . . . .
if (stat(path, &statbuf) == 0) {
status_t status = client.scanFile(path, statbuf.st_mtime, 0,
true /*isDirectory*/, childNoMedia);
. . . . . .
}
// and now process its contents
strcat(fileSpot, "/");
MediaScanResult result = doProcessDirectory(path, pathRemaining - nameLength - 1,
client, childNoMedia);
. . . . . .
} else if (type == DT_REG) { // 普通文件
stat(path, &statbuf);
status_t status = client.scanFile(path, statbuf.st_mtime, statbuf.st_size,
false /*isDirectory*/, noMedia);
. . . . . .
}
return MEDIA_SCAN_RESULT_OK;
}
不管当前处理的入口类型是“目录”还是“文件”,最终都是依靠client的scanFile()来处理,只不过前者倒数第二个参数(isDirectory)为true,后者为false而已。
client.scanFile()最终也是要调回到Java层的,MyMediaScannerClient的scanFile()代码截选如下:
【frameworks/base/media/jni/android_media_MediaScanner.cpp】
virtual status_t scanFile(const char* path, long long lastModified,
long long fileSize, bool isDirectory, bool noMedia)
{
. . . . . .
jstring pathStr;
if ((pathStr = mEnv->NewStringUTF(path)) == NULL) {
mEnv->ExceptionClear();
return NO_MEMORY;
}
mEnv->CallVoidMethod(mClient, mScanFileMethodID, pathStr, lastModified,
fileSize, isDirectory, noMedia);
mEnv->DeleteLocalRef(pathStr);
return checkAndClearExceptionFromCallback(mEnv, "scanFile");
}
【frameworks/base/media/java/android/media/MediaScanner.java】
@Override
public void scanFile(String path, long lastModified, long fileSize,
boolean isDirectory, boolean noMedia) {
doScanFile(path, null, lastModified, fileSize, isDirectory, false, noMedia);
}
调用到doScanFile()函数。
现在我们再画一张关于扫描目录的简单示意图:
2.3.4.3 doScanFile()和MediaProvider
站在Java层次来看,不管是扫描具体的文件,还是扫描一个目录,最终都会走到Java层MyMediaScannerClient的doScanFile()。在前文我们已经列出过这个函数的代码,为了说明问题,这里再列一下其中的重要句子:
【frameworks/base/media/java/android/media/MediaScanner.java】
public Uri doScanFile(String path, String mimeType, long lastModified,
long fileSize, boolean isDirectory, boolean scanAlways, boolean noMedia) {
. . . . . .
FileEntry entry = beginFile(path, mimeType, lastModified,
fileSize, isDirectory, noMedia);
. . . . . .
if (isaudio || isvideo) {
processFile(path, mimeType, this);
}
if (isimage) {
processImageFile(path);
}
result = endFile(entry, ringtones, notifications, alarms, music, podcasts);
. . . . . .
return result;
}
本小节着重看一下其中和MediaProvider相关的beginFile()和endFile()。
beginFile()是为了后续和MediaProvider打交道,准备一个FileEntry。FileEntry的定义如下:
【frameworks/base/media/java/android/media/MediaScanner.java】
private static class FileEntry {
long mRowId;
String mPath;
long mLastModified;
int mFormat;
boolean mLastModifiedChanged;
FileEntry(long rowId, String path, long lastModified, int format) {
mRowId = rowId;
mPath = path;
mLastModified = lastModified;
mFormat = format;
mLastModifiedChanged = false;
}
. . . . . .
}
FileEntry的几个成员变量,其实体现了查表时的若干列的值。
beginFile()的代码截选如下:
【frameworks/base/media/java/android/media/MediaScanner.java】
public FileEntry beginFile(String path, String mimeType, long lastModified,
long fileSize, boolean isDirectory, boolean noMedia) {
. . . . . .
FileEntry entry = makeEntryFor(path); // 从MediaProvider中查出该文件或目录对应的入口
. . . . . .
if (entry == null || wasModified) {
if (wasModified) {
entry.mLastModified = lastModified;
} else {
// 如果前面没查到FileEntry,就在这里new一个新的FileEntry
entry = new FileEntry(0, path, lastModified,
(isDirectory ? MtpConstants.FORMAT_ASSOCIATION : 0));
}
entry.mLastModifiedChanged = true;
}
. . . . . .
return entry;
}
其中调用的makeEntryFor()内部就会查询MediaProvider:
FileEntry makeEntryFor(String path) {
String where;
String[] selectionArgs;
Cursor c = null;
try {
where = Files.FileColumns.DATA + "=?";
selectionArgs = new String[] { path };
c = mMediaProvider.query(mPackageName, mFilesUriNoNotify,
FILES_PRESCAN_PROJECTION,
where, selectionArgs, null, null);
if (c.moveToFirst()) {
long rowId = c.getLong(FILES_PRESCAN_ID_COLUMN_INDEX);
int format = c.getInt(FILES_PRESCAN_FORMAT_COLUMN_INDEX);
long lastModified = c.getLong(FILES_PRESCAN_DATE_MODIFIED_COLUMN_INDEX);
return new FileEntry(rowId, path, lastModified, format);
}
} catch (RemoteException e) {
} finally {
if (c != null) {
c.close();
}
}
return null;
}
查询语句中用的FILES_PRESCAN_PROJECTION的定义如下:
private static final String[] FILES_PRESCAN_PROJECTION = new String[] {
Files.FileColumns._ID, // 0
Files.FileColumns.DATA, // 1
Files.FileColumns.FORMAT, // 2
Files.FileColumns.DATE_MODIFIED, // 3
};
看到了吗,特意要去查一下MediaProvider中记录的待查文件的最后修改日期。能查到就返回一个FileEntry,如果查询时出现异常就返回null。beginFile()的lastModified参数可以理解为是从文件系统里拿到的待查文件的最后修改日期,它应该是最准确的。而MediaProvider里记录的信息则有可能“较老”。beginFile()内部通过比对这两个“最后修改日期”,就可以知道该文件是不是真的改动了。如果的确改动了,就要把FileEntry里的mLastModified调整成最新数据。
基本上而言,beginFile()会返回一个FileEntry。如果该阶段没能在MediaProvider里找到文件对应的记录,那么FileEntry对象的mRowId会为0,而如果找到了,则为非0值。
与beginFile()相对的,就是endFile()了。endFile()是真正向MediaProvider数据库插入数据或更新数据的地方。当FileEntry的mRowId为0时,会考虑调用:
result = mMediaProvider.insert(mPackageName, tableUri, values);
而当mRowId为非0值时,则会考虑调用:
mMediaProvider.update(mPackageName, result, values, null, null);
这就是改变MediaProvider中相关信息的最核心句子啦。
endFile()的代码截选如下:
【frameworks/base/media/java/android/media/MediaScanner.java】
private Uri endFile(FileEntry entry, boolean ringtones, boolean notifications,
boolean alarms, boolean music, boolean podcasts)
throws RemoteException {
. . . . . .
ContentValues values = toValues();
String title = values.getAsString(MediaStore.MediaColumns.TITLE);
if (title == null || TextUtils.isEmpty(title.trim())) {
title = MediaFile.getFileTitle(values.getAsString(MediaStore.MediaColumns.DATA));
values.put(MediaStore.MediaColumns.TITLE, title);
}
. . . . . .
long rowId = entry.mRowId;
if (MediaFile.isAudioFileType(mFileType) && (rowId == 0 || mMtpObjectHandle != 0)) {
. . . . . .
values.put(Audio.Media.IS_ALARM, alarms);
values.put(Audio.Media.IS_MUSIC, music);
values.put(Audio.Media.IS_PODCAST, podcasts);
} else if (mFileType == MediaFile.FILE_TYPE_JPEG && !mNoMedia) {
. . . . . .
}
. . . . . .
if (rowId == 0) {
. . . . . .
// 扫描的是新文件,insert记录。如果是目录的话,必须比它所含有的所有文件更早插入记录,
// 所以在批量插入时,就需要有更高的优先权。如果是文件的话,而且我们现在就需要其对应
// 的rowId,那么应该立即进行插入,此时不过多考虑批量插入。
if (inserter == null || needToSetSettings) {
if (inserter != null) {
inserter.flushAll();
}
result = mMediaProvider.insert(mPackageName, tableUri, values);
} else if (entry.mFormat == MtpConstants.FORMAT_ASSOCIATION) {
inserter.insertwithPriority(tableUri, values);
} else {
inserter.insert(tableUri, values);
}
if (result != null) {
rowId = ContentUris.parseId(result);
entry.mRowId = rowId;
}
} else {
. . . . . .
mMediaProvider.update(mPackageName, result, values, null, null);
}
. . . . . .
return result;
}
除了直接调用mMediaProvider.insert()向MediaProvider中写入数据,函数中还有一种方式是经由inserter对象,其类型为MediaInserter。
MediaInserter也是向MediaProvider中写入数据,最终大体上会走到其flush()函数,该函数的代码如下:
【frameworks/base/media/java/android/media/MediaInserter.java】
private void flush(Uri tableUri, List<ContentValues> list) throws RemoteException {
if (!list.isEmpty()) {
ContentValues[] valuesArray = new ContentValues[list.size()];
valuesArray = list.toArray(valuesArray);
mProvider.bulkInsert(mPackageName, tableUri, valuesArray);
list.clear();
}
}
3 小节
写了这么多,终于看到MediaScannerService是如何更新MediaProvider的了。当然,里面还有大量的细节,本文就不展开来讲了,要不然相信大家头壳都得炸掉。那么就先写这么多了。