概述
因為工作關(guān)系,最近有涉及到ADF(Atomic Display Framework)相關(guān)的內(nèi)容,部分內(nèi)容來自互聯(lián)網(wǎng)
ADF(Atomic Display Framework)是Google新增的Display框架,用來替換Framebuffer。 ADF在Android hwcomposer HAL和內(nèi)核驅(qū)動程序之間提供了以dma-buf為基礎(chǔ)的顯示框架原型
ADF的結(jié)構(gòu)圖引用自:http://blog.csdn.net/Lost_qwe/article/details/43113301
接下來就簡單說一下這些文件的作用。
接下來就簡單說一下這些文件的作用。
Driver:即使用ADF框架的custom編寫的程序
adf_fops.c:負(fù)責(zé)與user space交互的一個文件,實現(xiàn)了一些方法(open \ release \ read \ poll等)
adf_fobs32.c:用于兼容32位的一個文件,具體實現(xiàn)會在掉用到adf_fops.c這個文件。
adf_fbdev.c:fb設(shè)備對外的接口類,負(fù)責(zé)與fb設(shè)備兼容。
adf.c:這是整個ADF模塊的核心文件,會提供模塊內(nèi)部的各種服務(wù),主要提供了消息機制、同步機制(fence)以及整體ADF的初始化工作。
adf_client.c:主要用于調(diào)用custom編寫的驅(qū)動代碼以及喚醒(wake up)等。相當(dāng)于整個fromwork的消息最終出口。
adf_format.c:用于描述本啟動支持哪些圖像格式(RBG \ YUV以及具體的格式定義)。
adf_sysfs.c:與sysfs交互的一個文件。
adf_memblock.c:與內(nèi)存管理的一個文件,實現(xiàn)了一些DMA的ops然后注冊到DMA模塊中,實現(xiàn)對內(nèi)存的操作。
- 主要數(shù)據(jù)結(jié)構(gòu)
struct adf_obj;
struct adf_obj_ops;
struct adf_device;
struct adf_device_ops;
struct adf_interface;
struct adf_interface_ops;
struct adf_overlay_engine;
struct adf_overlay_engine_ops;
如上圖所示, adf子系統(tǒng)主要由通用數(shù)據(jù)接口和ops,顯示設(shè)備,顯示接口以及overlay的數(shù)據(jù)結(jié)構(gòu)和ops
”adf_obj“是用于創(chuàng)建sysfs文件系統(tǒng)的關(guān)鍵,所以在介紹其他類型之前,我們首先看看它的數(shù)據(jù)結(jié)構(gòu)
adf內(nèi)核文件系統(tǒng)基礎(chǔ)數(shù)據(jù)結(jié)構(gòu)
struct adf_file {
struct list_head head;//adf內(nèi)核文件系統(tǒng)雙向鏈表
struct adf_obj *obj;//sys文件節(jié)點數(shù)據(jù)結(jié)構(gòu),用于創(chuàng)建adf設(shè)備節(jié)點
DECLARE_BITMAP(event_subscriptions, ADF_EVENT_TYPE_MAX);
u8 event_buf[4096];//adf同步信號環(huán)形緩沖隊列
int event_head;
int event_tail;
wait_queue_head_t event_wait;//adf同步信號鎖
};
adf支持的event類型,我們用的多是就是vsync信號了
enum adf_event_type {
ADF_EVENT_VSYNC = 0,
ADF_EVENT_HOTPLUG = 1,
ADF_EVENT_DEVICE_CUSTOM = 128,
ADF_EVENT_TYPE_MAX = 255,
};
adf設(shè)備節(jié)點基礎(chǔ)數(shù)據(jù)結(jié)構(gòu)
struct adf_obj {
enum adf_obj_type type;//adf同步信號類型,主要有vsync,hotplug,custom
char name[ADF_NAME_LEN];//adf設(shè)備名稱
struct adf_device *parent;//上一級adf設(shè)備
const struct adf_obj_ops *ops;//adf ops集合
struct device dev;
struct spinlock file_lock;//adf信號同步,內(nèi)核與用戶空間文件拷貝鎖
struct list_head file_list;//adf文件系統(tǒng)數(shù)據(jù)結(jié)構(gòu)雙向鏈表集合
struct mutex event_lock;
struct rb_root event_refcount;
int id;
int minor;
};
· 這里是整個adf和userspace交互的主要通道,主要有ADF_OBJ_DEVICE, ADF_OBJ_INTERFACE以及ADF_OBJ_OVERLAY_ENGINE三個接口
ADF_OBJ_DEVICE---主要負(fù)責(zé)dma-buf, fence,post的配置和管理
ADF_OBJ_INTERFACE---主要負(fù)責(zé)與dispc相關(guān)的blank,dpm等接口配置和管理
ADF_OBJ_OVERLAY_ENGINE---overlay相關(guān)
我們首先看下read ioctl,adf event(包括vsync)將會在這里從內(nèi)核空間拷貝到用戶空間
在adf.c中提供了三個不同的信號接口供我們將DISPC或者Display Driver中接受到同步信號發(fā)出去,然后會在adf_file_queue_event函數(shù)中喚醒”event_wait“等待隊列
long adf_file_ioctl(struct file *file, unsigned int cmd, unsigned long arg)
{
struct adf_file *fpriv = file->private_data;
struct adf_obj *obj = fpriv->obj;
long ret = -EINVAL;
dev_dbg(&obj->dev, "%s ioctl %u\n", dev_name(&obj->dev), _IOC_NR(cmd));
switch (obj->type) {
case ADF_OBJ_OVERLAY_ENGINE:
ret = adf_overlay_engine_ioctl(adf_obj_to_overlay_engine(obj),
fpriv, cmd, arg);
break;
case ADF_OBJ_INTERFACE:
ret = adf_interface_ioctl(adf_obj_to_interface(obj), fpriv, cmd,
arg);
break;
case ADF_OBJ_DEVICE:
ret = adf_device_ioctl(adf_obj_to_device(obj), fpriv, cmd, arg);
break;
}
return ret;
}
我們首先看下read ioctl,adf event(包括vsync)將會在這里從內(nèi)核空間拷貝到用戶空間
在adf.c中提供了三個不同的信號接口供我們將DISPC或者Display Driver中接受到同步信號發(fā)出去,然后會在adf_file_queue_event函數(shù)中喚醒”event_wait“等待隊列
”event_wait“等待隊列被adf同步信號喚醒后,應(yīng)用層就可以通過ioctl讀取了
"adf_device_ioctl"是控制著整個adf的dma-buf,fence的配置和使用,這是整個adf的核心內(nèi)容。要理解這一塊內(nèi)容需要先了解dma-buf相關(guān)的API接口和fence的原型
以下引用自”http://blog.csdn.net/YKDSea/article/details/39995075“的描述:
下圖是"adf_device_ioctl"相關(guān)的流程圖
下面是”adf_interface_ioctl“相關(guān)的流程圖
這兩個ioctl里面的內(nèi)容很多(圖可以放大看),弄明白這兩個ioctl基本上整個adf框架也就理解差不多了,在后面我會挑出來單獨試著分析下(可能會誤人子弟)
|