在介紹具體如何寫Nand Flash驅(qū)動之前,我們先要了解,大概的,整個系統(tǒng),和Nand Flash相關(guān)的部分的驅(qū)動工作流程,這樣,對于后面的驅(qū)動實現(xiàn),才能更加清楚機制,才更容易實現(xiàn),否則就是,即使寫完了代碼,也還是沒搞懂系統(tǒng)是如何工作的了。
讓我們以最常見的,Linux內(nèi)核中已經(jīng)有的三星的Nand Flash驅(qū)動,來解釋Nand Flash驅(qū)動具體流程和原理。
此處是參考2.6.29版本的Linux源碼中的\drivers\mtd\nand\s3c2410.c,以2410為例。
1. 在nand flash驅(qū)動加載后,第一步,調(diào)用對應(yīng)的init函數(shù) ---- s3c2410_nand_init: 去將nand flash驅(qū)動注冊到Linux驅(qū)動框架中。
2. 驅(qū)動本身真正的開始,是從probe函數(shù): s3c2410_nand_probe->s3c24xx_nand_probe,
在probe過程中:
clk_enable //打開nand flash控制器的clock時鐘,
request_mem_region //去申請驅(qū)動所需要的一些內(nèi)存等相關(guān)資源。
s3c2410_nand_inithw //去初始化硬件相關(guān)的部分,主要是關(guān)于時鐘頻率的計算,以及啟用nand flash控制器,使得硬件初始化好了,后面才能正常工作。
3. 需要多解釋一下的,是這部分代碼:
for (setno = 0; setno < nr_sets; setno++, nmtd++) {
pr_debug("initialising set %d (%p, info %p)\n", setno, nmtd, info);
/*調(diào)用init chip去掛載你的nand驅(qū)動的底層函數(shù)到"nand flash的結(jié)構(gòu)體"中,以及設(shè)置對應(yīng)的"ecc mode",掛載ecc相關(guān)的函數(shù) */
s3c2410_nand_init_chip(info, nmtd, sets);
/* scan_ident,掃描nand 設(shè)備,設(shè)置nand flash的默認函數(shù),獲得物理設(shè)備的具體型號以及對應(yīng)各個特性參數(shù),這部分算出來的一些值,對于nand flash來說,是最主要的參數(shù),比如nand flash的芯片的大小,塊大小,頁大小等。 */
nmtd->scan_res = nand_scan_ident(&nmtd->mtd, (sets) ? sets->nr_chips : 1);
if (nmtd->scan_res == 0) {
s3c2410_nand_update_chip(info, nmtd);
/*掃描的后一階段,經(jīng)過前面的scan_ident,我們已經(jīng)獲得對應(yīng)nand flash的硬件的各個參數(shù),
*然后就可以在scan tail中,根據(jù)這些參數(shù),去設(shè)置其他一些重要參數(shù),尤其是ecc的layout,即ecc是如何在oob中擺放的,
*最后,再去進行一些初始化操作,主要是根據(jù)你的驅(qū)動,如果沒有實現(xiàn)一些函數(shù)的話,那么就用系統(tǒng)默認的。 */
nand_scan_tail(&nmtd->mtd);
/*add partion,根據(jù)你的nand flash的分區(qū)設(shè)置,去分區(qū) */
s3c2410_nand_add_partition(info, nmtd, sets);
}
if (sets != NULL)
sets++;
}
4. 等所有的參數(shù)都計算好了,函數(shù)都掛載完畢,系統(tǒng)就可以正常工作了。
上層訪問你的nand falsh中的數(shù)據(jù)的時候,通過MTD層,一層層調(diào)用,最后調(diào)用到你所實現(xiàn)的那些底層訪問硬件數(shù)據(jù)/緩存的函數(shù)中。
|