Android8.1 将crash信息保存到sdcard

Android8.1 将crash信息保存到sdcard

背景:我做的是mtk平台,mtk本身有一个mtklogger可以保存日志,但是这个只在eng版本下会自动打开,user版本默认是关闭的,而且打开之后mtklogger有一个不可关闭的通知,这个看起来就很烦;
而且有一些发生crash的手机不能及时拿到,所以也无从分析问题所在,我的目标是将crash信息保存到手机中,然后在适当的时候上传的服务器,这样就可以远程分析问题了。

首先要在代码中找到crash信息,我通过crash的dialog最终找到了这个地方
framework/base/services/core/java/com/android/server/am/AppErrors.java
这个类看起来就感觉功能很明确,查看代码

void crashApplicationInner(ProcessRecord r, ApplicationErrorReport.CrashInfo crashInfo,
            int callingPid, int callingUid) {
        long timeMillis = System.currentTimeMillis();
        String shortMsg = crashInfo.exceptionClassName;
        String longMsg = crashInfo.exceptionMessage;
        String stackTrace = crashInfo.stackTrace;

        ......
    }

只看需要的部分,这里可以看到一个变量 stackTrace ,基本就可以确定是这个了,保险起见,打印log可以发现正是我所需要的crash信息。
现在的问题就是怎么把这个信息保存到手机sdcard中,在这里直接进行io读写,并不能成功。网上查了下,都说framework对文件进行读写操作比较麻烦,所以我就转移思路,把这个信息传递到apk模块内,在模块内在进行读写操作,我这里参考了IKeyguardService.aidl

新建framework/base/core/java/com/android/internal/policy/IErrorInfo.aidl

package com.android.internal.policy;


oneway interface IErrorInfo {

   void setError(String error);

}

新建framework/base/services/core/java/com/android/server/policy/ErrorInfoWrapper.java

package com.android.server.policy;

import android.content.Context;
import android.os.Bundle;
import android.os.IBinder;
import android.os.RemoteException;
import com.android.internal.policy.IErrorInfo;


public class ErrorInfoWrapper implements IErrorInfo {

    private IErrorInfo mService;

    public ErrorInfoWrapper(Context context, IErrorInfo service) {
        mService = service;
    }

    @Override
    public void setError(String error) {
        try {
            mService.setError(error);
        } catch (RemoteException e) {

        }
    }

    @Override // Binder interface
    public IBinder asBinder() {
        return mService.asBinder();
    }

}

新建framework/base/services/core/java/com/android/server/policy/ErrorInfoDelegate.java

package com.android.server.policy;

import android.app.ActivityManager;
import android.content.ComponentName;
import android.content.Context;
import android.content.Intent;
import android.content.ServiceConnection;
import android.os.Handler;
import com.android.internal.policy.IErrorInfo;
import com.android.server.UiThread;
import android.os.IBinder;
import android.os.UserHandle;

public class ErrorInfoDelegate {

    protected ErrorInfoWrapper mErrorInfoService;
    private final Context mContext;
    private final Handler mHandler;

    public ErrorInfoDelegate(Context context) {
        mContext = context;
        mHandler = UiThread.getHandler();
        bindService(context);
    }

    public void bindService(Context context) {
        Intent intent = new Intent();
        final ComponentName errorInfoComponent = ComponentName.unflattenFromString("com.android.systemui/com.android.systemui.keyguard.ErrorInfoService");
        intent.addFlags(Intent.FLAG_DEBUG_TRIAGED_MISSING);
        intent.setComponent(errorInfoComponent);

        context.bindServiceAsUser(intent, mErrorInfoConnection,
                Context.BIND_AUTO_CREATE, mHandler, UserHandle.SYSTEM);

    }

    private final ServiceConnection mErrorInfoConnection = new ServiceConnection() {
        @Override
        public void onServiceConnected(ComponentName name, IBinder service) {
            mErrorInfoService = new ErrorInfoWrapper(mContext,
                    IErrorInfo.Stub.asInterface(service));

        }

        @Override
        public void onServiceDisconnected(ComponentName name) {
            mErrorInfoService = null;

        }
    };

    public void setError(String error) {
        if (mErrorInfoService != null) {
            mErrorInfoService.setError(error);
        }
    }

}

在AppErrors.java中将crash信息传递到aidl中
services/core/java/com/android/server/am/AppErrors.java

private ErrorInfoDelegate mErrorInfoDelegate;
AppErrors(Context context, ActivityManagerService service) {
        context.assertRuntimeOverlayThemable();
        mService = service;
        mContext = context;
        /* begin */
        mService.mUiHandler.postDelayed(new Runnable() {
            @Override
            public void run() {
                mErrorInfoDelegate = new ErrorInfoDelegate(context);
            }
        }, 100000);
        /* end */
    }
void crashApplicationInner(ProcessRecord r, ApplicationErrorReport.CrashInfo crashInfo,
            int callingPid, int callingUid) {
        long timeMillis = System.currentTimeMillis();
        String shortMsg = crashInfo.exceptionClassName;
        String longMsg = crashInfo.exceptionMessage;
        String stackTrace = crashInfo.stackTrace;
        /* begin */
        if (mErrorInfoDelegate != null) {
            mErrorInfoDelegate.setError(shortMsg + longMsg + stackTrace);
        }
        /* end */
        if (shortMsg != null && longMsg != null) {
            longMsg = shortMsg + ": " + longMsg;
        } else if (shortMsg != null) {
            longMsg = shortMsg;
        }

        ......
}

这里实例化ErrorInfoDelegate的时候做了延时操作,主要怕影响到已有的逻辑。

到这里services中的操作已经完成,下面就是apk模块内的,我这里是在SystemUI中新增了一个service
AndroidManifest.xml中注册:

<service
                android:name=".keyguard.ErrorInfoService"
                android:exported="true"
                android:enabled="@bool/config_enableKeyguardService" />

新增ErrorInfoService.java

package com.android.systemui.keyguard;

import android.app.Service;
import android.content.Intent;
import android.os.*;
import com.android.internal.policy.IErrorInfo;
import com.android.systemui.util.LogUtil;

import java.io.File;
import java.io.FileOutputStream;
import java.io.IOException;

public class ErrorInfoService extends Service {

    @Override
    public void onCreate() {
    }

    @Override
    public IBinder onBind(Intent intent) {
        return mBinder;
    }

    private final IErrorInfo.Stub mBinder = new IErrorInfo.Stub() {

        @Override // Binder interface
        public void setError(String error) {
            writeText(error);
        }
    };

    private void writeText(String content) {

        if (content == null || content.equals("")) {
            return;
        }

        String filePath = "/storage/emulated/0/crashlog/";
        File dirFile = new File(filePath);
        if (!dirFile.exists()) {
            dirFile.mkdir();
        }
        String fileName = System.currentTimeMillis() + ".txt";
        File file = new File(filePath + fileName);
        FileOutputStream fos = null;
        try {
            fos = new FileOutputStream(file);
            fos.write(content.getBytes());
        } catch (Exception e) {
            e.printStackTrace();
        } finally {
            if (fos != null) {
                try {
                    fos.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
        }

    }

}

ok,这个时候如果有app发生了crash,那么在手机里就能保存这个crash信息了

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值