Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

MediaPlayer setDataSource failed with status=0x80000000 for Ringtone set by filepath on 2.3.4

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):

  • I have received few logs with exception: 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:

  • works with setDataSource("content://media/internal/audio/media/52")
  • throws exception: 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.

like image 692
Koger Avatar asked May 06 '13 09:05

Koger


4 Answers

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;
}
like image 99
Koger Avatar answered Nov 02 '22 11:11

Koger


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());

like image 30
Subtle Fox Avatar answered Nov 02 '22 12:11

Subtle Fox


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);
}
like image 39
Lorne Laliberte Avatar answered Nov 02 '22 11:11

Lorne Laliberte


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.

like image 21
VIVEK CHOUDHARY Avatar answered Nov 02 '22 12:11

VIVEK CHOUDHARY