基于S3C2440的嵌入式Linux驅(qū)動(dòng)——AT24C02(EEPROM I2C接口)驅(qū)動(dòng)解讀
本文將介紹Linux中AT24C02驅(qū)動(dòng)。AT24C02是一種EEPROM,使用I2C接口來(lái)訪問(wèn)。
在開(kāi)發(fā)板中,使用I2C控制器0和AT24C02連接,這里就不給出原理圖了,如需要,可以搜索TQ2440開(kāi)發(fā)板的原理圖。
目標(biāo)平臺(tái):TQ2440
CPU:s3c2440
內(nèi)核版本:2.6.32
本文所有的代碼均位于內(nèi)核源碼:linux/drivers/misc/eeprom/at24.c中。
staticint__initat24_init(void)
{
/*將io_limit向下圓整到最近的2的冪*/
io_limit=rounddown_pow_of_two(io_limit);
returni2c_add_driver(&at24_driver);/*i2c驅(qū)動(dòng)注冊(cè)*/
}
module_init(at24_init);
staticvoid__exitat24_exit(void)
{
i2c_del_driver(&at24_driver);
}
module_exit(at24_exit);
MODULE_DESCRIPTION("DriverformostI2CEEPROMs");
MODULE_AUTHOR("DavidBrownellandWolframSang");
MODULE_LICENSE("GPL");
注冊(cè)函數(shù)很簡(jiǎn)單。io_limit為寫(xiě)入時(shí)允許一次寫(xiě)入的最大字節(jié),該參數(shù)為驅(qū)動(dòng)模塊參數(shù),可由用戶設(shè)置,默認(rèn)值為128字節(jié)。
首先對(duì)io_limit向下圓整到最近的2的冪,接著直接調(diào)用了i2c_add_driver來(lái)注冊(cè)一個(gè)i2c驅(qū)動(dòng)。
注銷函數(shù)更簡(jiǎn)單。注銷之前注冊(cè)的i2c驅(qū)動(dòng)。
熟悉I2C驅(qū)動(dòng)架構(gòu)的可能會(huì)知道I2C驅(qū)動(dòng)的match函數(shù),該函數(shù)將使用id表(struct i2c_device_id)和i2c設(shè)備(struct i2c_client)進(jìn)行匹配,判斷是否有name字段相同,如果相同則匹配完成,即可完成設(shè)備和驅(qū)動(dòng)的綁定,接著便會(huì)調(diào)用驅(qū)動(dòng)提供的probe方法。我們來(lái)看下驅(qū)動(dòng)提供的id表。
staticstructi2c_driverat24_driver={
.driver={
.name="at24",
.owner=THIS_MODULE,
},
.probe=at24_probe,
.remove=__devexit_p(at24_remove),
.id_table=at24_ids,
};
驅(qū)動(dòng)提供的id為at24_ids,如下:
staticconststructi2c_device_idat24_ids[]={
/*needs8addressesasA0-A2areignored*/
{"24c00",AT24_DEVICE_MAGIC(128/8,AT24_FLAG_TAKE8ADDR)},
/*oldvariantscan'tbehandledwiththisgenericentry!*/
{"24c01",AT24_DEVICE_MAGIC(1024/8,0)},
{"24c02",AT24_DEVICE_MAGIC(2048/8,0)},
/*spdisa24c02inmemoryDIMMs*/
{"spd",AT24_DEVICE_MAGIC(2048/8,
AT24_FLAG_READONLY|AT24_FLAG_IRUGO)},
{"24c04",AT24_DEVICE_MAGIC(4096/8,0)},
/*24rf08quirkishandledati2c-core*/
{"24c08",AT24_DEVICE_MAGIC(8192/8,0)},
{"24c16",AT24_DEVICE_MAGIC(16384/8,0)},
{"24c32",AT24_DEVICE_MAGIC(32768/8,AT24_FLAG_ADDR16)},
{"24c64",AT24_DEVICE_MAGIC(65536/8,AT24_FLAG_ADDR16)},
{"24c128",AT24_DEVICE_MAGIC(131072/8,AT24_FLAG_ADDR16)},
{"24c256",AT24_DEVICE_MAGIC(262144/8,AT24_FLAG_ADDR16)},
{"24c512",AT24_DEVICE_MAGIC(524288/8,AT24_FLAG_ADDR16)},
{"24c1024",AT24_DEVICE_MAGIC(1048576/8,AT24_FLAG_ADDR16)},
{"at24",0},
{/*ENDOFLIST*/}
};
結(jié)構(gòu)體成員的第一個(gè)參數(shù)即為name,表示設(shè)備的名字。第二個(gè)參數(shù),在該驅(qū)動(dòng)中,為一個(gè)幻術(shù)(magic),通過(guò)AT24_DEVICE_MAGIC宏計(jì)算。
宏第一個(gè)參數(shù)為eeprom的大小,第二參數(shù)為一些標(biāo)志位。我們看下這個(gè)宏:
#defineAT24_SIZE_BYTELEN5
#defineAT24_SIZE_FLAGS8
/*createnon-zeromagicvalueforgiveneepromparameters*/
#defineAT24_DEVICE_MAGIC(_len,_flags)
((1< < 在這個(gè)表中,針對(duì)這里講解的24c02,其大小為256字節(jié),標(biāo)志位為空。 當(dāng)i2c總線完成設(shè)備驅(qū)動(dòng)綁定后,就會(huì)調(diào)用probe方法了。具體看下這個(gè)函數(shù)。 staticintat24_probe(structi2c_client*client,conststructi2c_device_id*id) { structat24_platform_datachip; boolwritable; booluse_smbus=false; structat24_data*at24; interr; unsignedi,num_addresses; kernel_ulong_tmagic; /*獲取板級(jí)設(shè)備信息*/ if(client->dev.platform_data){ chip=*(structat24_platform_data*)client->dev.platform_data; }else{ /*沒(méi)有板級(jí)設(shè)備信息,也沒(méi)有driver_data,直接出錯(cuò)*/ if(!id->driver_data){ err=-ENODEV; gotoerr_out; } magic=id->driver_data; chip.byte_len=BIT(magic&AT24_BITMASK(AT24_SIZE_BYTELEN)); magic>>=AT24_SIZE_BYTELEN; chip.flags=magic&AT24_BITMASK(AT24_SIZE_FLAGS); /* *Thisisslow,butwecan'tknowalleeproms,sowebetter *playsafe.Specifyingcustomeeprom-typesviaplatform_data *isrecommendedanyhow. */ chip.page_size=1; chip.setup=NULL; chip.context=NULL; } /*檢查參數(shù), byte_len和page_size必須為2的冪,不是則打印警告*/ if(!is_power_of_2(chip.byte_len)) dev_warn(&client->dev, "byte_lenlookssuspicious(nopowerof2)!n"); if(!is_power_of_2(chip.page_size)) dev_warn(&client->dev, "page_sizelookssuspicious(nopowerof2)!n"); /* Use I2C operations unless we're stuck with SMBus e





