Title says most of it.
My application has been playing ringtones pointed by uri like content://media/internal/audio/media/387
or content://media/external/audio/media/1655
(for custom ringtones on SDcard I believe) using both setDataSource(fileInfo)
and setDataSource(mContext, Uri.parse(fileInfo))
.
In each case I have received logs with information about setDataSource failed.: status=0x80000000
exception on phones using Android 4.x (different versions).
Seeing that the error happens only to ringtones pointed by content uri, but not to single files pointed by path, I have decided to use paths for ringtones as well which fixed problem on above phones (while still using setDataSource(mContext, Uri.parse(fileInfo))
)
It has however started problems on phones with Android 2.3.4-2.3.6 (not on mine 2.3.3 though):
setDataSource failed.: status=0x80000000
for files with paths like /system/media/audio/ringtones/TwirlAway.ogg
I have also received a log about MediaPlayer.onErrorListener.onError(int what, int extra)
method call with what=1
and extra=-2147483648
, which, from what I know, suggest either that file is missing or it is corrupted. However I perform
File file = new File(fileInfo);
if (!file.exists())
check in such situation and it returned that file does exist - is it corrupted then? Highly unlikely for music file in internal memory.
To sum up:
setDataSource("content://media/internal/audio/media/52")
setDataSource failed.: status=0x80000000
for setDataSource(mContext, "/system/media/audio/ringtones/TwirlAway.ogg")
Interestingly, first few lines of setDataSource(Context context, Uri uri, Headers headers)
method which is called by setDataSource(Context context, Uri uri)
are (from GrepCode source for 2.3.4):
String scheme = uri.getScheme();
if(scheme == null || scheme.equals("file")) {
setDataSource(uri.getPath());
return;
}
So, after all, it just fails for setDataSource("/system/media/audio/ringtones/TwirlAway.ogg")
. I have taken paths to ringtones from uris by using:
private static String getRingtonePathFromContentUri(Context context,
Uri contentUri) {
String[] proj = { MediaStore.Audio.Media.DATA };
Cursor ringtoneCursor = context.getContentResolver().query(contentUri,
proj, null, null, null);
ringtoneCursor.moveToFirst();
return ringtoneCursor.getString(ringtoneCursor
.getColumnIndexOrThrow(MediaStore.Audio.Media.DATA));
}
Any ideas what can be causing error throwing? Maybe those are some issues caused by lack of reading permissions? I guess source code for native setDataSource(String path) function would help a lot, but I wasn't able to find it.
Answer by Lorne below was most helpful when dealing with this problem.
For anyone else struggling with it, here is the code that I have been using for over 6 months now with errors almost not reported anymore.
fileinfo
can be both of below (examples):
/system/media/audio/alarms/Walk_in_the_forest.ogg
content://media/internal/audio/media/20
public static void setMediaPlayerDataSource(Context context,
MediaPlayer mp, String fileInfo) throws Exception {
if (fileInfo.startsWith("content://")) {
try {
Uri uri = Uri.parse(fileInfo);
fileInfo = getRingtonePathFromContentUri(context, uri);
} catch (Exception e) {
}
}
try {
if (android.os.Build.VERSION.SDK_INT < android.os.Build.VERSION_CODES.HONEYCOMB)
try {
setMediaPlayerDataSourcePreHoneyComb(context, mp, fileInfo);
} catch (Exception e) {
setMediaPlayerDataSourcePostHoneyComb(context, mp, fileInfo);
}
else
setMediaPlayerDataSourcePostHoneyComb(context, mp, fileInfo);
} catch (Exception e) {
try {
setMediaPlayerDataSourceUsingFileDescriptor(context, mp,
fileInfo);
} catch (Exception ee) {
String uri = getRingtoneUriFromPath(context, fileInfo);
mp.reset();
mp.setDataSource(uri);
}
}
}
private static void setMediaPlayerDataSourcePreHoneyComb(Context context,
MediaPlayer mp, String fileInfo) throws Exception {
mp.reset();
mp.setDataSource(fileInfo);
}
private static void setMediaPlayerDataSourcePostHoneyComb(Context context,
MediaPlayer mp, String fileInfo) throws Exception {
mp.reset();
mp.setDataSource(context, Uri.parse(Uri.encode(fileInfo)));
}
private static void setMediaPlayerDataSourceUsingFileDescriptor(
Context context, MediaPlayer mp, String fileInfo) throws Exception {
File file = new File(fileInfo);
FileInputStream inputStream = new FileInputStream(file);
mp.reset();
mp.setDataSource(inputStream.getFD());
inputStream.close();
}
private static String getRingtoneUriFromPath(Context context, String path) {
Uri ringtonesUri = MediaStore.Audio.Media.getContentUriForPath(path);
Cursor ringtoneCursor = context.getContentResolver().query(
ringtonesUri, null,
MediaStore.Audio.Media.DATA + "='" + path + "'", null, null);
ringtoneCursor.moveToFirst();
long id = ringtoneCursor.getLong(ringtoneCursor
.getColumnIndex(MediaStore.Audio.Media._ID));
ringtoneCursor.close();
if (!ringtonesUri.toString().endsWith(String.valueOf(id))) {
return ringtonesUri + "/" + id;
}
return ringtonesUri.toString();
}
public static String getRingtonePathFromContentUri(Context context,
Uri contentUri) {
String[] proj = { MediaStore.Audio.Media.DATA };
Cursor ringtoneCursor = context.getContentResolver().query(contentUri,
proj, null, null, null);
ringtoneCursor.moveToFirst();
String path = ringtoneCursor.getString(ringtoneCursor
.getColumnIndexOrThrow(MediaStore.Audio.Media.DATA));
ringtoneCursor.close();
return path;
}
You have to explicit set the length of your file. Use the overloaded method:AssetFileDescriptor afd = ctx.getAssets().openFd([your asset name]);
mediaPlayer.setDataSource(afd.getFileDescriptor(), afd.getStartOffset(), afd.getLength());
There was a change in the behaviour of setDataSource(String path)
due to a bug fix in Android 4.1.1. In 4.1.1 or later you need to use a local path (without protocol). However, on 4.0.4 and earlier you needed to use a URI (e.g. with file:// protocol).
Here is an incomplete code snippet that should illustrate the workaround:
// as of 4.1.1 (JELLY_BEAN) we need to use a local path (without protocol)
// on 4.0.4 and earlier we needed a URI (with file:// protocol)
final String cachedFile = android.os.Build.VERSION.SDK_INT >= 16 // android.os.Build.VERSION_CODES.JELLY_BEAN
? getCacheFilePath(file)
: getCacheFileUri(file);
// for the purpose of this example
// assume cacheFolder is a String and getCacheFile returns a String
public String getCacheFilePath(String file) {
return cacheFolder + getCacheFile(file);
}
public String getCacheFileUri(String file) {
return "file://" + cacheFolder + getCacheFile(file);
}
This may be happening because of file format you are trying to play or compress. I was compressing a mp4 file which works well but when I compress a mov file the application crashed giving the setDataSource failed exception.
If you love us? You can donate to us via Paypal or buy me a coffee so we can maintain and grow! Thank you!
Donate Us With