當前位置:首頁 > 公眾號精選 > 嵌入式大雜燴
[導(dǎo)讀]開發(fā)板上有AP3216三合一整合型光感測器,看了看出廠SDK包中并未添加相關(guān)驅(qū)動。本次我們就一起來學(xué)習(xí)一下。


1、前言

開發(fā)板上有AP3216三合一整合型光感測器,看了看出廠SDK包中并未添加相關(guān)驅(qū)動。本次我們就一起來學(xué)習(xí)一下。

2、AP3216簡介

AP3216C 芯片集成了光強傳感器( ALS: Ambient Light Sensor),接近傳感器( PS: Proximity Sensor),還有一個紅外 LED( IR LED)。

這個芯片設(shè)計的用途是給手機之類的使用,比如:返回當前環(huán)境光強以便調(diào)整屏幕亮度;用戶接聽電話時,將手機放置在耳邊后,自動關(guān)閉屏幕避免用戶誤觸碰 。

該芯片通過 I2C 接口與主控制器相連, 如:

3、IIC驅(qū)動簡介

Linux下IIC有兩種驅(qū)動方式:一種是按照字符設(shè)備驅(qū)動方式來驅(qū)動IIC;另一種是走Linux下IIC的框架。按照字符設(shè)備驅(qū)動的方式可以查閱這一篇文章:Linux IIC 字符設(shè)備 驅(qū)動例子。

這里我們淺淺地(真的很淺~~)了解學(xué)習(xí)一下第二種方式,因為找到的AP3216的驅(qū)動就是基于IIC驅(qū)動框架的,哈哈。

整個IIC的驅(qū)動框架相關(guān)代碼在drivers\i2c中,包含的內(nèi)容有:

IIC驅(qū)動框架圖如(圖片來源于網(wǎng)絡(luò),鏈接見文末參考資料):

IIC驅(qū)動框架可大體分為兩大部分:

①  I2C 總線驅(qū)動:SOC 的 I2C 控制器驅(qū)動,也叫做 I2C 適配器驅(qū)動。

②  I2C 設(shè)備驅(qū)動:針對具體的 I2C 設(shè)備而編寫的驅(qū)動。

其中,訪問抽象層與I2C核心層數(shù)據(jù)I2C 總線驅(qū)動部分;driver驅(qū)動層屬于I2C設(shè)備驅(qū)動部分。

上面框圖對應(yīng)的代碼調(diào)用層次圖如:

下面的AP3216驅(qū)動可以對照這張圖來看看。

4、AP3216實驗

我們使用設(shè)備樹來描述AP3216設(shè)備信息,首先我們沒有在設(shè)備樹中添加AP3216相關(guān)節(jié)點時,我們系統(tǒng)的I2C設(shè)備如:

添加I2C pinctrl,板子上AP3216接的是I2C1:

配置寄存器的值都設(shè)為0x4001b8b0,這一段是什么意思我們在什么是Pinctrl子系統(tǒng)及GPIO子系統(tǒng)?這篇筆記中也有寫到,就是幾個寄存器及其配置。

接下來在i2c1節(jié)點下添加ap3216節(jié)點:

編譯設(shè)備樹,傳到開發(fā)板上,重啟。此時我們系統(tǒng)的I2C設(shè)備有:

可見,新增的AP3216 I2C設(shè)備名就是我們設(shè)備樹里設(shè)置的。

下面編寫AP3216驅(qū)動(以下代碼來源于網(wǎng)絡(luò)):

ap3216.c:

#include  #include  #include  #include  #include  #include  #include  #include  #include  #include  #include  #include  #include  #include  #include  #include  #include  #include "ap3216creg.h" /***************************************************************
文件名  : ap3216c.c
描述     : AP3216C驅(qū)動程序
***************************************************************/ #define AP3216C_CNT 1 #define AP3216C_NAME "ap3216c" struct ap3216c_dev { dev_t devid; /* 設(shè)備號   */ struct cdev cdev; /* cdev  */ struct class *class; /* 類   */ struct device *device; /* 設(shè)備   */ struct device_node *nd; /* 設(shè)備節(jié)點 */ int major; /* 主設(shè)備號 */ void *private_data; /* 私有數(shù)據(jù) */ unsigned short ir, als, ps; /* 三個光傳感器數(shù)據(jù) */ }; static struct ap3216c_dev ap3216cdev; /*
 * @description : 從ap3216c讀取多個寄存器數(shù)據(jù)
 * @param - dev:  ap3216c設(shè)備
 * @param - reg:  要讀取的寄存器首地址
 * @param - val:  讀取到的數(shù)據(jù)
 * @param - len:  要讀取的數(shù)據(jù)長度
 * @return   : 操作結(jié)果
 */ static int ap3216c_read_regs(struct ap3216c_dev *dev, u8 reg, void *val, int len) { int ret; struct i2c_msg msg[2]; struct i2c_client *client = (struct i2c_client *)dev->private_data; /* msg[0]為發(fā)送要讀取的首地址 */ msg[0].addr = client->addr; /* ap3216c地址 */ msg[0].flags = 0; /* 標記為發(fā)送數(shù)據(jù) */ msg[0].buf = ® /* 讀取的首地址 */ msg[0].len = 1; /* reg長度*/ /* msg[1]讀取數(shù)據(jù) */ msg[1].addr = client->addr; /* ap3216c地址 */ msg[1].flags = I2C_M_RD; /* 標記為讀取數(shù)據(jù)*/ msg[1].buf = val; /* 讀取數(shù)據(jù)緩沖區(qū) */ msg[1].len = len; /* 要讀取的數(shù)據(jù)長度*/ ret = i2c_transfer(client->adapter, msg, 2); if(ret == 2) {
  ret = 0;
 } else {
  printk("i2c rd failed=%d reg=%06x len=%d\n",ret, reg, len);
  ret = -EREMOTEIO;
 } return ret;
} /*
 * @description : 向ap3216c多個寄存器寫入數(shù)據(jù)
 * @param - dev:  ap3216c設(shè)備
 * @param - reg:  要寫入的寄存器首地址
 * @param - val:  要寫入的數(shù)據(jù)緩沖區(qū)
 * @param - len:  要寫入的數(shù)據(jù)長度
 * @return    :   操作結(jié)果
 */ static s32 ap3216c_write_regs(struct ap3216c_dev *dev, u8 reg, u8 *buf, u8 len) {
 u8 b[256]; struct i2c_msg msg; struct i2c_client *client = (struct i2c_client *)dev->private_data; b[0] = reg; /* 寄存器首地址 */ memcpy(&b[1],buf,len); /* 將要寫入的數(shù)據(jù)拷貝到數(shù)組b里面 */ msg.addr = client->addr; /* ap3216c地址 */ msg.flags = 0; /* 標記為寫數(shù)據(jù) */ msg.buf = b; /* 要寫入的數(shù)據(jù)緩沖區(qū) */ msg.len = len + 1; /* 要寫入的數(shù)據(jù)長度 */ return i2c_transfer(client->adapter, &msg, 1);
} /*
 * @description : 讀取ap3216c指定寄存器值,讀取一個寄存器
 * @param - dev:  ap3216c設(shè)備
 * @param - reg:  要讀取的寄存器
 * @return    :   讀取到的寄存器值
 */ static unsigned char ap3216c_read_reg(struct ap3216c_dev *dev, u8 reg) {
 u8 data = 0;
 
 ap3216c_read_regs(dev, reg, &data, 1); return data; #if 0 struct i2c_client *client = (struct i2c_client *)dev->private_data; return i2c_smbus_read_byte_data(client, reg); #endif } /*
 * @description : 向ap3216c指定寄存器寫入指定的值,寫一個寄存器
 * @param - dev:  ap3216c設(shè)備
 * @param - reg:  要寫的寄存器
 * @param - data: 要寫入的值
 * @return   :    無
 */ static void ap3216c_write_reg(struct ap3216c_dev *dev, u8 reg, u8 data) {
 u8 buf = 0;
 buf = data;
 ap3216c_write_regs(dev, reg, &buf, 1);
} /*
 * @description : 讀取AP3216C的數(shù)據(jù),讀取原始數(shù)據(jù),包括ALS,PS和IR, 注意!
 *    : 如果同時打開ALS,IR+PS的話兩次數(shù)據(jù)讀取的時間間隔要大于112.5ms
 * @param - ir : ir數(shù)據(jù)
 * @param - ps  : ps數(shù)據(jù)
 * @param - ps  : als數(shù)據(jù) 
 * @return   : 無。
 */ void ap3216c_readdata(struct ap3216c_dev *dev) { unsigned char i =0; unsigned char buf[6]; /* 循環(huán)讀取所有傳感器數(shù)據(jù) */ for(i = 0; i < 6; i++) 
    {
        buf[i] = ap3216c_read_reg(dev, AP3216C_IRDATALOW + i); 
    } if(buf[0] & 0X80) /* IR_OF位為1,則數(shù)據(jù)無效 */ dev->ir = 0; else /* 讀取IR傳感器的數(shù)據(jù)     */ dev->ir = ((unsigned short)buf[1] << 2) | (buf[0] & 0X03);    
 
 dev->als = ((unsigned short)buf[3] << 8) | buf[2]; /* 讀取ALS傳感器的數(shù)據(jù)     */ if(buf[4] & 0x40) /* IR_OF位為1,則數(shù)據(jù)無效    */ dev->ps = 0; else /* 讀取PS傳感器的數(shù)據(jù)    */ dev->ps = ((unsigned short)(buf[5] & 0X3F) << 4) | (buf[4] & 0X0F); 
} /*
 * @description  : 打開設(shè)備
 * @param - inode  : 傳遞給驅(qū)動的inode
 * @param - filp  : 設(shè)備文件,file結(jié)構(gòu)體有個叫做private_data的成員變量
 *        一般在open的時候?qū)rivate_data指向設(shè)備結(jié)構(gòu)體。
 * @return    : 0 成功;其他 失敗
 */ static int ap3216c_open(struct inode *inode, struct file *filp) {
 filp->private_data = &ap3216cdev; /* 初始化AP3216C */ ap3216c_write_reg(&ap3216cdev, AP3216C_SYSTEMCONG, 0x04); /* 復(fù)位AP3216C    */ mdelay(50); /* AP3216C復(fù)位最少10ms  */ ap3216c_write_reg(&ap3216cdev, AP3216C_SYSTEMCONG, 0X03); /* 開啟ALS、PS+IR   */ return 0;
} /*
 * @description  : 從設(shè)備讀取數(shù)據(jù) 
 * @param - filp  : 要打開的設(shè)備文件(文件描述符)
 * @param - buf  : 返回給用戶空間的數(shù)據(jù)緩沖區(qū)
 * @param - cnt  : 要讀取的數(shù)據(jù)長度
 * @param - offt  : 相對于文件首地址的偏移
 * @return    : 讀取的字節(jié)數(shù),如果為負值,表示讀取失敗
 */ static ssize_t ap3216c_read(struct file *filp, char __user *buf, size_t cnt, loff_t *off) {
 short data[3]; long err = 0; struct ap3216c_dev *dev = (struct ap3216c_dev *)filp->private_data; ap3216c_readdata(dev);
 
 data[0] = dev->ir;
 data[1] = dev->als;
 data[2] = dev->ps;
 err = copy_to_user(buf, data, sizeof(data)); return 0;
} /*
 * @description  : 關(guān)閉/釋放設(shè)備
 * @param - filp  : 要關(guān)閉的設(shè)備文件(文件描述符)
 * @return    : 0 成功;其他 失敗
 */ static int ap3216c_release(struct inode *inode, struct file *filp) { return 0;
} /* AP3216C操作函數(shù) */ static const struct file_operations ap3216c_ops = { .owner = THIS_MODULE,
 .open = ap3216c_open,
 .read = ap3216c_read,
 .release = ap3216c_release,
}; /*
  * @description     : i2c驅(qū)動的probe函數(shù),當驅(qū)動與
  *                    設(shè)備匹配以后此函數(shù)就會執(zhí)行
  * @param - client  : i2c設(shè)備
  * @param - id      : i2c設(shè)備ID
  * @return          : 0,成功;其他負值,失敗
  */ static int ap3216c_probe(struct i2c_client *client, const struct i2c_device_id *id) { /* 1、構(gòu)建設(shè)備號 */ if (ap3216cdev.major) {
  ap3216cdev.devid = MKDEV(ap3216cdev.major, 0);
  register_chrdev_region(ap3216cdev.devid, AP3216C_CNT, AP3216C_NAME);
 } else {
  alloc_chrdev_region(&ap3216cdev.devid, 0, AP3216C_CNT, AP3216C_NAME);
  ap3216cdev.major = MAJOR(ap3216cdev.devid);
 } /* 2、注冊設(shè)備 */ cdev_init(&ap3216cdev.cdev, &ap3216c_ops);
 cdev_add(&ap3216cdev.cdev, ap3216cdev.devid, AP3216C_CNT); /* 3、創(chuàng)建類 */ ap3216cdev.class = class_create(THIS_MODULE, AP3216C_NAME); if (IS_ERR(ap3216cdev.class)) { return PTR_ERR(ap3216cdev.class);
 } /* 4、創(chuàng)建設(shè)備 */ ap3216cdev.device = device_create(ap3216cdev.class, NULL, ap3216cdev.devid, NULL, AP3216C_NAME); if (IS_ERR(ap3216cdev.device)) { return PTR_ERR(ap3216cdev.device);
 }
 
 ap3216cdev.private_data = client; return 0;
} /*
 * @description     : i2c驅(qū)動的remove函數(shù),移除i2c驅(qū)動的時候此函數(shù)會執(zhí)行
 * @param - client  : i2c設(shè)備
 * @return          : 0,成功;其他負值,失敗
 */ static int ap3216c_remove(struct i2c_client *client) { /* 刪除設(shè)備 */ cdev_del(&ap3216cdev.cdev);
 unregister_chrdev_region(ap3216cdev.devid, AP3216C_CNT); /* 注銷掉類和設(shè)備 */ device_destroy(ap3216cdev.class, ap3216cdev.devid);
 class_destroy(ap3216cdev.class); return 0;
} /* 傳統(tǒng)匹配方式ID列表 */ static const struct i2c_device_id ap3216c_id[] = { {"iot,ap3216c", 0},  
 {}
}; /* 設(shè)備樹匹配列表 */ static const struct of_device_id ap3216c_of_match[] = { { .compatible = "iot,ap3216c" },
 { /* Sentinel */ }
}; /* i2c驅(qū)動結(jié)構(gòu)體 */ static struct i2c_driver ap3216c_driver = { .probe = ap3216c_probe,
 .remove = ap3216c_remove,
 .driver = {
   .owner = THIS_MODULE,
      .name = "ap3216c",
      .of_match_table = ap3216c_of_match, 
     },
 .id_table = ap3216c_id,
}; /*
 * @description : 驅(qū)動入口函數(shù)
 * @param   : 無
 * @return   : 無
 */ static int __init ap3216c_init(void) { int ret = 0;
 
 ret = i2c_add_driver(&ap3216c_driver); return ret;
} /*
 * @description : 驅(qū)動出口函數(shù)
 * @param   : 無
 * @return   : 無
 */ static void __exit ap3216c_exit(void) {
 i2c_del_driver(&ap3216c_driver);
} /* module_i2c_driver(ap3216c_driver) */ module_init(ap3216c_init);
module_exit(ap3216c_exit);
MODULE_LICENSE("GPL");
MODULE_AUTHOR("pjw");

驅(qū)動詳解可查閱注釋及配合上訴的I2C驅(qū)動框架的框圖及數(shù)據(jù)手冊理解。

ap3216creg.h:

#ifndef AP3216C_H #define AP3216C_H /***************************************************************
文件名  : ap3216creg.h
描述     : AP3216C寄存器地址描述頭文件
***************************************************************/ #define AP3216C_ADDR     0X1E /* AP3216C器件地址  */ /* AP3316C寄存器 */ #define AP3216C_SYSTEMCONG 0x00 /* 配置寄存器       */ #define AP3216C_INTSTATUS 0X01 /* 中斷狀態(tài)寄存器   */ #define AP3216C_INTCLEAR 0X02 /* 中斷清除寄存器   */ #define AP3216C_IRDATALOW 0x0A /* IR數(shù)據(jù)低字節(jié)     */ #define AP3216C_IRDATAHIGH 0x0B /* IR數(shù)據(jù)高字節(jié)     */ #define AP3216C_ALSDATALOW 0x0C /* ALS數(shù)據(jù)低字節(jié)    */ #define AP3216C_ALSDATAHIGH 0X0D /* ALS數(shù)據(jù)高字節(jié)    */ #define AP3216C_PSDATALOW 0X0E /* PS數(shù)據(jù)低字節(jié)     */ #define AP3216C_PSDATAHIGH 0X0F /* PS數(shù)據(jù)高字節(jié)     */ #endif 

ap3216應(yīng)用:

ap3216cApp.c:

#include "stdio.h" #include "unistd.h" #include "sys/types.h" #include "sys/stat.h" #include "sys/ioctl.h" #include "fcntl.h" #include "stdlib.h" #include "string.h" #include  #include  #include  #include  #include  /***************************************************************
文件名  : ap3216cApp.c
描述     : ap3216c設(shè)備測試APP。
使用方法  :./ap3216cApp /dev/ap3216c
***************************************************************/ /*
 * @description  : main主程序
 * @param - argc  : argv數(shù)組元素個數(shù)
 * @param - argv  : 具體參數(shù)
 * @return    : 0 成功;其他 失敗
 */ int main(int argc, char *argv[]) { int fd; char *filename; unsigned short databuf[3]; unsigned short ir, als, ps; int ret = 0; if (argc != 2) { printf("Error Usage!\r\n"); return -1;
 }
 
 filename = argv[1];
 fd = open(filename, O_RDWR); if(fd < 0) { printf("can't open file %s\r\n", filename); return -1;
 } while (1) {
  ret = read(fd, databuf, sizeof(databuf)); if(ret == 0) { /* 數(shù)據(jù)讀取成功 */ ir =  databuf[0]; /* ir傳感器數(shù)據(jù) */ als = databuf[1]; /* als傳感器數(shù)據(jù) */ ps =  databuf[2]; /* ps傳感器數(shù)據(jù) */ printf("ir = %d, als = %d, ps = %d\r\n", ir, als, ps);
  }
  usleep(200000); /*100ms */ }
 close(fd); /* 關(guān)閉文件 */ return 0;
}

編寫Makefile,從之前的文章=======拷貝過來修改:

KERN_DIR = /home/book/100ask_imx6ull-sdk/Linux-4.9.88

all:
 make -C $(KERN_DIR) M=`pwd` modules  $(CROSS_COMPILE)gcc -o ap3216cApp ap3216cApp.c  clean:
 make -C $(KERN_DIR) M=`pwd` modules clean
 rm -rf modules.order
 rm -f ap3216cApp  # 參考內(nèi)核源碼drivers/char/ipmi/Makefile # 要想把a.c, b.c編譯成ab.ko, 可以這樣指定: # ab-y := a.o b.o # obj-m += ab.o obj-m += ap3216.o
			

編譯得到ap3216.ko及ap3216cApp,傳到板子上運行:

以上就是本次的實驗分享,如果文章對你有幫助,歡迎轉(zhuǎn)發(fā),謝謝!


免責(zé)聲明:本文內(nèi)容由21ic獲得授權(quán)后發(fā)布,版權(quán)歸原作者所有,本平臺僅提供信息存儲服務(wù)。文章僅代表作者個人觀點,不代表本平臺立場,如有問題,請聯(lián)系我們,謝謝!

本站聲明: 本文章由作者或相關(guān)機構(gòu)授權(quán)發(fā)布,目的在于傳遞更多信息,并不代表本站贊同其觀點,本站亦不保證或承諾內(nèi)容真實性等。需要轉(zhuǎn)載請聯(lián)系該專欄作者,如若文章內(nèi)容侵犯您的權(quán)益,請及時聯(lián)系本站刪除。
換一批
延伸閱讀

9月2日消息,不造車的華為或?qū)⒋呱龈蟮莫毥谦F公司,隨著阿維塔和賽力斯的入局,華為引望愈發(fā)顯得引人矚目。

關(guān)鍵字: 阿維塔 塞力斯 華為

加利福尼亞州圣克拉拉縣2024年8月30日 /美通社/ -- 數(shù)字化轉(zhuǎn)型技術(shù)解決方案公司Trianz今天宣布,該公司與Amazon Web Services (AWS)簽訂了...

關(guān)鍵字: AWS AN BSP 數(shù)字化

倫敦2024年8月29日 /美通社/ -- 英國汽車技術(shù)公司SODA.Auto推出其旗艦產(chǎn)品SODA V,這是全球首款涵蓋汽車工程師從創(chuàng)意到認證的所有需求的工具,可用于創(chuàng)建軟件定義汽車。 SODA V工具的開發(fā)耗時1.5...

關(guān)鍵字: 汽車 人工智能 智能驅(qū)動 BSP

北京2024年8月28日 /美通社/ -- 越來越多用戶希望企業(yè)業(yè)務(wù)能7×24不間斷運行,同時企業(yè)卻面臨越來越多業(yè)務(wù)中斷的風(fēng)險,如企業(yè)系統(tǒng)復(fù)雜性的增加,頻繁的功能更新和發(fā)布等。如何確保業(yè)務(wù)連續(xù)性,提升韌性,成...

關(guān)鍵字: 亞馬遜 解密 控制平面 BSP

8月30日消息,據(jù)媒體報道,騰訊和網(wǎng)易近期正在縮減他們對日本游戲市場的投資。

關(guān)鍵字: 騰訊 編碼器 CPU

8月28日消息,今天上午,2024中國國際大數(shù)據(jù)產(chǎn)業(yè)博覽會開幕式在貴陽舉行,華為董事、質(zhì)量流程IT總裁陶景文發(fā)表了演講。

關(guān)鍵字: 華為 12nm EDA 半導(dǎo)體

8月28日消息,在2024中國國際大數(shù)據(jù)產(chǎn)業(yè)博覽會上,華為常務(wù)董事、華為云CEO張平安發(fā)表演講稱,數(shù)字世界的話語權(quán)最終是由生態(tài)的繁榮決定的。

關(guān)鍵字: 華為 12nm 手機 衛(wèi)星通信

要點: 有效應(yīng)對環(huán)境變化,經(jīng)營業(yè)績穩(wěn)中有升 落實提質(zhì)增效舉措,毛利潤率延續(xù)升勢 戰(zhàn)略布局成效顯著,戰(zhàn)新業(yè)務(wù)引領(lǐng)增長 以科技創(chuàng)新為引領(lǐng),提升企業(yè)核心競爭力 堅持高質(zhì)量發(fā)展策略,塑強核心競爭優(yōu)勢...

關(guān)鍵字: 通信 BSP 電信運營商 數(shù)字經(jīng)濟

北京2024年8月27日 /美通社/ -- 8月21日,由中央廣播電視總臺與中國電影電視技術(shù)學(xué)會聯(lián)合牽頭組建的NVI技術(shù)創(chuàng)新聯(lián)盟在BIRTV2024超高清全產(chǎn)業(yè)鏈發(fā)展研討會上宣布正式成立。 活動現(xiàn)場 NVI技術(shù)創(chuàng)新聯(lián)...

關(guān)鍵字: VI 傳輸協(xié)議 音頻 BSP

北京2024年8月27日 /美通社/ -- 在8月23日舉辦的2024年長三角生態(tài)綠色一體化發(fā)展示范區(qū)聯(lián)合招商會上,軟通動力信息技術(shù)(集團)股份有限公司(以下簡稱"軟通動力")與長三角投資(上海)有限...

關(guān)鍵字: BSP 信息技術(shù)
關(guān)閉
關(guān)閉