HAL层访问Linux内核驱动程序

上一章学习了从C空间访问Linux内核驱动程序,这一章我们一起来学习如何在硬件抽象层中增加硬件模块来访问内核驱动程序。本章还是跟上一章一样,通过传统设备/dev/hello来连接硬件抽象层模块和内核驱动程序。

1.进入到/hardware/libhardware/include/hardware目录

新建hello.h
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
#ifndef ANDROID_HELLO_INTERFACE_H
#define ANDROID_HELLO_INTERFACE_H
#include <hardware/hardware.h>

__BEGIN_DECLS

/*定义模块ID*/
#define HELLO_HARDWARE_MODULE_ID "hello"

/*硬件模块结构体*/
struct hello_module_t {
struct hw_module_t common;
};

/*硬件接口结构体*/
struct hello_device_t {
struct hw_device_t common; //硬件设备结构
int fd; //设备文件描述符
int (*set_val)(struct hello_device_t* dev, int val);//写信号量
int (*get_val)(struct hello_device_t* dev, int* val);//读信号量
};

__END_DECLS

#endif

这里定义了模块ID—hello,模块结构体,硬件接口结构体。

2.进入到hardware/libhardware/modules目录,新建hello目录

添加hello.c文件
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
#define LOG_TAG "HelloStub"

#include <hardware/hardware.h>
#include <hardware/hello.h>
#include <fcntl.h>
#include <errno.h>
#include <cutils/log.h>
#include <cutils/atomic.h>

#define DEVICE_NAME "/dev/hello"
#define MODULE_NAME "Hello"

/*设备打开和关闭接口*/
static int hello_device_open(const struct hw_module_t* module, \
const char* name, struct hw_device_t** device)
;

static int hello_device_close(struct hw_device_t* device);

/*设备访问接口*/
static int hello_set_val(struct hello_device_t* dev, int val);
static int hello_get_val(struct hello_device_t* dev, int* val);

/*模块方法表*/
static struct hw_module_methods_t hello_module_methods = {
open: hello_device_open
};

/*模块实例变量*/
struct hello_module_t HAL_MODULE_INFO_SYM = {
common: {
tag: HARDWARE_MODULE_TAG,
version_major: 1,
version_minor: 0,
id: HELLO_HARDWARE_MODULE_ID,
name: MODULE_NAME,
methods: &hello_module_methods,
}
};

/*设备打开接口*/
static int hello_device_open(const struct hw_module_t* module, \
const char* name, struct hw_device_t** device)
{

struct hello_device_t* dev;
dev = (struct hello_device_t*)malloc(sizeof(struct hello_device_t));

if(!dev) {
ALOGE("Hello Stub: failed to alloc space");
return -EFAULT;
}

/*初始化设备成员,并完成相应的赋值操作*/
memset(dev, 0, sizeof(struct hello_device_t));
dev->common.tag = HARDWARE_DEVICE_TAG;
dev->common.version = 0;
dev->common.module = (hw_module_t*)module;
dev->common.close = hello_device_close;
dev->set_val = hello_set_val;dev->get_val = hello_get_val;

/*调用open函数,打开/dev/hello这个设备驱动*/
if((dev->fd = open(DEVICE_NAME, O_RDWR)) == -1) {
ALOGE("Hello Stub: failed to open /dev/hello -- %s.", strerror(errno));free(dev);
return -EFAULT;
}
/*设备指针指向赋值*/
*device = &(dev->common);
ALOGI("Hello Stub: open /dev/hello successfully.");

return 0;
}

/*设备关闭接口*/
static int hello_device_close(struct hw_device_t* device) {
struct hello_device_t* hello_device = (struct hello_device_t*)device;

/*如果设备句柄处于激活窗台,则关闭该句柄,清空内存*/
if(hello_device) {
close(hello_device->fd);
free(hello_device);
}

return 0;
}

/*设置信号量*/
static int hello_set_val(struct hello_device_t* dev, int val) {
ALOGI("Hello Stub: set value %d to device.", val);

/*调用write接口,往hello驱动写入寄存器值*/
write(dev->fd, &val, sizeof(val));

return 0;
}

/*读取信号量*/
static int hello_get_val(struct hello_device_t* dev, int* val) {
if(!val) {
ALOGE("Hello Stub: error val pointer");
return -EFAULT;
}

/*调用read接口,读取hello驱动的信号量值*/
read(dev->fd, val, sizeof(*val));

ALOGI("Hello Stub: get value %d from device", *val);

return 0;
}

该程序主要实现了/dev/hello驱动的打开和关闭,信号量的读写操作。

3.编译环境配置:

hello目录中创建Android.mk文件:

LOCAL_PATH := $(call my-dir)
include $(CLEAR_VARS)
LOCAL_MODULE_TAGS := optional
LOCAL_PRELINK_MODULE := false
LOCAL_MODULE_PATH := $(TARGET_OUT_SHARED_LIBRARIES)/hw
LOCAL_SHARED_LIBRARIES := liblog
LOCAL_SRC_FILES := hello.c
LOCAL_MODULE := hello.default
include $(BUILD_SHARED_LIBRARY)

这里的LOCAL_MODULE用的不是hello,而是hello.default,这样的好处是能够保证我们的模块总能被硬件抽象层访问到。

4.编译:

mmm hardware/libhardware/modules/hello
编译目录:out/target/product/generic/system/lib/hw/hello.default.so

5.打包成system.image

make snod

这样我们在为自己的设备驱动增加了一个硬件抽象层模块,后面我们一起学习JNI方法调用硬件抽象层接口的操作。

注:代码程序主要参考《老罗的Android之旅》来完成。

Tianger Ge wechat
如果您喜欢这篇文章,欢迎扫一扫我的微信公众号!