Search code examples
androidandroid-mediaplayerringtone

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.


Solution

  • 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;
    }