S3C2410的中斷過(guò)程中保存和還原現場(chǎng)問(wèn)題分析
在我的一個(gè)中斷處理例程中有一下一段:
本文引用地址:http://dyxdggzs.com/article/201611/340835.htmsave_flags(flags);
cli();
set_gpio_mode_user(k->gpio_port, GPIO_MODE_IN);
up = read_gpio_bit(k->gpio_port);
set_external_irq(k->irq_no, EXT_BOTH_EDGES, GPIO_PULLUP_DIS);
restore_flags(flags);
我有若干問(wèn)題問(wèn)各位大蝦:
1.關(guān)于save_flags和restore_flags的源代碼的閱讀,可以知道他們的作用是保存和還原現場(chǎng)。但是具體在什么時(shí)候應該保存,什么時(shí)候還原現場(chǎng)。
2.關(guān)于cli(),他是作為關(guān)中斷而起作用的,但是在后面為什么會(huì )沒(méi)有sti()進(jìn)行開(kāi)中斷。
3.在這里,set_external_irq(tmp->buttons_int_irq, EXT_BOTH_EDGES, GPIO_PULLUP_DIS);難道可以起到開(kāi)中斷的作用,因為前面有cli的操作。
說(shuō)說(shuō)我的看法:
使用save_flags()等的原因是:如果在這個(gè)函數之前已經(jīng)禁止了中斷,那么就會(huì )出現一定的危險,因為在以后調用開(kāi)中斷的時(shí)候,它會(huì )無(wú)條件地激活中斷,所以需要一種機制把中斷恢復到以前的狀態(tài)而不是簡(jiǎn)單地禁止或激活.
后面之所以沒(méi)有sti()是因為后面調用了restore_flags(),就恢復到了以前的狀態(tài)了!
S3C2410中的cpsr的低八位可以實(shí)現中斷的開(kāi)啟和關(guān)閉的作用.
原先一次save_flags()實(shí)際上是保存了原來(lái)的CPSR的數值
而后cli關(guān)中斷
最后的restore_flags()就是開(kāi)中斷,因為我們保存了原先的flags.
類(lèi)UNIX系統一直采用cli和sti函數來(lái)禁用和啟用中斷。而在現代的Linux系統中卻不鼓勵直接使用它們。
如果必須禁用中斷,那么最好使用下列調用:
unsigned long flags;
save_flags(flags);
cli();
/*下面的代碼在中斷被禁用的情況下運行*/
restore_flags(flags);
其中save_flags是一個(gè)宏,用于保存標志的參數flags被直接傳入,不帶&操作符。
另外中斷處理的順序一般是:
1. 保存現場(chǎng)
2. 關(guān)中斷
3. 執行中斷處理Makefile文件:
4. 開(kāi)中斷并恢復現場(chǎng)
最近在寫(xiě)linux驅動(dòng).遇到這么一個(gè)問(wèn)題:
: 有這么一個(gè)函數 __raw_readl. 我查了他的原型是:
: #define __raw_readl(a) (__chk_io_ptr(a), *(volatile unsigned int
: __force *)(a))
: 而__chk_io_ptr(x)的原型是
: #ifdef __CHECKER__
: extern void __chk_io_ptr(void __iomem *);
: #else
: # define __chk_io_ptr(x) (void)0
: #endif
: __raw_readl(23)展開(kāi)以后就是((void)0,*(volatile unsigned int _force *)(23)).
: 請問(wèn)上面的展開(kāi)式式什么意思???我所不理解的是(xx, xxxxxx)這樣的語(yǔ)法是什么意思?
只有后面這個(gè)表達式是有效表達式
上面等同,*(volatile unsigned int _force *)(23);
評論