chinese直男口爆体育生外卖, 99久久er热在这里只有精品99, 又色又爽又黄18禁美女裸身无遮挡, gogogo高清免费观看日本电视,私密按摩师高清版在线,人妻视频毛茸茸,91论坛 兴趣闲谈,欧美 亚洲 精品 8区,国产精品久久久久精品免费

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

完善資料讓更多小伙伴認識你,還能領取20積分哦,立即完善>

3天內不再提示

Linux驅動開發(fā)-編寫(EEPROM)AT24C02驅動

DS小龍哥-嵌入式技術 ? 2022-09-17 15:29 ? 次閱讀
加入交流群
微信小助手二維碼

掃碼添加小助手

加入工程師交流群

【摘要】 AT24C02是IIC接口的EEPROM存儲芯片,這顆芯片非常經典,百度搜索可以找到非常多的資料,大多都是51、STM32單片機的示例代碼,大多采用模擬時序、裸機系統(tǒng)運行。當前文章介紹在Linux系統(tǒng)里如何編寫AT24C02的驅動,并且在應用層完成驅動讀寫測試,將AT24C02的存儲空間映射成文件,在應用層,用戶可以直接將AT24C02當做一個普通文件的形式進行讀寫,偏移文件指針;在Linux內核

1. 前言

AT24C02是IIC接口的EEPROM存儲芯片,這顆芯片非常經典,百度搜索可以找到非常多的資料,大多都是51、STM32單片機的示例代碼,大多采用模擬時序、裸機系統(tǒng)運行。當前文章介紹在Linux系統(tǒng)里如何編寫AT24C02的驅動,并且在應用層完成驅動讀寫測試,將AT24C02的存儲空間映射成文件,在應用層,用戶可以直接將AT24C02當做一個普通文件的形式進行讀寫,偏移文件指針;在Linux內核里有一套標準的IIC子系統(tǒng)框架專門讀寫IIC接口設備,采用平臺設備模型框架,編寫驅動非常方便。

當前開發(fā)板采用友善之臂的Tiny4412,CPU三星的EXYNOS4412,4412是三星的第一款四核處理器,主頻是1.5GHZ,穩(wěn)定頻率是1.4GHZ。

2. 硬件原理圖

當前的開發(fā)板上自帶了一顆EEPROM存儲芯片(具體型號是24AA025E48,代碼與AT24C02一樣的),原理圖如下:

image-20220104085514044image-20220104085817035

自帶的內核里沒有內置EEPROM的驅動:

image-20220104085843239

存儲芯片的數據手冊介紹:

image-20220104090012199

設備地址:

image-20220104090059967

寫字節(jié)、頁寫時序:

image-20220104090126403

讀數據時序:

image-20220104090200161

3. 示例代碼

3.1 EEPROM驅動端代碼

#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
static struct work_struct work;
static struct i2c_client *eeprom_client;

#define MAX_SIZE 255  //EEPROM大小
#define EEPROM_PAGE  16 //頁字節(jié)大小

static u8 eeprom_buff[255];
static int tiny4412_open(struct inode *inode, struct file *file)
{
	printk("tiny4412_open-->ok\n");
	return 0;
}

static ssize_t tiny4412_read(struct file *file, char __user *buf, size_t size, loff_t *seek)
{	
	unsigned long err;
	//判斷位置是否超出范圍
	if(*seek+size>MAX_SIZE)
	{
		size=MAX_SIZE-*seek;
	}
	//讀取數據
	i2c_smbus_read_i2c_block_data(eeprom_client,*seek,size,eeprom_buff);
	err=copy_to_user(buf,eeprom_buff,size);
	if(err!=0)return -1;
	*seek+=size;
	return size;
}

static ssize_t tiny4412_write(struct file *file, const char __user *buf, size_t size, loff_t *seek)
{
	size_t write_ok_cnt=0;
	unsigned long err;
	err=copy_from_user(eeprom_buff,buf,size);
	if(err!=0)return -1;
	//判斷位置是否超出范圍
	if(*seek+size>MAX_SIZE)
	{
		size=MAX_SIZE-*seek;
	}
	
	int write_byte=0;
	u8 *write_p=eeprom_buff;
	while(1)
	{
		if(size>EEPROM_PAGE)
		{
			write_byte=EEPROM_PAGE;
			size-=EEPROM_PAGE;
		}
		else
		{
			write_byte=size;
		}
		
		//寫數據
		i2c_smbus_write_i2c_block_data(eeprom_client,*seek,write_byte,write_p);
		*seek+=write_byte;
		write_p+=write_byte;
		write_ok_cnt+=write_byte;  //記錄寫成功的字節(jié)數
		//等待寫完成
		msleep(10);
		if(write_byte==size)break; //寫完畢
	}
	return write_ok_cnt;
}

/*
filp:待操作的設備文件file結構體指針
off:待操作的定位偏移值(可正可負)
whence:待操作的定位起始位置
返回:返回移位后的新文件讀、寫位置,并且新位置總為正值
定位起始位置
  SEEK_SET:0,表示文件開頭
  SEEK_CUR:1,表示當前位置
  SEEK_END:2,表示文件尾
*/
static loff_t tiny4412_llseek(struct file *filp, loff_t offset, int whence)
{
	loff_t newpos = 0;
	switch(whence)
	{
		case SEEK_SET:
			newpos = offset;
			break;
		case SEEK_CUR:
			newpos = filp->f_pos + offset;
			break;
		case SEEK_END:
			if(MAX_SIZE+offset>=MAX_SIZE)
			{
				newpos=MAX_SIZE;
			}
			else
			{
				newpos = MAX_SIZE + offset;
			}
			break;
		default:
			return -EINVAL;//無效的參數
	}
	filp->f_pos = newpos;
	return newpos;
}

static int tiny4412_release(struct inode *inode, struct file *file)
{
	printk("tiny4412_release-->ok\n");
	return 0;
}

static struct file_operations fops=
{
	.open=tiny4412_open,
	.read=tiny4412_read,
	.write=tiny4412_write,
	.release=tiny4412_release,
	.llseek=tiny4412_llseek
};

/*
Linux內核管理驅動---設備號
設備號是一個unsigned int 的變量--32位。
設備號=主設備號+次設備號
*/
static struct miscdevice misc=
{
	.minor = MISC_DYNAMIC_MINOR,  /*次設備號填255表示自動分配     主設備號固定為10*/
	.name = "tiny4412_eeprom",  /*/dev目錄下文件名稱*/
	.fops = &fops, /*文件操作接口*/
};
static int tiny4412_probe(struct i2c_client *client, const struct i2c_device_id *device_id)
{
	printk("probe調用成功:%#X\n",client->addr);
	eeprom_client=client;
	
	/*1. 雜項設備的注冊函數*/
	misc_register(&misc);
	
	return 0;
}

static int tiny4412_remove(struct i2c_client *client)
{
	/*2. 雜項設備的注銷函數*/
	misc_deregister(&misc);
	printk("remove調用成功.\n");
	return 0;
}

static struct i2c_device_id id_table[]=
{
	{"tiny4412_eeprom",0},
	{}
};

static struct i2c_driver drv=
{
	.probe=tiny4412_probe,
	.remove=tiny4412_remove,
	.driver=
	{
		.name="eeprom_iic"
	},
	.id_table=id_table
};

static int __init tiny4412_drv_init(void)
{	
	/*注冊IIC驅動端*/
	i2c_add_driver(&drv);
    printk("IIC驅動端: 驅動安裝成功\n");
    return 0;
}

static void __exit tiny4412_drv_cleanup(void)
{
	/*注銷IIC驅動端*/
	i2c_del_driver(&drv);
    printk("IIC驅動端: 驅動卸載成功\n");
}

module_init(tiny4412_drv_init);    /*驅動入口--安裝驅動的時候執(zhí)行*/
module_exit(tiny4412_drv_cleanup); /*驅動出口--卸載驅動的時候執(zhí)行*/

MODULE_LICENSE("GPL");  /*設置模塊的許可證--GPL*/

3.2 EEPROM設備端代碼

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

static struct i2c_client *i2c_dev=NULL;
static struct i2c_adapter *adap=NULL;
static struct i2c_board_info info=
{
	.type="tiny4412_eeprom",
	.addr=0x50, /*設備地址*/
};

static int __init tiny4412_drv_init(void)
{	
	/*根據總線編號獲取是適配器*/
	adap=i2c_get_adapter(0);
	/*注冊IIC設備端*/
	i2c_dev=i2c_new_device(adap,&info);
    printk("IIC設備端: 驅動安裝成功\n");
    return 0;
}

static void __exit tiny4412_drv_cleanup(void)
{
	/*注銷IIC設備*/
	i2c_unregister_device(i2c_dev);
	i2c_put_adapter(adap);
    printk("IIC設備端: 驅動卸載成功\n");
}

module_init(tiny4412_drv_init);    /*驅動入口--安裝驅動的時候執(zhí)行*/
module_exit(tiny4412_drv_cleanup); /*驅動出口--卸載驅動的時候執(zhí)行*/

MODULE_LICENSE("GPL");  /*設置模塊的許可證--GPL*/

3.3 應用層測試代碼

#include 
#include 
#include 
#include 

#define EEPROM_DEV "/dev/tiny4412_eeprom"

int main(int argc,char **argv)
{
    /*1. 打開設備文件*/
    int fd=open(EEPROM_DEV,O_RDWR);
    if(fd<0)
    {
        printf("%s 設備驅動打開失敗.\n",EEPROM_DEV);
        return 0;
	}
    /*3.讀寫數據*/
	unsigned char buff[255];
	int cnt;
	int i;
	for(i=0;i<255;i++)buff[i]=i;
	cnt=write(fd,buff,255);
    printf("write成功:%d Byte\n",cnt);
	
	//偏移文件指針
	lseek(fd,SEEK_SET,0);
	
	unsigned char buff_r[255];
	cnt=read(fd,buff_r,255);
	printf("read成功:%d Byte\n",cnt);
	for(i=0;i;i++)>
聲明:本文內容及配圖由入駐作者撰寫或者入駐合作網站授權轉載。文章觀點僅代表作者本人,不代表電子發(fā)燒友網立場。文章及其配圖僅供工程師學習之用,如有內容侵權或者其他違規(guī)問題,請聯系本站處理。 舉報投訴
  • 驅動
    +關注

    關注

    12

    文章

    1924

    瀏覽量

    87876
  • EEPROM
    +關注

    關注

    9

    文章

    1103

    瀏覽量

    85023
  • AT24C02
    +關注

    關注

    0

    文章

    79

    瀏覽量

    24941
收藏 人收藏
加入交流群
微信小助手二維碼

掃碼添加小助手

加入工程師交流群

    評論

    相關推薦
    熱點推薦

    串行EEPROM P24C256H產品介紹

    P24C256H是I2C兼容的串行EEPROM(電可擦除可編程存儲器)設備。它包含一個256Kbits (32Kbytes)的內存陣列,每頁64bytes。
    的頭像 發(fā)表于 08-08 17:05 ?1521次閱讀
    串行<b class='flag-5'>EEPROM</b> P<b class='flag-5'>24C</b>256H產品介紹

    【沁恒CH585開發(fā)板免費試用體驗】I2C 讀寫EEPROM (三)

    ) { goto cmd_fail; /* EEPROM器件無應答 */ } /* 第4步:發(fā)送字節(jié)地址,24C02只有256字節(jié),因此1個字節(jié)就夠了,如果是24C04以上,那么此處需要連發(fā)多個地址
    發(fā)表于 08-05 22:25

    【沁恒CH585開發(fā)板免費試用體驗】I2C 讀寫EEPROM (二)

    寫不同的是,數據會一直發(fā),直到主機發(fā)送停止信號。 /** * @brief在EEPROM的一個寫循環(huán)中可以寫多個字節(jié),但一次寫入的字節(jié)數 * 不能超過EEPROM頁的大小,AT24C02每頁有8個字
    發(fā)表于 08-05 22:13

    如何增加CYBLE-416045-02EEPROM?

    我正在使用 CYBLE-416045-02 開發(fā)產品,但似乎最大模擬 EEPROM 是 32kBytes。 有沒有辦法進一步增加 EEPROM,因為我需要它來存儲記錄的數據。
    發(fā)表于 06-30 08:26

    單片機實例項目:AT24C02EEPROM存儲器

    單片機實例項目:AT24C02EEPROM存儲器,推薦下載!
    發(fā)表于 06-03 20:50

    itop-3568開發(fā)驅動開發(fā)指南-實驗程序的編寫

    驅動例程\\\\02。 本章實驗將編寫 Linux 下的驅動傳參實例代碼,通過“insmod”命令進行參數的傳遞,并將相應的參數打印到串口
    發(fā)表于 05-19 10:26

    【RA-Eco-RA4M2開發(fā)板評測】I2C讀取EEPROM(二)

    */ i2c_event = p_args->event; } } 主要配置I2C模式、低電平占空比、I2C尋址模式以及通信速率,最后使能I2C設備。 初始化完成后就是對
    發(fā)表于 04-28 23:14

    基于APM32F407如何制作I2C EEPROMAT24C02型號)的MDK-Keil下載算法

    基于APM32F407如何制作I2C EEPROMAT24C02型號)的Keil下載算法,這樣在我們下載代碼時可以一鍵把數據燒錄到EEPROM中。
    的頭像 發(fā)表于 04-11 11:06 ?1600次閱讀
    基于APM32F407如何制作I2<b class='flag-5'>C</b> <b class='flag-5'>EEPROM</b>(<b class='flag-5'>AT24C02</b>型號)的MDK-Keil下載算法

    迅為RK3568開發(fā)板helloworld 驅動實驗-驅動編寫

    在學習 C 語言或者其他語言的時候,我們通常是打印一句“helloworld”來開啟編程世界的大門。學習驅動程序編程亦可以如此,使用 helloworld 作為我們的第一個驅動程序。接下來開始
    發(fā)表于 04-01 15:20

    【CW32模塊使用】AT24C02-EEPROM存儲器

    可以在電腦上或專用設備上擦除已有信息,重新編程。一般用在即插即用。AT24C02是一個2K位串行CMOS E2PROM, 內部含有256個8位字節(jié),CATALYST公司的先進CMOS技術實質上減少
    的頭像 發(fā)表于 03-29 17:26 ?1246次閱讀
    【CW32模塊使用】AT<b class='flag-5'>24C02-EEPROM</b>存儲器

    STM32F407的兩個USB口能否分別實現讀卡器和讀U盤的功能以及硬件IIC讀寫AT24C02?

    需求如下: 1.OTS_FS(PA 11,PA12)實現TF卡的讀卡器功能, 2.OTS_HS(PB14,PB15)實現讀寫U盤功能 3.硬件IIC(PB8,PB9)讀寫AT24C02,硬件IIC是否有BUG? 以上是否有相關的庫例程? 謝謝!
    發(fā)表于 03-11 06:23

    迅為RK3568開發(fā)驅動指南Linux中通用SPI設備驅動

    迅為RK3568開發(fā)驅動指南Linux中通用SPI設備驅動
    的頭像 發(fā)表于 01-23 11:02 ?3250次閱讀
    迅為RK3568<b class='flag-5'>開發(fā)</b>板<b class='flag-5'>驅動</b>指南<b class='flag-5'>Linux</b>中通用SPI設備<b class='flag-5'>驅動</b>

    迅為RK3568開發(fā)板SPI驅動指南-mcp2515驅動編寫:讀寄存器函數

    迅為RK3568開發(fā)板SPI驅動指南-mcp2515驅動編寫:讀寄存器函數
    的頭像 發(fā)表于 01-20 14:43 ?1319次閱讀
    迅為RK3568<b class='flag-5'>開發(fā)</b>板SPI<b class='flag-5'>驅動</b>指南-mcp2515<b class='flag-5'>驅動</b><b class='flag-5'>編寫</b>:讀寄存器函數

    數據保存100年!納祥科技國產EEPROM芯片NX24C02A,兼容替代AT24C02

    NAXIANGTECHNOLOGY納祥科技NX24C02AEEPROMEEPROM(ElectricallyErasableProgrammablereadonlymemory),帶電
    的頭像 發(fā)表于 11-14 01:08 ?1497次閱讀
    數據保存100年!納祥科技國產<b class='flag-5'>EEPROM</b>芯片NX<b class='flag-5'>24C02</b>A,兼容替代<b class='flag-5'>AT24C02</b>

    linux內核中通用HID觸摸驅動

    linux內核中,為HID觸摸面板實現了一個通用的驅動程序,位于/drivers/hid/hid-multitouch.c文件中。hid觸摸驅動是以struct hid_driver
    的頭像 發(fā)表于 10-29 10:55 ?3114次閱讀
    <b class='flag-5'>linux</b>內核中通用HID觸摸<b class='flag-5'>驅動</b>