<kbd id="afajh"><form id="afajh"></form></kbd>
<strong id="afajh"><dl id="afajh"></dl></strong>
    <del id="afajh"><form id="afajh"></form></del>
        1. <th id="afajh"><progress id="afajh"></progress></th>
          <b id="afajh"><abbr id="afajh"></abbr></b>
          <th id="afajh"><progress id="afajh"></progress></th>

          困擾了幾天,代碼優(yōu)化導(dǎo)致的奇葩問題

          共 2755字,需瀏覽 6分鐘

           ·

          2020-10-28 08:58

          關(guān)注、星標(biāo)公眾號(hào),直達(dá)精彩內(nèi)容

          ID:嵌入式Linux

          作者:寫代碼的籃球癡


          這個(gè)是今天在微信群里討論的一個(gè)問題,先看圖片

          點(diǎn)擊查看大圖

          代碼流程大概是這個(gè)樣子的

          點(diǎn)擊查看大圖

          查看 lengthspace1 的值,明顯看到 length 小于 space1 的值,即使是這樣小白都能搞懂流程的情況下,代碼還是跑到else里面區(qū)執(zhí)行

          調(diào)試查看數(shù)據(jù)

          然后 我們就在群里討論,有的大神說這個(gè)是內(nèi)存越界,也有大神說可能是人品有問題,也有大神說這個(gè)是因?yàn)閷懘a前沒有選好一個(gè)良辰吉日,反正大家想法都非常多,也非常古怪,這可能就是討論群存在的一個(gè)原因了。

          經(jīng)過不斷的驗(yàn)證,發(fā)現(xiàn)這個(gè)問題是因?yàn)?code style="font-size: 14px;padding: 2px 4px;border-radius: 4px;margin-right: 2px;margin-left: 2px;background-color: rgba(27, 31, 35, 0.05);font-family: "Operator Mono", Consolas, Monaco, Menlo, monospace;word-break: break-all;color: rgb(212, 65, 243);font-weight: bold;">編譯器優(yōu)化的問題。

          如果在設(shè)置里面把優(yōu)化選項(xiàng)去掉代碼就執(zhí)行正確

          編譯器對(duì)代碼優(yōu)化

          當(dāng)然還有一個(gè)問題,就是如果我想開啟優(yōu)化,畢竟代碼太大占用的存儲(chǔ)空間是很大的,如果能節(jié)省點(diǎn)空間是最好的了。所以就出現(xiàn)了一種,只針對(duì)某些代碼不優(yōu)化的設(shè)置

          像這樣


          我們使用GCC編譯的時(shí)候,也是有可能因?yàn)榇a優(yōu)化導(dǎo)致這樣的問題的,慶幸的是,GCC也有設(shè)置不進(jìn)行優(yōu)化的開關(guān)。

          #使用GCC編譯器設(shè)置選擇性不優(yōu)化某段代碼

          #pragma?GCC?push_options
          #pragma?GCC?optimize?("O0")
          #pragma?GCC?pop_options

          push 的意思是把當(dāng)前的編譯優(yōu)化選項(xiàng)壓棧,然后再設(shè)置當(dāng)前的優(yōu)化選項(xiàng),之后再出棧,把之前壓棧的選項(xiàng)提取出來。

          具體可以參考鏈接:

          https://gcc.gnu.org/onlinedocs/gcc-4.4.2/gcc/Function-Specific-Option-Pragmas.html

          鏈接里面還介紹了一些其他的用法,原文如下

          5.52.12?Function?Specific?Option?Pragmas
          #pragma?GCC?target?("string"...)
          This?pragma?allows?you?to?set?target?specific?options?for?functions?defined?later?in?the?source?file.?One?or?more?strings?can?be?specified.?Each?function?that?is?defined?after?this?point?will?be?as?if?attribute((target("STRING")))?was?specified?for?that?function.?The?parenthesis?around?the?options?is?optional.?See?Function?Attributes,?for?more?information?about?the?target?attribute?and?the?attribute?syntax.
          The?`#pragma?GCC?target'?pragma?is?not?implemented?in?GCC?versions?earlier?than?4.4,?and?is?currently?only?implemented?for?the?386?and?x86_64?backends.

          #pragma?GCC?optimize?("string"...)
          This?pragma?allows?you?to?set?global?optimization?options?for?functions?defined?later?in?the?source?file.?One?or?more?strings?can?be?specified.?Each?function?that?is?defined?after?this?point?will?be?as?if?attribute((optimize("STRING")))?was?specified?for?that?function.?The?parenthesis?around?the?options?is?optional.?See?Function?Attributes,?for?more?information?about?the?optimize?attribute?and?the?attribute?syntax.
          The?`#pragma?GCC?optimize'?pragma?is?not?implemented?in?GCC?versions?earlier?than?4.4.

          #pragma?GCC?push_options
          #pragma?GCC?pop_options
          These?pragmas?maintain?a?stack?of?the?current?target?and?optimization?options.?It?is?intended?for?include?files?where?you?temporarily?want?to?switch?to?using?a?different?`#pragma?GCC?target'?or?`#pragma?GCC?optimize'?and?then?to?pop?back?to?the?previous?options.
          The?`#pragma?GCC?push_options'?and?`#pragma?GCC?pop_options'?pragmas?are?not?implemented?in?GCC?versions?earlier?than?4.4.

          #pragma?GCC?reset_options
          This?pragma?clears?the?current?#pragma?GCC?target?and?#pragma?GCC?optimize?to?use?the?default?switches?as?specified?on?the?command?line.
          The?`#pragma?GCC?reset_options'?pragma?is?not?implemented?in?GCC?versions?earlier?than?4.4.

          #當(dāng)然還有指定某個(gè)函數(shù)設(shè)置優(yōu)化等級(jí)

          int?max(int?a,?int?b)?__attribute__((optimize("O0")));
          {
          ?return?a?}

          #使用volatile 關(guān)鍵字避免編譯器優(yōu)化

          volatile 的作用是提醒CPU,如果遇到被volatile 修飾的變量,要從內(nèi)存里面去取值,而不要偷懶,直接從緩存里面取值,我們一般是用在那些被中斷處理函數(shù)使用的那些變量。

          如果有些代碼,你不希望CPU偷懶,那你就可以加上volatile ,讓CPU從內(nèi)存取數(shù)據(jù)。

          CSDN上有這樣一個(gè)例子

          https://blog.csdn.net/qq_28637193/article/details/88988951今天碰到一個(gè)gcc優(yōu)化相關(guān)的問題,為了讓一個(gè)頁(yè)變成臟頁(yè)(頁(yè)表中dirty位被置上),需要執(zhí)行下面這段代碼:

          1?uint32_t?*page;
          2?//?...
          3?page[0]?=?page[0];
          最后一行代碼很有可能被gcc優(yōu)化掉,因?yàn)檫@段代碼看起來沒有任何實(shí)際的作用。那么如何防止gcc對(duì)這段代碼做優(yōu)化呢?

          設(shè)置gcc編譯時(shí)優(yōu)化級(jí)別為-O0肯定是不合適的,這樣對(duì)程序性能影響會(huì)比較大。stackoverflow上的Dietrich Epp給出了一個(gè)強(qiáng)制類型轉(zhuǎn)換的方案:

          ((unsigned?char?volatile?*)page)[0]?=?page[0];
          通過volatile關(guān)鍵字禁止gcc的優(yōu)化

          #總結(jié)、什么情況會(huì)導(dǎo)致這樣的問題?

          1、堆棧溢出應(yīng)該是一個(gè)原因,之前我有遇到的情況是棧空間設(shè)置太小,然后溢出到堆空間導(dǎo)致問題。


          2、使用某個(gè)函數(shù)導(dǎo)致溢出,我們使用的函數(shù),比如,內(nèi)存拷貝函數(shù),如果長(zhǎng)度設(shè)置不對(duì),也會(huì)導(dǎo)致影響到其他的代碼。


          3、還有就是上面說的編譯器優(yōu)化導(dǎo)致的問題。


          評(píng)論說說你在開發(fā)過程總遇到過哪些奇葩的問題,又是如何解決的呢?


          推薦閱讀:


          嵌入式編程專輯
          Linux 學(xué)習(xí)專輯
          C/C++編程專輯
          Qt進(jìn)階學(xué)習(xí)專輯
          關(guān)注微信公眾號(hào)『技術(shù)讓夢(mèng)想更偉大』,后臺(tái)回復(fù)“m”查看更多內(nèi)容。

          長(zhǎng)按前往圖中包含的公眾號(hào)關(guān)注

          瀏覽 55
          點(diǎn)贊
          評(píng)論
          收藏
          分享

          手機(jī)掃一掃分享

          分享
          舉報(bào)
          評(píng)論
          圖片
          表情
          推薦
          點(diǎn)贊
          評(píng)論
          收藏
          分享

          手機(jī)掃一掃分享

          分享
          舉報(bào)
          <kbd id="afajh"><form id="afajh"></form></kbd>
          <strong id="afajh"><dl id="afajh"></dl></strong>
            <del id="afajh"><form id="afajh"></form></del>
                1. <th id="afajh"><progress id="afajh"></progress></th>
                  <b id="afajh"><abbr id="afajh"></abbr></b>
                  <th id="afajh"><progress id="afajh"></progress></th>
                  久久综合五月天 | AV漫画网| 日韩成人AV在线播放 | 啪啪啪视频免费观看 | 国产精品人人妻人人爽人人牛 |