亚洲欧美第一页_禁久久精品乱码_粉嫩av一区二区三区免费野_久草精品视频

蟲蟲首頁| 資源下載| 資源專輯| 精品軟件
登錄| 注冊

您現(xiàn)在的位置是:首頁 > 技術(shù)閱讀 >  漫談C變量——優(yōu)化天敵“volatile”

漫談C變量——優(yōu)化天敵“volatile”

時間:2024-02-09

【說在前面的話】


自從紅警1重制以來,除了生病、上班、看漫畫、補番以外,我最大的樂趣就是在steam上參加夜間多人運動——當(dāng)然,也就沒有啥興致去更新。上周發(fā)了一篇原創(chuàng)以后,冷不丁的被人用“打賞”狠狠的催更了一番,好歹也是十六進制兩位數(shù)的打賞——手中的鬼畜般“Acknowledge, Affirmtive”頓時就不香了——趕忙開始更新。

【正文】


在前面的文章《編譯器玄學(xué)報告第一期》中,我們了解到:volatile實際上是告訴編譯器“絕不允許對被修飾的變量動手動腳(做優(yōu)化)”,因為在“編譯器不知道的情況下”,這個變量的值是可能會因為各種原因被更新或者是改變的。實際使用中,volatile 阻斷了編譯器利用通用寄存器對靜態(tài)變量的操作進行優(yōu)化,雖然能保證操作的正確性,卻無法在某些可以優(yōu)化的地方提升性能。例如:

static volatile uint32_t s_wVPort = 0;
void set_vport_u8(uint8_t chValue, uint8_t chOffset){ uint32_t wMask = 0xFF <<chOffset; //!<獲取正確的掩碼    s_wVPort &= ~wMask;                         //!<步驟1:將掩碼對應(yīng)的位置清零 s_wVPort |= ((uint32_t)chValue<<chOffset); //!<步驟2:設(shè)置新值到虛擬端口}


由于volatile的存在,步驟1和步驟2這樣的“讀改寫操作”都會獨立生成針對s_wVPort的讀寫操作,因此上述代碼等效為:

void set_vport_u8(uint8_t chValue, uint8_t chOffset){    uint32_t wMask = 0xFF <<chOffset;        //!<獲取正確的掩碼
//! s_wVPort&= ~wMask; 的等效展開 uint32_t wTemp1 = s_wVPort; //!<步驟1.1 讀取s_wVPort    wTemp1 &= ~wMask;                        //!<步驟1.2 改寫wTemp1 s_wVPort = wTemp1; //!<步驟1.3 將wTemp1寫回s_wVPort
//! s_wVPort |= ((uint32_t)chValue<<chOffset);的等效展開 uint32_t wTemp1 = s_wVPort; //!<步驟2.1 讀取s_wVPort    wTemp1 |= ((uint32_t)chValue<<chOffset); //!<步驟2.2 改寫wTemp1 s_wVPort = wTemp1; //!<步驟2.3 將wTemp1寫回s_wVPort}

顯然,步驟1.3和2.1是多余的,我們可以手工將其優(yōu)化為:

void set_vport_u8(uint8_t chValue, uint8_t chOffset){    uint32_t wMask = 0xFF <<chOffset;        //!<獲取正確的掩碼
//! 將s_wVPort讀取到通用寄存器中(wTemp1編譯器會用通用寄存器來保存) uint32_t wTemp1 = s_wVPort; //!<步驟1.1 讀取s_wVPort
//! 對保存在通用寄存器中的值進行統(tǒng)一修改    wTemp1 &= ~wMask;                         //!<步驟1.2 改寫wTemp1    wTemp1 |= ((uint32_t)chValue<<chOffset);  //!<步驟2.2 改寫wTemp1
//! 將修改后的值寫回s_wVPort s_wVPort = wTemp1; //!<步驟2.3 將wTemp1寫回s_wVPort}

這就是一個手工對volatile修飾的變量進行局部優(yōu)化的例子,本質(zhì)上就是替代編譯器在合適的位置使用通用寄存器對靜態(tài)變量進行“手工窺孔優(yōu)化”。需要注意的是,需要volatile進行修飾的變量通常與多任務(wù)或者中斷/異常有關(guān),因此,進行手工窺孔優(yōu)化時,尤其需要注意“確保數(shù)據(jù)操作的完整性(原子性)”,相關(guān)內(nèi)容,我們將在隨后的文章中為您詳細展開。

volatile的應(yīng)用范圍非常廣泛,尤其是在嵌入式系統(tǒng)中,幾乎所有的外設(shè)寄存器都可以表述為如下的形式:
//!已知某32位外設(shè)寄存器的地址為  XXXXX_IO_REG_BASE_ADDRESS,則對應(yīng)的寄存器可以定義為#defineXXXXX_IO_REG   ( *((volatile uint32_t*)XXXX_IO_REG_BASE_ADDRESS) )

 

考慮到這種情況,應(yīng)用中很多針對外設(shè)寄存器的連續(xù)操作都可以通過“手工窺孔優(yōu)化”來大幅度提高效率。如果可能(在保證程序邏輯正確的情況下),應(yīng)該盡可能減少volatile的使用;或者是限制其使用的范圍;萬不得已的情況下,則應(yīng)該對volatile參與的運算熱點進行“手工窺孔優(yōu)化”

主站蜘蛛池模板: 邢台市| 榆树市| 布尔津县| 康保县| 武汉市| 荥经县| 通化市| 滦南县| 北安市| 嘉义县| 迁西县| 安西县| 泾川县| 乐至县| 灌云县| 大英县| 德钦县| 丹东市| 邵东县| 红桥区| 新龙县| 原平市| 射洪县| 龙泉市| 宁陵县| 克东县| 繁峙县| 那曲县| 紫金县| 息烽县| 涞水县| 彭州市| 南宁市| 五河县| 龙川县| 即墨市| 公安县| 长沙县| 偏关县| 长寿区| 平罗县|