久久国产成人av_抖音国产毛片_a片网站免费观看_A片无码播放手机在线观看,色五月在线观看,亚洲精品m在线观看,女人自慰的免费网址,悠悠在线观看精品视频,一级日本片免费的,亚洲精品久,国产精品成人久久久久久久

分享

實踐 | 基于Linux的AP3216三合一整合型光感測器實驗分享

 嵌入式大雜燴 2021-08-20

1,、前言

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

2,、AP3216簡介

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

這個芯片設(shè)計的用途是給手機(jī)之類的使用,比如:返回當(dāng)前環(huán)境光強(qiáng)以便調(diào)整屏幕亮度,;用戶接聽電話時,,將手機(jī)放置在耳邊后,自動關(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é)點(diǎn)時,,我們系統(tǒng)的I2C設(shè)備如:

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

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

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

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

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

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

ap3216.c:

#include <linux/types.h>
#include <linux/kernel.h>
#include <linux/delay.h>
#include <linux/ide.h>
#include <linux/init.h>
#include <linux/module.h>
#include <linux/errno.h>
#include <linux/gpio.h>
#include <linux/cdev.h>
#include <linux/device.h>
#include <linux/of_gpio.h>
#include <linux/semaphore.h>
#include <linux/timer.h>
#include <linux/i2c.h>
#include <asm/mach/map.h>
#include <asm/uaccess.h>
#include <asm/io.h>
#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é)點(diǎn) */
 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;     /* 標(biāo)記為發(fā)送數(shù)據(jù) */
 msg[0].buf = &reg;     /* 讀取的首地址 */
 msg[0].len = 1;      /* reg長度*/
 
 /* msg[1]讀取數(shù)據(jù) */
 msg[1].addr = client->addr;   /* ap3216c地址 */
 msg[1].flags = I2C_M_RD;   /* 標(biāo)記為讀取數(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;    /* 標(biāo)記為寫數(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ù),,如果為負(fù)值,,表示讀取失敗
 */

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ù),當(dāng)驅(qū)動與
  *                    設(shè)備匹配以后此函數(shù)就會執(zhí)行
  * @param - client  : i2c設(shè)備
  * @param - id      : i2c設(shè)備ID
  * @return          : 0,,成功;其他負(fù)值,失敗
  */

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_MODULEAP3216C_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,,成功;其他負(fù)值,失敗
 */

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 <poll.h>
#include <sys/select.h>
#include <sys/time.h>
#include <signal.h>
#include <fcntl.h>
/***************************************************************
文件名  : 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(bǔ).c, b.c編譯成ab.ko, 可以這樣指定:
# ab-y := a.o b.o
# obj-m += ab.o

obj-m += ap3216.o

編譯得到ap3216.ko及ap3216cApp,,傳到板子上運(yùn)行:

    轉(zhuǎn)藏 分享 獻(xiàn)花(0

    0條評論

    發(fā)表

    請遵守用戶 評論公約

    類似文章 更多