0
  • 聊天消息
  • 系統(tǒng)消息
  • 評(píng)論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會(huì)員中心
創(chuàng)作中心

完善資料讓更多小伙伴認(rèn)識(shí)你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

Linux驅(qū)動(dòng)開發(fā)-編寫MMA7660三軸加速度傳感器

DS小龍哥-嵌入式技術(shù) ? 2022-09-17 15:22 ? 次閱讀

【摘要】 通過MMA7660可以做出很多項(xiàng)目: 比如: 老人防跌倒手環(huán)、自行車自動(dòng)剎車燈,智能鬧鐘,烤火爐跌倒自動(dòng)斷電、運(yùn)動(dòng)手環(huán)等等。 這篇文章就介紹如何在Linux下編寫MMA7660三軸加速度芯片的驅(qū)動(dòng),讀取當(dāng)前芯片的方向姿態(tài),得到X,Y,Z三個(gè)軸的數(shù)據(jù)。MMA7660是IIC接口的,當(dāng)前驅(qū)動(dòng)就采用標(biāo)準(zhǔn)的IIC子系統(tǒng)編寫驅(qū)動(dòng),使用字符設(shè)備框架將得到的數(shù)據(jù)上傳遞給應(yīng)用層。

1. MMA7660芯片介紹

MMA7660FC 是 ± 1.5 克的三軸數(shù)字輸出、超低功率、緊湊型電容式微電機(jī)的三軸加速度計(jì),是非常低功耗,小型容性 MEMS傳感器。具有低通濾波器,用于偏移和增益誤差補(bǔ)償, 以及用戶可配置的轉(zhuǎn)換成 6 位分辨率,用戶可配置輸出速率等功能。MMA7660芯片可以通過中斷引腳(INT)向外通知傳感器數(shù)據(jù)變化、方向、姿態(tài)識(shí)別等信息。模擬工作電壓范圍是 2.4V 至 3.6V,數(shù)字工作電壓范圍是 1.71V 到 3.6V 。常用在手機(jī)、掌上電腦、車載導(dǎo)航,便攜式電腦的防盜,自動(dòng)自行車剎車燈、運(yùn)動(dòng)檢測(cè)手環(huán)、數(shù)碼機(jī)、自動(dòng)叫醒鬧鐘里等等。

特別是計(jì)步的功能是現(xiàn)在最常見,不管是智能手環(huán)、還是手機(jī)都帶有三軸加速度計(jì),可以記錄每天的步數(shù),計(jì)算運(yùn)動(dòng)量等?,F(xiàn)在很多的不倒翁,無人機(jī)、相機(jī)云臺(tái),很多常見的產(chǎn)品里都能看到三軸加速計(jì)的身影。

通過MMA7660可以做出很多項(xiàng)目: 比如: 老人防跌倒手環(huán)、自行車自動(dòng)剎車燈,智能鬧鐘,烤火爐跌倒自動(dòng)斷電、運(yùn)動(dòng)手環(huán)等等。

這篇文章就介紹如何在Linux下編寫MMA7660三軸加速度芯片的驅(qū)動(dòng),讀取當(dāng)前芯片的方向姿態(tài),得到X,Y,Z三個(gè)軸的數(shù)據(jù)。MMA7660是IIC接口的,當(dāng)前驅(qū)動(dòng)就采用標(biāo)準(zhǔn)的IIC子系統(tǒng)編寫驅(qū)動(dòng),使用字符設(shè)備框架將得到的數(shù)據(jù)上傳遞給應(yīng)用層。

image-20220108211859294image-20220108211941645

2. 硬件連線

當(dāng)前使用的開發(fā)板是友善之臂Tiny4412開發(fā)板,使用三星EXYNOS4412芯片,板子本身自帶了一顆MMA7660芯片,芯片的原理圖如下:

image-20220108211848037image-20220108211911111

內(nèi)核本身有MMA7660的驅(qū)動(dòng),下面是源碼的路徑:

image-20220108212008898image-20220108212025997image-20220108212036925

如果加載自己編寫的驅(qū)動(dòng),還需要去掉原來內(nèi)核自帶的驅(qū)動(dòng),不然無法匹配。

Device Drivers  --->
	<*> Hardware Monitoring support  --->  
        <*>   Freescale MMA7660 Accelerometer   (將*號(hào)去掉,編譯內(nèi)核、燒寫內(nèi)核即可)
image-20220108212133903

3. 源代碼

3.1 mma7660設(shè)備端代碼: IIC子系統(tǒng)

#include 
#include 
#include 
#include 
#include 
#include 
#include 

static struct i2c_client *i2cClient = NULL;

static unsigned short  i2c_addr_list[]= {0x4c, I2C_CLIENT_END};/*地址隊(duì)列*/

/*
1. 獲取控制器(總線)
2. 探測(cè)設(shè)備是否存在
3. 定義一個(gè)名字用于找到驅(qū)動(dòng)端
 */
static int __init mma7660_dev_init(void)
{
	/*mach-tiny4412.c*/
	struct i2c_adapter *i2c_adap=NULL;  /*獲取到的總線存放在這個(gè)結(jié)構(gòu)體*/
	struct i2c_board_info i2c_info;     /*設(shè)備描述結(jié)構(gòu)體,里面存放著設(shè)備的名字還有地址*/

	/*1. 獲取IIC控制器*/
	i2c_adap = i2c_get_adapter(3);     /*要使用IIC_3號(hào)總線*/
	if(!i2c_adap)
	{
		printk("獲取IIC控制器信息失敗!\n");
		return -1;
	}
	
	memset(&i2c_info,0,sizeof(struct i2c_board_info));      	/*清空結(jié)構(gòu)體*/
	strlcpy(i2c_info.type,"mma7660_drv",I2C_NAME_SIZE);    /*名稱的賦值*/
	i2c_info.irq=EXYNOS4_GPX3(1); /*中斷IO口*/

	/*2. 創(chuàng)建IIC設(shè)備客戶端*/
	i2cClient = i2c_new_probed_device(i2c_adap,&i2c_info,i2c_addr_list,NULL);
	if(!i2cClient)
	{
		printk("mma7660_探測(cè)地址出現(xiàn)錯(cuò)誤!!\n");
		return -1;
	}

	i2c_put_adapter(i2c_adap);/*設(shè)置模塊使用計(jì)數(shù)*/
	
	printk("mma7660_dev_init!!\n");
	return 0;
}
static void __exit mma7660_dev_exit(void)//平臺(tái)設(shè)備端的出口函數(shù)
{
	printk(" mma7660_dev_exit ok!!\n");

	/*注銷設(shè)備*/
	i2c_unregister_device(i2cClient);

	/*釋放*/
	i2c_release_client(i2cClient);
}
module_init(mma7660_dev_init);
module_exit(mma7660_dev_exit);
MODULE_LICENSE("GPL");

3.2 mma7660驅(qū)動(dòng)端代碼: IIC子系統(tǒng)

#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 

/* MMA7760 Registers */
#define MMA7660_XOUT			0x00	// 6-bit output value X
#define MMA7660_YOUT			0x01	// 6-bit output value Y
#define MMA7660_ZOUT			0x02	// 6-bit output value Z
#define MMA7660_TILT			0x03	// Tilt status
#define MMA7660_SRST			0x04	// Sampling Rate Status
#define MMA7660_SPCNT			0x05	// Sleep Count
#define MMA7660_INTSU			0x06	// Interrupt Setup
#define MMA7660_MODE			0x07	// Mode
#define MMA7660_SR				0x08	// Auto-Wake/Sleep and Debounce Filter
#define MMA7660_PDET			0x09	// Tap Detection
#define MMA7660_PD				0x0a	// Tap Debounce Count

static const struct i2c_device_id mma7660_id[] =
{
	{"mma7660_drv",0}, /*設(shè)備端的名字,0表示不需要私有數(shù)據(jù)*/
	{}
};
static u32 mma7660_irq; /*觸摸屏的中斷編號(hào)*/
static struct i2c_client *mma7660_client=NULL;
static int	last_tilt = 0;

#define __need_retry(__v)	(__v & (1 << 6))
#define __is_negative(__v)	(__v & (1 << 5))

static const char *mma7660_bafro[] = {
	"未知", "前面", "背面"
};

static const char *mma7660_pola[] = {
	"未知",
	"左面", "向右",
	"保留", "保留",
	"向下", "向上",
	"保留",
};

/*
函數(shù)功能:讀取一個(gè)字節(jié)的數(shù)據(jù)
*/
static int mma7660_read_tilt(struct i2c_client *client, int *tilt)
{
	int val;
	do {
		val = i2c_smbus_read_byte_data(client, MMA7660_TILT);
	} while (__need_retry(val));
	*tilt = (val & 0xff);
	return 0;
}
/*
函數(shù)功能: 讀取XYZ坐標(biāo)數(shù)據(jù)
*/
static int mma7660_read_xyz(struct i2c_client *client, int idx, int *xyz)
{
	int val;
	do {
		val = i2c_smbus_read_byte_data(client, idx + MMA7660_XOUT);
	} while (__need_retry(val));
	*xyz = __is_negative(val) ? (val | ~0x3f) : (val & 0x3f);
	return 0;
}
/*
工作隊(duì)列處理函數(shù)
*/
static void mma7660_worker(struct work_struct *work)
{
	int bafro, pola, shake, tap;
	int val = 0;

	mma7660_read_tilt(mma7660_client,&val);

	/* TODO: report it ? */
	bafro = val & 0x03;
	if (bafro != (last_tilt & 0x03)) {
		printk("%s\n", mma7660_bafro[bafro]);
	}

	pola = (val >> 2) & 0x07;
	if (pola != ((last_tilt >> 2) & 0x07)) {
		printk("%s\n", mma7660_pola[pola]);
	}

	shake = (val >> 5) & 0x01;
	if (shake && shake != ((last_tilt >> 5) & 0x01)) {
		printk("Shake\n");
	}

	tap = (val >> 7) & 0x01;
	if (tap && tap != ((last_tilt >> 7) & 0x01)) {
		printk("Tap\n");
	}

	/* Save current status */
	last_tilt = val;
	
	int axis[3];
	int i;
	for (i = 0; i < 3; i++)
	{
		mma7660_read_xyz(mma7660_client, i, &axis[i]);
	}
	printk("ABS_X=%d\n",axis[0]);
	printk("ABS_Y=%d\n",axis[1]);
	printk("ABS_Z=%d\n",axis[2]);
}
/*
函數(shù)功能: mma7660初始化
*/
static int mma7660_initialize(struct i2c_client *client)
{
	int val;

	/* Using test mode to probe chip */
	i2c_smbus_write_byte_data(client, MMA7660_MODE, 0x00);
	mdelay(10);
	i2c_smbus_write_byte_data(client, MMA7660_MODE, 0x04);
	mdelay(10);
	i2c_smbus_write_byte_data(client, MMA7660_XOUT, 0x3f);
	i2c_smbus_write_byte_data(client, MMA7660_YOUT, 0x01);
	i2c_smbus_write_byte_data(client, MMA7660_ZOUT, 0x15);
	val = i2c_smbus_read_byte_data(client, MMA7660_ZOUT);
	if (val != 0x15) {
		dev_err(&client->dev, "no device\n");
		return -ENODEV;
	}

	/* Goto standby mode for configuration */
	i2c_smbus_write_byte_data(client, MMA7660_MODE, 0x00);
	mdelay(10);

	/* Sample rate: 64Hz / 16Hz; Filt: 3 samples  */
	i2c_smbus_write_byte_data(client, MMA7660_SR, ((2<<5) | (1<<3) | 1));

	/* Sleep count */
	i2c_smbus_write_byte_data(client, MMA7660_SPCNT, 0xA0);

	/* Tap detect and debounce ~4ms */
	i2c_smbus_write_byte_data(client, MMA7660_PDET, 4);
	i2c_smbus_write_byte_data(client, MMA7660_PD, 15);

	/* Enable interrupt except exiting Auto-Sleep */
	i2c_smbus_write_byte_data(client, MMA7660_INTSU, 0xe7);

	/* IPP, Auto-wake, auto-sleep and standby */
	i2c_smbus_write_byte_data(client, MMA7660_MODE, 0x59);
	mdelay(10);

	/* Save current tilt status */
	mma7660_read_tilt(client, &last_tilt);

	mma7660_client = client;
	return 0;
}
/*
靜態(tài)方式初始化工作隊(duì)列
*/
DECLARE_WORK(mma7660_work,mma7660_worker);
static irqreturn_t mma7660_interrupt(int irq, void *dev_id)
{
	/*調(diào)度共享工作隊(duì)列*/
	schedule_work(&mma7660_work);
	return IRQ_HANDLED;
}
/*
匹配成功時(shí)調(diào)用
*/
static int mma7660_probe(struct i2c_client *client, const struct i2c_device_id *device_id)
{
	printk("mma7660_probe!!!\n");
	printk("驅(qū)動(dòng)端IIC匹配的地址=0x%x\n",client->addr);
	
	mma7660_client=client;
	
	/*1. 注冊(cè)中斷*/
	mma7660_irq=gpio_to_irq(client->irq);/*獲取中斷編號(hào)*/
    if(request_irq(mma7660_irq,mma7660_interrupt,IRQF_TRIGGER_FALLING,"mma7660_irq",NULL)!=0)
    {
		printk("mma7660_中斷注冊(cè)失敗!\n");
	}
	
	/*2. 初始化mma7660*/
	if(mma7660_initialize(client) < 0)
	{
		printk(" 初始化mma7660失敗!\n");
	}
    return 0;
}
static int mma7660_remove(struct i2c_client *client)
{
	free_irq(mma7660_irq,NULL);
	printk("mma7660_remove!!!\n");
	return 0;
}
struct i2c_driver i2c_drv =
{
	.driver =
	{
		.name = "mma7660",
		.owner = THIS_MODULE,
	},	
	.probe = mma7660_probe,   //探測(cè)函數(shù)
	.remove = mma7660_remove, //資源卸載
	.id_table = mma7660_id,   //里面有一個(gè)名字的參數(shù)用來匹配設(shè)備端名字
};
static int __init mma7660_drv_init(void)
{
	/*向iic總線注冊(cè)一個(gè)驅(qū)動(dòng)*/
	i2c_add_driver(&i2c_drv);
	return 0;
}

static void __exit mma7660_drv_exit(void)
{
	/*從iic總線注銷一個(gè)驅(qū)動(dòng)*/
	i2c_del_driver(&i2c_drv);
}

module_init(mma7660_drv_init);
module_exit(mma7660_drv_exit);
MODULE_LICENSE("GPL");
聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點(diǎn)僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場(chǎng)。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問題,請(qǐng)聯(lián)系本站處理。 舉報(bào)投訴
  • Linux
    +關(guān)注

    關(guān)注

    87

    文章

    11123

    瀏覽量

    207918
  • 加速度傳感器
    +關(guān)注

    關(guān)注

    12

    文章

    465

    瀏覽量

    55050
  • mma7660
    +關(guān)注

    關(guān)注

    0

    文章

    3

    瀏覽量

    7323
收藏 人收藏

    評(píng)論

    相關(guān)推薦

    國(guó)產(chǎn)士蘭微MEMS加速度傳感器的多元應(yīng)用場(chǎng)景

    國(guó)產(chǎn)士蘭微SC7A20H加速度傳感器
    的頭像 發(fā)表于 08-08 14:39 ?490次閱讀

    LIS2DE12加速度傳感器能否保存數(shù)據(jù)一次獲???當(dāng)讀取xyz加速度時(shí)獲得的是當(dāng)前實(shí)時(shí)加速度嗎?

    LIS2DE12加速度傳感器能否保存數(shù)據(jù)一次獲取,當(dāng)讀取xyz加速度時(shí)獲得的是當(dāng)前實(shí)時(shí)加速度嗎?
    發(fā)表于 04-02 06:53

    愛普生加速度計(jì)傳感器HGPM01

    HGPM01是基于高性能陀螺儀陀傳感器加速度計(jì)開發(fā)的模組,內(nèi)置3陀螺儀和3加速度計(jì)
    的頭像 發(fā)表于 03-05 14:36 ?345次閱讀
    愛普生<b class='flag-5'>三</b><b class='flag-5'>軸</b><b class='flag-5'>加速度</b>計(jì)<b class='flag-5'>傳感器</b>HGPM01

    加速度傳感器的基本力學(xué)模型是什么

    加速度傳感器的基本力學(xué)模型是一個(gè)受力物體的運(yùn)動(dòng)學(xué)和動(dòng)力學(xué)模型的組合。本文將從以下幾個(gè)方面介紹加速度傳感器的基本力學(xué)模型。 一、運(yùn)動(dòng)學(xué)模型 加速度
    的頭像 發(fā)表于 01-17 11:08 ?946次閱讀

    加速度傳感器屬于什么類型傳感器

    加速度傳感器屬于慣性傳感器類型的傳感器。慣性傳感器主要通過測(cè)量物體運(yùn)動(dòng)狀態(tài)中的加速度
    的頭像 發(fā)表于 01-12 11:12 ?716次閱讀

    加速度傳感器原理及其應(yīng)用

    常見的加速度傳感器工作原理有種:壓電原理、電容原理和微機(jī)電系統(tǒng)(MEMS)原理。 1.壓電原理:壓電加速度傳感器通過壓電材料的壓電效應(yīng)來測(cè)
    的頭像 發(fā)表于 01-05 08:39 ?1365次閱讀
    <b class='flag-5'>加速度</b><b class='flag-5'>傳感器</b>原理及其應(yīng)用

    ADXL103 CE加速度傳感器能否測(cè)豎直方向的振動(dòng)?

    ADXL 103 CE 加速度傳感器能否測(cè)豎直方向的振動(dòng)?該款傳感器標(biāo)著測(cè)試方向是水平面內(nèi)的X或Y,我發(fā)現(xiàn)豎直放置使用時(shí),零g偏移電平在3.5V,這樣能可以正常使用嗎?影響測(cè)量精度嗎
    發(fā)表于 01-01 06:56

    加速度計(jì)和加速度計(jì)的使用區(qū)別?

    有個(gè)問題請(qǐng)教一下:在靜態(tài)測(cè)量的情況下,兩加速度計(jì)在測(cè)得X、Y上的加速度后,是不是就可以根據(jù)這兩個(gè)值和重力加速度g算出Z
    發(fā)表于 12-29 06:06

    加速度傳感器在跌倒檢測(cè)中應(yīng)用說明

    電子發(fā)燒友網(wǎng)站提供《加速度傳感器在跌倒檢測(cè)中應(yīng)用說明.pdf》資料免費(fèi)下載
    發(fā)表于 11-22 14:27 ?6次下載
    <b class='flag-5'>三</b><b class='flag-5'>軸</b><b class='flag-5'>加速度</b><b class='flag-5'>傳感器</b>在跌倒檢測(cè)中應(yīng)用說明

    加速度傳感器在跌倒檢測(cè)中的應(yīng)用

    電子發(fā)燒友網(wǎng)站提供《加速度傳感器在跌倒檢測(cè)中的應(yīng)用.pdf》資料免費(fèi)下載
    發(fā)表于 11-22 10:53 ?0次下載
    <b class='flag-5'>三</b><b class='flag-5'>軸</b><b class='flag-5'>加速度</b><b class='flag-5'>傳感器</b>在跌倒檢測(cè)中的應(yīng)用

    或六加速度傳感器哪種靈敏度高一些?

    DIY平衡控制系統(tǒng),需要使用加速度傳感器,或六的均可以,哪種靈敏度高一些?
    發(fā)表于 11-10 06:46

    可以通過角速度傳感器來獲得角加速度嗎?

    我想讓單片機(jī)實(shí)現(xiàn)接受角加速度傳感器的信息,并且將獲得的信息進(jìn)?行函數(shù)運(yùn)算,我想知道有角加速度傳感器嗎?我查了一下有角速度傳?感
    發(fā)表于 11-09 08:21

    陀螺儀傳感器加速度傳感器的區(qū)別

    加速度傳感器則是測(cè)量物體加速度傳感器,其工作原理基于牛頓第二定律,即物體的加速度與所受力成正比。加速度
    的頭像 發(fā)表于 10-22 10:58 ?1397次閱讀

    加速度傳感器在跌倒檢測(cè)中的應(yīng)用

    人們?cè)絹碓綗嶂杂趯?duì)跌倒檢測(cè)以及跌倒預(yù)報(bào)儀器的研制。近年來,隨著 MEMS 加速度傳感器技術(shù)的發(fā)展,使得設(shè)計(jì)基于加速度
    發(fā)表于 09-26 08:29

    QMA8658A 6加速度傳感器概述和主要特點(diǎn)

    QMA8658A 是一款功能強(qiáng)大的6加速度傳感器,其內(nèi)置了3加速度計(jì)和3陀螺儀,能夠同時(shí)測(cè)
    發(fā)表于 09-24 14:36 ?638次閱讀
    QMA8658A 6<b class='flag-5'>軸</b><b class='flag-5'>加速度</b><b class='flag-5'>傳感器</b>概述和主要特點(diǎn)