如何在Android项目中使用ARM汇编代码?

Pho*_*non 14 assembly android arm native-code android-ndk

我对Assembly和ARM并不是特别有经验,但我能够在其中编写一些例程,我想看看它们是如何在配备ARM的Android设备(Nexus S)上运行的.将汇编代码文件包含到Android项目中的过程是什么?我是否只能从本机代码或Java中调用它?

小智 10

您可以使用Java Native Interface和Android NDK从Android调用程序集.

Cedric提到使用asm关键字,而我更喜欢包含汇编源代码.我已在我的网站上发布了一个教程:http: //www.eggwall.com/2011/09/android-arm-assembly-calling-assembly.html

您可以下载我的示例的源代码,看看它是如何工作的.一旦看到一个正常运行的示例,就可以根据需要轻松修改它.


Cir*_*四事件 5

带有内联和单独源文件的最小示例

必须注意不要在错误的架构下编译原始程序集。这里我们使用:

  • #ifdefs 在 C 文件上
  • ifeq儿子 Android.mk

这个例子在 GitHub 上。在 Ubuntu 16.04、Android NDK 12、Sony Xperia Z3 D6643 (ARMv7) 和 Android 5.1.1 上测试。

jni/main.c

#include <stdio.h>

#include <jni.h>

#ifdef __arm__
int asm_main(void);
#endif

jstring Java_com_cirosantilli_android_1cheat_ndk_1asm_Main_jniMethod(
        JNIEnv* env, jobject thiz) {
    enum Constexpr { N = 256 };
    char s[N];
    size_t cur = 0;

    int x = 0;
#ifdef __arm__
    cur += snprintf(s + cur, N - cur, "arm ");
    /* Inline test. Increment x by 1. */
    asm (
        "add %0, #1"
        : "=r" (x)
        : "0" (x)
    );
    /* Separate source test. Increment x by 1. */
    x += asm_main();
#endif
    if (x == 2)
        cur += snprintf(s + cur, N - cur, "%s", "0");
    else
        cur += snprintf(s + cur, N - cur, "%s", "1");

    return (*env)->NewStringUTF(env, s);
}
Run Code Online (Sandbox Code Playgroud)

jni/main_asm.S

.text
/* Function that just returns 1. */
.global asm_main
asm_main:
    mov r0, #1
    bx lr
Run Code Online (Sandbox Code Playgroud)

jni/Android.mk

LOCAL_PATH := $(call my-dir)
include $(CLEAR_VARS)
LOCAL_MODULE := main
LOCAL_SRC_FILES := main.c
# http://stackoverflow.com/questions/12614417/android-ndk-how-to-get-compiler-architecture-in-android-mk-dynamically
ifneq (,$(filter $(TARGET_ARCH_ABI),armeabi armeabi-v7a))
    LOCAL_SRC_FILES := $(LOCAL_SRC_FILES) main_asm.S
endif
include $(BUILD_SHARED_LIBRARY)
Run Code Online (Sandbox Code Playgroud)

com/cirosantilli/android_cheat/ndk_asm/Main.java

package com.cirosantilli.android_cheat.ndk_asm;

import android.app.Activity;
import android.widget.TextView;
import android.os.Bundle;

public class Main extends Activity {
    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        TextView tv = new TextView(this);
        tv.setText(jniMethod());
        setContentView(tv);
    }
    public native String jniMethod();
    static {
        System.loadLibrary("main");
    }
}
Run Code Online (Sandbox Code Playgroud)