Android底层驱动开发记录:01_JNI

最近项目中需要用到了Android底层的开发,正好疫情居家所以又把韦老师的老教程第四期Android教程翻出来学习学习,手边也没有合适的板子,找了一块AIO-3288C的板子接了一块HDMI的屏来用。本来之前一直做单片机的,因此学起来还比较吃力,所以这里做一些记录,以备以后查看。

1. java学习,基本上跟着走了一遍,无波澜。C++因为会一点点就暂时跳过了,不行回来再补吧。

2. AIO-3288C环境搭建,最好是有一台真机 4G以上内存,硬盘至少100G,我这虚拟机空间有限编译不能通过,最后拿出很久之前买的腾讯云,Ubuntu18+2G内存+设置虚拟16G内存+100G硬盘,这个配置是偏低的但勉强用能编译SDK能通过。环境搭建上主要是open-7-jdk比较麻烦,可以参考我另一篇文章。最后达成结果就是,固件能完成RK3288C刷机,能Android studio联调app;mobaxterm ssh sftp获取云服务器资源调试源码(试图在腾讯云上装samba来着,很麻烦,放弃了);串口可以打印RK3288C启动日志;使用Android studio自带的adb设置一下环境变量用来adb调试RK3288C。

3. 开始敲代码吧。

(1)做一个APP,3288真机仿真一下,没难度,跟着做就好了。

package com.example.app_0001_leddemo;

import androidx.appcompat.app.AppCompatActivity;

import android.os.Bundle;
import android.view.View;
import android.widget.Button;
import android.widget.CheckBox;
import android.widget.Toast;
import com.example.hardlibrary.*;

public class MainActivity extends AppCompatActivity {

    private boolean ioonoff=false;
    private Button button=null;
    private CheckBox checkBox_io1=null;
    private CheckBox checkBox_io2=null;
    private CheckBox checkBox_io3=null;
    private CheckBox checkBox_io4=null;

    public void onCheckboxClicked(View view) {
        boolean checked= ((CheckBox)view).isChecked();

        switch (view.getId()){
            case R.id.checkbox_io1:
                if(checked){
                    Toast.makeText(getApplicationContext(), "checkbox1 on ",Toast.LENGTH_SHORT).show();
                    HardCtrl.ioCtrl(0,1);
                }else {
                    Toast.makeText(getApplicationContext(), "checkbox1 off",Toast.LENGTH_SHORT).show();
                    HardCtrl.ioCtrl(0,0);
                }
                break;
            case R.id.checkbox_io2:
                if(checked){
                    Toast.makeText(getApplicationContext(), "checkbox2 on ",Toast.LENGTH_SHORT).show();
                    HardCtrl.ioCtrl(1,1);
                }else {
                    Toast.makeText(getApplicationContext(), "checkbox2 off",Toast.LENGTH_SHORT).show();
                    HardCtrl.ioCtrl(1,0);
                }
                break;
            case R.id.checkbox_io3:
                if(checked){
                    Toast.makeText(getApplicationContext(), "checkbox3 on ",Toast.LENGTH_SHORT).show();
                    HardCtrl.ioCtrl(2,1);
                }else {
                    Toast.makeText(getApplicationContext(), "checkbox3 off",Toast.LENGTH_SHORT).show();
                    HardCtrl.ioCtrl(2,0);
                }
                break;
            case R.id.checkbox_io4:
                if(checked){
                    Toast.makeText(getApplicationContext(), "checkbox4 on ",Toast.LENGTH_SHORT).show();
                    HardCtrl.ioCtrl(3,1);
                }else {
                    Toast.makeText(getApplicationContext(), "checkbox4 off",Toast.LENGTH_SHORT).show();
                    HardCtrl.ioCtrl(3,0);
                }
                break;
        }
    }

    class MyButtonListener implements View.OnClickListener{
        @Override
        public void onClick(View v) {

            //HardCtrl hardCtrl = new HardCtrl();

            ioonoff = !ioonoff;
            if(ioonoff==true){
                button.setText("ALL OFF");
                checkBox_io1.setChecked(true);
                checkBox_io2.setChecked(true);
                checkBox_io3.setChecked(true);
                checkBox_io4.setChecked(true);

                for(int i=0;i<4;i++)
                    HardCtrl.ioCtrl(i,1);
            }else {
                button.setText("ALL ON");
                checkBox_io1.setChecked(false);
                checkBox_io2.setChecked(false);
                checkBox_io3.setChecked(false);
                checkBox_io4.setChecked(false);

                for(int i=0;i<4;i++)
                    HardCtrl.ioCtrl(i,0);
            }
        }
    }

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        HardCtrl.ioOpen();

        button = (Button)findViewById(R.id.bottom_id1);
        checkBox_io1 = (CheckBox)findViewById(R.id.checkbox_io1);
        checkBox_io2 = (CheckBox)findViewById(R.id.checkbox_io2);
        checkBox_io3 = (CheckBox)findViewById(R.id.checkbox_io3);
        checkBox_io4 = (CheckBox)findViewById(R.id.checkbox_io4);

        button.setOnClickListener(new MyButtonListener());
//        button.setOnClickListener(new View.OnClickListener() {
//            public void onClick(View v) {
//                // Code here executes on main thread after user presses button
//
//                ioonoff = !ioonoff;
//                if(ioonoff==true){
//                    button.setText("ALL ON");
//                }else {
//                    button.setText("ALL OFF");
//                }
//            }
//        });
    }
}

(2)app加载jni部分,也没什么难度。

package com.example.hardlibrary;

public class HardCtrl{
    public static native int ioCtrl(int which,int status);
    public static native int ioOpen();
    public static native void ioClose();

    static {
        try {
            System.loadLibrary("hardctrl");
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
}

(3)JNI部分,这部分稍微麻烦一些。完成c代码编写是第一步,第二步在我的腾讯云(TXY)上安装arm-linux-gcc 4.3.2(其他的交叉编译工具好像是有些问题搞这个),并编译jni c代码命令如下,

arm-linux-gcc  -fPIC -shared hardctrl.c -o libhardctrl.so -I /usr/lib/jvm/java-1.7.0-openjdk-amd64/include/ -nostdlib /home/ubuntu/firefly_rk3288_android5.1/prebuilts/ndk/9/platforms/android-19/arch-arm/usr/lib/libc.so -I /home/ubuntu/firefly_rk3288_android5.1/prebuilts/ndk/9/platforms/android-19/arch-arm/usr/include/ /home/ubuntu/firefly_rk3288_android5.1/prebuilts/ndk/9/platforms/android-14/arch-arm/usr/lib/liblog.so

没有-nostdlib  编译可以通过但调试会奔溃,因为libc的依赖库不对,可以用ldd xxx.so 查看库依赖。第三步编译好的so库放在app目录下app/libs中。第四步配置Android studio的app工程libs的路径,主要是sourceSets 和debug,好了可以调试仿真了。

   sourceSets {
        main {
            jniLibs.srcDirs = ['libs']
        }
    }

    buildTypes {
        release {
            minifyEnabled false
            proguardFiles getDefaultProguardFile('proguard-android-optimize.txt'), 'proguard-rules.pro'
        }
        debug {
            ndk {
                abiFilters "armeabi","x86"
            }
        }
    }

此时, 只是app上能点击IO开关了,还没有深入到硬件呢。下面是jni主要的代码。

static const JNINativeMethod methods[] = {
	{"ioOpen", "()I", (void *)ioopen},
	{"ioClose", "()V", (void *)ioclose},
	{"ioCtrl", "(II)I", (void *)ioctrl},
};




/* System.loadLibrary */
JNIEXPORT jint JNICALL
JNI_OnLoad(JavaVM *jvm, void *reserved)
{
	JNIEnv *env;
	jclass cls;

	if ((*jvm)->GetEnv(jvm, (void **)&env, JNI_VERSION_1_4)) {
		return JNI_ERR; /* JNI version not supported */
	}
	cls = (*env)->FindClass(env, "com/example/hardlibrary/HardCtrl");
	if (cls == NULL) {
		return JNI_ERR;
	}

	/* 2. map java hello <-->c c_hello */
	if ((*env)->RegisterNatives(env, cls, methods, sizeof(methods)/sizeof(methods[0])) < 0)
		return JNI_ERR;

	return JNI_VERSION_1_4;
}

(4)RK3288c开发板IO驱动编写,这个麻烦就大一些了。 按照其官网维基并结合正点教程完成了驱动。但jni open这个驱动却出现问题,纠其原因是该驱动 权限不足,导致jni open不了,这里找到两个办法解决,

                ① adb root,chmod 777 /dev/xxx  重启后无效

                ②可以在ueventd.rc中赋权比如:system/core/rootdir/ueventd.rc,

/dev/tty*0777rootsystem
设备节点修改权限该节点的用户所属的组

                       修改完成后,使用make bootimage 编译命令 会更新ramdisk.img,boot.img,   ueventd.rc

这一阶段的测试就到这里,源码在这里:https://download.csdn.net/download/bwolf1986/85473568?spm=1001.2014.3001.5501

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值