de项目需要,要实现在Android中实现串口的收发功能,有几种方法可以参考使用。

1. 标准的Android HAL层思想,把串口的功能加入framework的API中(类似于android中sensor的实现)

a. 确保驱动层中基于tty的串口驱动可以正常read、write、poll数据,当然了,也可以自己写一个字符驱动来实现串口的读写功能。

b. 在BSP的HAL层中添加串口读写功能的回调函数(linux 应用层 c/c++)

c. Android framework中添加jni层,解析HAL中生成的module,然后对回调函数进行封装,生成.so库,提供给java层。

d. 添加远程调用接口,使用aidl在framework中添加远程调用

e. 添加serviceManagement

2. 绕过HAL,直接使用JNI来完成读写等回调函数,之后同1 。

3. 绕过android系统,直接编写jni库,在应用程序中直接调用jni接口,完成串口的收发。

------------------------------------------------------------------------------

以上都是可用的方法,这里我采用最简单的第三种方法,其中第一种方法最繁琐,但也是android最标准的方法,之后我会在can bus的移植中使用(先打个哑谜^0^),OK 废话不多说,开始码代码,工作!

首先是驱动层,我使用的是fsl的开发板,这边freescale已经帮我们实现了驱动,可以在/dev/下发现ttymxc0,ttymxc1.。。。这些就是CPU上各个串口的驱动文件,可以尝试echo "123" > /dev/mxctty0 之后可以看到串口终端上会打印出“123”。

但是,我们做驱动的不能就这样拿着别人的东西就用,咱要分析,要学习,要膜拜,要抄袭,要。。。貌似我最喜欢干这种事情了,好吧,这里我自己照着Linux设备驱动详解这书写了一个虚拟的字符驱动,当做我们的串口吧。

提供了跟串口同样的功能,这个驱动中我使用阻塞的方式来读写数据,一边看书,一边学习,一边自己写代码,一边学习jni,一边学习android的框架,何乐而不为呢?

首先,我们要注册一个字符驱动,然后初始化等待队列,初始化信号量,初始化变量,给结构体分配内存空间,老一套了。。。是个写驱动的都知道要干这些事情。


看到没,这里使用了miscdevice驱动,这个简单容易实现,HOHO~~偷懒了。这里给我们的全局结构体分配了内存空间,然后把结构体操作函数挂到我们的全局结构体变量中,最后注册这个miscdevice驱动。


看到我们的globalfifo结构体的定义了吧,这里,就是这里,所以在init函数中,我们要初始化信号量,初始化读写等待队列头。要不咱先来讲讲这里的阻塞的概念吧。

顾名思义,就是堵在那边不动了,其实是真的不动了,利用等待队列实现设备的阻塞,当用户进程访问系统资源的时候,当这个资源不能被访问,我们又不想让之后的事情继续发生,这样的话我们就可以阻塞在那边,放心,我们可以让该进程进入休眠,这样的话就不会浪费CPU的资源了,然而等到这个资源可以访问的时候,我们就可以唤醒该阻塞的进程,继续让他执行下去,如果没有地方唤醒他,那他就真的“堵死”在那边了。

简单的介绍了下,接下来看看我们要实现哪些功能函数


咱有读,写,打开。。。。等函数,继续往下分析。


open和release函数没什么好说的了,其实这里还是蛮有讲究的,比如说这个设备我们只能让一个用户进行访问,那我们可以再open函数里面做点手脚,一般我们读内核驱动模型的时候都会看到很多时候在open函数中都会设计引用计数自加1,这样的话可以更好的管控我们设备被打开次数。

但是这里我们没做什么,我们只是把我们的全局结构体变量赋值给了这里filp的一个私有成员变量中,这样的话我们可以再每一个功能函数中取出这个私有成员,有利于代码的可读性,release就不讲了。


然后这就是我们的读函数,在进行读之前,我们把等待队列加进我们的队列链表中,然后检查我们的buff是否为空,如果为空的话,那就没什么好读的了,所以我们让进城休眠,当有货给我们读了,再唤醒我们的队列。

首先是把当前进城加入等待队列中add_wait_queue(&dev->r_wait, &wait);

没东西读的时候,使进程睡眠,在调度到别的任务去


这段代码比较关键,与写函数中一样,当我们的buff被写满时,我们也会发生阻塞。


写函数最后会唤醒我们的等待队列,因为写进去东西了,就可以去读了,就是这样,这部分跟我们的串口收发相同。

别的功能我就不说了,OK,驱动完成之后,我们加载进去,然后进行测试下。

首先我们去cat /dev/globalfifo

发生阻塞,一直停在那,这时候我们再打开一个终端,去写数据

echo "123" > /dev/globalfifo

写完之后,我们立马会发现之前的cat有东西出来了,每次都会把数据全部读出来。

==================================================

下面是我们的jni,首先咱要明确我们做的事情,打开设备,读设备,最后不用的话就关闭设备,所以我们至少要实现这3个API,


这是我们的初始化函数,定义了一个全局的文件描述符,init函数只做了open的动作。


这个API封装了read函数,然后把读到的buff转换成为java中能识别的string,最后返回到java层的是string类型的字符串。


最后这是我们的exit函数,调用了close来关闭我们的设备。然后编写Android.mk文件,最后编译,生成globalfifo库

===========================================

接下来我们创建一个Android 工程,导入jni库并且定义native API


然后在适当的地方调用。设置3个按键,先试打开,然后read,按下read按键的时候开启一个thread去读数据。


安装apk,然后运行程序,点击open,然后点击read,使用adb shell进入系统,然后往里面写东西

echo "Jay Zhang" > dev/globalfifo

可以看到有Jay Zhang 吐出来。

=================================================

这样就模拟了串口,之后我们会用标准的android流程来完成can bus在android 设备上的开发。

更多相关文章

  1. 箭头函数的基础使用
  2. Python技巧匿名函数、回调函数和高阶函数
  3. Unity3D 调用 Android(安卓)jar 包制作方法
  4. Android(安卓)GPIO LED 驱动与HAL分析
  5. android 双击屏幕触发事件,放大图片,不用接入借口api,不用OnGestur
  6. Android(安卓)P 图像显示系统(三)Android(安卓)HWUI 绘制流程
  7. 一个使用FFmpeg库读取3gp视频的例子-Android中使用FFmpeg媒体库
  8. 浅析Android线程模型
  9. Android(安卓)系统的血液:看看我是怎么对Handler用的得心应手

随机推荐

  1. android学习笔记(一)
  2. Android(安卓)mediaplayer
  3. Android中UI设计的一些技巧!!!
  4. Android(安卓)JNI介绍
  5. Android(安卓)解决ListView 和 ScrollVie
  6. 解决 Android(安卓)在Eclipse 开发中 Cla
  7. Android(安卓)布局 精准定位 平衡定位 相
  8. android之SharedPreferences进行数据存储
  9. android salite3 使用实例
  10. cocos2d-x学习笔记06:如何将win32移植到an