java - Xamarin Android native 调用堆栈

标签 java c# xamarin xamarin.android monodevelop

我是新的 Xamarin 开发人员,我想为 Xamarin 开发准备诊断工具。就我而言,我想从 C# 异常获取托管堆栈跟踪,并从其余库(如果可能的话,C++、Java)获取非托管堆栈跟踪。我尝试调用 C++ 方法,使用一些奇怪的选项初始化控件,调用 java 库(谷歌地图),但我仍然没有 native 调用堆栈的堆栈跟踪。

您能帮我解释一下如何从 Xamarin Android 应用程序获取托管和非托管堆栈帧的堆栈跟踪吗?

非常感谢!

最佳答案

unmanaged stack trace from rest of libraries (C++, java)

就 C 而言,没有异常处理,C++ 确实有异常,但它们与 CIL 管理的异常是完全不同的动物。现在,您可以在 NDK/C++ 代码中实现 C++ 异常,如果可能的话,根据异常和继续运行的能力,抑制异常,并通过 Mono 运行时创建一个托管异常,以某种方式表示 C++ 异常。有关运行时异常传播的更多信息,请参阅 Mono 文档。就我个人而言,我只是在 C++ 级别处理 C++ 异常、记录它们、报告它们等等...并且不会尝试将异常传播回托管代码(开销太大,工作返回太少)。

就各种 Xamarin.Android/Android 异常而言:

Mono 运行时/NDK 崩溃

考虑类似 libc 的字符串 strncpy:

[DllImport("libc.so")]
private static extern void strncpy(StringBuilder dest, string src, uint n);

并使用 null(或 IntPtr.Zero)调用它:

strncpy(null, null, 1);

一次严重的运行时崩溃:

[mono-rt] Stacktrace:
[mono-rt] 
[mono-rt]   at <unknown> <0xffffffff>
[mono-rt]   at (wrapper managed-to-native) Android_Clipboard.MainActivity.strncpy (System.Text.StringBuilder,string,uint) <0x00012>
[mono-rt]   at Android_Clipboard.MainActivity/<>c__DisplayClass5_0.<OnCreate>b__0 (object,System.EventArgs) [0x0004f] in /Volumes/Code/code/Projects/Android_Clipboard/Android_Clipboard/MainActivity.cs:57
[mono-rt]   at Android.Views.View/IOnClickListenerImplementor.OnClick (Android.Views.View) [0x00011] in <25661073a35344a89f215a4cf81af37c>:0
[mono-rt]   at Android.Views.View/IOnClickListenerInvoker.n_OnClick_Landroid_view_View_ (intptr,intptr,intptr) [0x00011] in <25661073a35344a89f215a4cf81af37c>:0
[mono-rt]   at (wrapper dynamic-method) object.cea8479a-a63b-41ac-a9f6-b0fe3b09df49 (intptr,intptr,intptr) [0x00017] in <f32579baafc1404fa37ba3ec1abdc0bd>:0
[mono-rt]   at (wrapper native-to-managed) object.cea8479a-a63b-41ac-a9f6-b0fe3b09df49 (intptr,intptr,intptr) [0x00022] in <f32579baafc1404fa37ba3ec1abdc0bd>:0
[mono-rt] /proc/self/maps:
[mono-rt] 12c00000-42c00000 rw-p 00000000 00:01 207683     /dev/ashmem/dalvik-main space (region space) (deleted)
[mono-rt] 70dca000-70f78000 rw-p 00000000 fc:00 106531     /data/dalvik-cache/x86/system@<a href="https://stackoverflow.com/cdn-cgi/l/email-protection" class="__cf_email__" data-cfemail="88eefae9e5edffe7fae3c8eae7e7fca6e9fafc" rel="noreferrer noopener nofollow">[email protected]</a>
[mono-rt] 70f78000-71014000 rw-p 00000000 fc:00 106532     
~~~
[mono-rt] No native Android stacktrace (see debuggerd output).
[mono-rt] 
[mono-rt] 
[mono-rt] =================================================================
[mono-rt] Got a SIGSEGV while executing native code. This usually indicates
[mono-rt] a fatal error in the mono runtime or one of the native libraries 
[mono-rt] used by your application.
[mono-rt] =================================================================
[mono-rt] 
[libc] Fatal signal 11 (SIGSEGV), code 1, fault addr 0x0 in tid 14916 (droid_Clipboard), pid 14916 (droid_Clipboard)

Mono/CIL 托管异常

从 CIL Mono VM 到 Java 的任何调用都会通过 VM2VM 桥(CilVM<->Mono/JNI<->JavaVM)传递,并由 Mono JNIEnv 处理,因此由 Mono 运行时包装在托管异常中。

考虑反射(reflect)不存在的 Java 字段(来自 C#):

var someWidget = FindViewById<Button>(Resource.Id.myButton);
Java.Lang.Reflect.Field f = someWidget.Class.GetField("SomethingThatDoesNotExist");

生成一个基于 Java 的 java.lang.NoSuchFieldException,该异常被包装并呈现为托管异常:

MonoDroid] UNHANDLED EXCEPTION:
[MonoDroid] Java.Lang.NoSuchFieldException: SomethingThatDoesNotExist
[MonoDroid]   at System.Runtime.ExceptionServices.ExceptionDispatchInfo.Throw () [0x0000c] in <f32579baafc1404fa37ba3ec1abdc0bd>:0 
[MonoDroid]   at Java.Interop.JniEnvironment+InstanceMethods.CallObjectMethod (Java.Interop.JniObjectReference instance, Java.Interop.JniMethodInfo method, Java.Interop.JniArgumentValue* args) [0x00069] in <09bf3e262b934ffab2ba01f9fc7fd54d>:0 
[MonoDroid]   at Java.Interop.JniPeerMembers+JniInstanceMethods.InvokeAbstractObjectMethod (System.String encodedMember, Java.Interop.IJavaPeerable self, Java.Interop.JniArgumentValue* parameters) [0x00014] in <09bf3e262b934ffab2ba01f9fc7fd54d>:0 
[MonoDroid]   at Java.Lang.Class.GetField (System.String name) [0x00029] in <25661073a35344a89f215a4cf81af37c>:0 
[MonoDroid]   at Android_Clipboard.MainActivity+<>c__DisplayClass5_0.<OnCreate>b__0 (System.Object <p0>, System.EventArgs <p1>) [0x00048] in /Volumes/Code/code/Projects/Android_Clipboard/Android_Clipboard/MainActivity.cs:53 
[MonoDroid]   at Android.Views.View+IOnClickListenerImplementor.OnClick (Android.Views.View v) [0x00011] in <25661073a35344a89f215a4cf81af37c>:0 
[MonoDroid]   at Android.Views.View+IOnClickListenerInvoker.n_OnClick_Landroid_view_View_ (System.IntPtr jnienv, System.IntPtr native__this, System.IntPtr native_v) [0x0000f] in <25661073a35344a89f215a4cf81af37c>:0 
[MonoDroid]   at (wrapper dynamic-method) System.Object.7ad62bce-4e52-4d67-9099-969546fceb57(intptr,intptr,intptr)
[MonoDroid]   --- End of managed Java.Lang.NoSuchFieldException stack trace ---
[MonoDroid] java.lang.NoSuchFieldException: SomethingThatDoesNotExist
[MonoDroid]     at java.lang.Class.getField(Class.java:1601)
[MonoDroid]     at mono.android.view.View_OnClickListenerImplementor.n_onClick(Native Method)
[MonoDroid]     at mono.android.view.View_OnClickListenerImplementor.onClick(View_OnClickListenerImplementor.java:30)
[MonoDroid]     at android.view.View.performClick(View.java:6294)
[MonoDroid]     at android.view.View$PerformClick.run(View.java:24770)
[MonoDroid]     at android.os.Handler.handleCallback(Handler.java:790)
[MonoDroid]     at android.os.Handler.dispatchMessage(Handler.java:99)
[MonoDroid]     at android.os.Looper.loop(Looper.java:164)
[MonoDroid]     at android.app.ActivityThread.main(ActivityThread.java:6494)
[MonoDroid]     at java.lang.reflect.Method.invoke(Native Method)
[MonoDroid]     at com.android.internal.os.RuntimeInit$MethodAndArgsCaller.run(RuntimeInit.java:438)
[MonoDroid]     at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:807)
[MonoDroid] 

“原生”Java 异常

考虑一个 Java 异常,它不会在 Mono/CIL 调用的上下文中发生:

Java:

private static void stackOverflowLoop() {
    CrashMe.stackOverflowLoop();
}
public void stackOverflow() {
    new Handler().postDelayed(new Runnable() {
        public void run() {
            CrashMe.stackOverflowLoop();
        }
    }, 2000);
}

C#:

new CrashMe().StackOverflow();

产生一个在 Android 运行时中捕获的“纯”Java 异常(请记住,Xamarin.Android 应用程序只是一个带有 CIL 的基于 NDK 的 C 应用程序) VM 和对 Java VM 的访问)

AndroidRuntime] Shutting down VM
[AndroidRuntime] FATAL EXCEPTION: main
[AndroidRuntime] Process: com.sushihangover.Android_Clipboard, PID: 23664
[AndroidRuntime] java.lang.StackOverflowError: stack size 8MB
[AndroidRuntime]    at com.sushihangover.mylibrary.CrashMe.stackOverflowLoop(CrashMe.java:7)
~~~~
[AndroidRuntime]    at com.sushihangover.mylibrary.CrashMe.stackOverflowLoop(CrashMe.java:7)
[AndroidRuntime]    at com.sushihangover.mylibrary.CrashMe.access$000(CrashMe.java:5)
[AndroidRuntime]    at com.sushihangover.mylibrary.CrashMe$1.run(CrashMe.java:12)
[AndroidRuntime]    at android.os.Handler.handleCallback(Handler.java:790)
[AndroidRuntime]    at android.os.Handler.dispatchMessage(Handler.java:99)
[AndroidRuntime]    at android.os.Looper.loop(Looper.java:164)
[AndroidRuntime]    at android.app.ActivityThread.main(ActivityThread.java:6494)
[AndroidRuntime]    at java.lang.reflect.Method.invoke(Native Method)
[AndroidRuntime]    at com.android.internal.os.RuntimeInit$MethodAndArgsCaller.run(RuntimeInit.java:438)
[AndroidRuntime]    at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:807)

基于 Java 的 Android 无响应 (ANR)

仅被 Android 操作系统捕获/困住。

Java:

public void androidNotReponding() {
    new Handler().postDelayed(new Runnable() {
        public void run() {
            while (true) {
            }
        }
    }, 2000);

C#

new CrashMe().AndroidNotReponding();

ANR Logcat:

[zygote] Thread[3,tid=25575,WaitingInMainSignalCatcherLoop,Thread*=0xb224ec00,peer=0x13880020,"Signal Catcher"]: reacting to signal 3
[zygote] 
[zygote] Wrote stack traces to '[tombstoned]'
E ActivityManager: ANR in com.sushihangover.Android_Clipboard (com.sushihangover.Android_Clipboard/md58bbdf49b937b7d185e88b8dfe7516c89.MainActivity)
E ActivityManager: PID: 25886
E ActivityManager: Reason: Input dispatching timed out (Waiting to send non-key event because the touched window has not finished processing certain input events that were delivered to it over 500.0ms ago. 

ANR 神石追踪:

Cmd line: com.sushihangover.Android_Clipboard
~~~~
Total blocking GC count: 0
Total blocking GC time: 0
~~~~
suspend all histogram:  Sum: 576us 99% C.I. 2us-329.200us Avg: 30.315us Max: 346us
DALVIK THREADS (13):
"main" prio=5 tid=1 Runnable
  | group="main" sCount=0 dsCount=0 obj=0x73997258 self=0xb4034500
  | sysTid=4484 nice=0 cgrp=default sched=0/0 handle=0xb7728c00
  | state=R schedstat=( 0 0 0 ) utm=6486 stm=688 core=3 HZ=100
  | stack=0xbf21d000-0xbf21f000 stackSize=8MB
  | held mutexes= "mutator lock"(shared held)
  at com.sushihangover.mylibrary.CrashMe$3.run(CrashMe.java:26)
  at android.os.Handler.handleCallback(Handler.java:739)
  at android.os.Handler.dispatchMessage(Handler.java:95)
  at android.os.Looper.loop(Looper.java:148)
  at android.app.ActivityThread.main(ActivityThread.java:5417)
  at java.lang.reflect.Method.invoke!(Native method)
  at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:726)
  at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:616)

"Signal Catcher" daemon prio=5 tid=2 Runnable
  | group="system" sCount=0 dsCount=0 obj=0x12c840a0 self=0xae4c2000
  | sysTid=4488 nice=0 cgrp=default sched=0/0 handle=0xb3432930
  | state=R schedstat=( 0 0 0 ) utm=0 stm=0 core=0 HZ=100
  | stack=0xb3336000-0xb3338000 stackSize=1014KB
  | held mutexes= "mutator lock"(shared held)
  native: #00 pc 0058bd02  /system/lib/libart.so (art::DumpNativeStack(std::__1::basic_ostream<char, std::__1::char_traits<char> >&, int, char const*, art::ArtMethod*, void*)+226)
  native: #01 pc 0055194e  /system/lib/libart.so (art::Thread::Dump(std::__1::basic_ostream<char, std::__1::char_traits<char> >&) const+286)
 ~~~~

关于java - Xamarin Android native 调用堆栈,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/50417941/

相关文章:

c# - Windows CE 上 GlobalMemoryStatusEx 的 PInvoke

ios - Xamarin iOS - 自定义字体未出现在主 Storyboard 中

xamarin - 我们可以通过 Xamarin 生成二维码吗

java - 使用私钥后如何从内存中清除私钥数据?

java - JSF Composite 组件仅在一页中运行

c# - ASP.NET Core,如何在依赖注入(inject)中配置通用服务?

c# - 如何从数据库中读取 byte[] 并将其转换为图像?

c# - 如何使用 UIAlertController 显示来自 UIViewController 的警报消息

java - 如何通过内部对象创建进行 JUnit 测试

Java 二维数组作为表格,打印出索引