在线观看www成人影院-在线观看www日本免费网站-在线观看www视频-在线观看操-欧美18在线-欧美1级

0
  • 聊天消息
  • 系統消息
  • 評論與回復
登錄后你可以
  • 下載海量資料
  • 學習在線課程
  • 觀看技術視頻
  • 寫文章/發帖/加入社區
會員中心
創作中心

完善資料讓更多小伙伴認識你,還能領取20積分哦,立即完善>

3天內不再提示

程序運行過程中,有些數據被莫名修改了怎么辦?

魚鷹談Linux ? 來源:魚鷹談單片機 ? 2023-08-10 09:11 ? 次閱讀

導讀程序運行過程中,有些數據被莫名修改了,在哪里修改的?又是怎么修改的?這個代碼我只想知道是否運行過,或者運行了多少次,但是不想讓程序停下來,或者僅打印調試信息,怎么辦?當這個變量設置成某個數據后,我想讓程序自動暫停下來進行分析,怎么辦?

以上問題的所有答案就在本節內容:斷點窗口(KEIL)。

本節內容將顛覆你之前對斷點調試的認知。這個調試技巧魚鷹也用了半年多了,當時知道這個調試方法的時候特別興奮,感覺發現了新大陸。而這個調試技巧也在魚鷹接手公司項目代碼的時候快速解決了不少疑難雜癥,而前些天又擴展學習了這個技巧的功能,更是讓魚鷹在學會之后輕松解決了好幾個一般調試方法很難解決的 BUG,相信這個技巧也將為魚鷹之后的開發調試之旅發揮更大的作用。

我們知道常規的斷點調試是在想觀察哪里的問題時就在對應的代碼地址設置斷點,并且一旦運行到斷點位置會讓程序自動暫停運行,這種斷點調試功能確實為開發者解決 bug 立下了汗馬功勞,但是這種方式有很大的局限性,因為很多時候我們并不需要讓程序停下來,而只想知道是否在這段代碼運行過,或者說發生問題的位置根本不能停下來,否則就會讓整個系統功能出現問題,比如中斷處理函數的調試,程序一旦停下了也就失去了所有中斷的后續響應;比如兩個設備通信,一方采用常規斷點的方式調試,肯定會打斷正常的通信過程,而這可不是我們想要的,我們只想知道在收到或發送數據后得到環境快照,而并不想讓程序停下來。以上這些問題可以采用打印方式解決,但是打印調試也有很多弊端:

以串口為例:

1、你必須添加必要的打印和串口驅動代碼,如果你使用 printf 函數,你還得重定向(如果對空間要求高的話,你得知道使用 printf 差不多要占用 1K 大小代碼空間)。

2、如果打印效率比較低,常規波特率 9600 和 115200 打印一個字符串耗時可能比較久,那么對于中斷頻率較高的函數就可能就不適用了。如果你使用 printf 函數,你還得考慮函數是否可重入問題。

3、在代碼中引入調試代碼有風險,本來程序運行沒有問題的,一旦引入調試代碼之后可能就出現了問題,這種情況對于擁有豐富開發經驗的人來說應該見怪不怪了。原因就在于打印輸出時間太久,打亂了程序運行的節奏(而這也是我推薦使用 ITM 調試的一個原因,因為它的輸出效率比串口要高得多),或者打印函數本身有問題,也會導致程序運行出現問題。

4、調試完畢之后,你必須把對應的調試代碼刪除(不管是刪除代碼還是使用宏,都要進行這一步),不然會影響運行效率。而人是健忘的(也不能說健忘,可能只是因為專注于 BUG 本身,容易忘記其它細枝末節,而解決 bug 之后的欣喜更可能忘記后續處理工作了)這個時候你可以嘗試用 #warnning。但是這一步還是必不可少。

而以上問題的解決方案就是 KEIL 的斷點調試窗口!

首先打開數據觀察點的窗口:

wKgZomTUOW2ADmrMAAB1dhoTLhU371.jpg

快捷鍵是 Ctrl + B。

可以看到如下窗口:

wKgZomTUOW2AY8TMAADMq-dSmjs058.jpg

當然你也可以通過下面這種方式打開并設置:

wKgaomTUOW2AAmtMAAA-uLtbxa0701.jpg

從這里你會發現,其實這個窗口就是用來管理你設置的斷點的。平常使用的設置斷點方法只是其中的一種特例罷了。

首先要知道的就是,調試器支持的斷點數量是有限的,具體有多少視情況而定,一旦 KEIL 警告你設置斷點太多,那么就要刪除一些斷點了:

wKgZomTUOW2AJVLJAABGcmUjByY947.jpg

常規用法

1、代碼位置運行次數

有些時候我們想知道某些代碼的運行次數,比如進入中斷處理函數的次數,尋常的斷點設置方式必然會讓程序停止在中斷程序中,但有些時候我們并不希望它停下來。這個時候,你只需要打開該窗口,找到已有的對應斷點位置,雙擊之后就可以看到類似下面的窗口:

wKgaomTUOW2ADc5VAADYv6G0mxQ847.jpg

此時,你將 Count 的值設置的盡可能大一些,那么就可以讓程序運行多次之后才停止。

比如我們設置 Count 的值為 100 次,那么必須在該代碼位置運行 100 次才會讓程序暫停。當你設置完后點擊【Define】后,就會詢問你是否需要重新定義,你選擇“是”即可。

wKgaomTUOW2AS0eBAACuHRco3xI616.jpg

這樣你的斷點變成了這樣:

wKgZomTUOW2ADwgBAABFBAzUj5Q331.jpg

后面的 count=100 表示剩余運行次數為 100,運行 100 次后將停止程序。前面的 00 代表斷點號,E 代表這是一個執行斷點,0x080016B0 代表代碼地址,后面的是源碼位置。

當這個斷點位置運行了 2 次,重新打開該窗口(刷新數據),發現這個數變成了 98,從而可以推算出,已經運行了多少了。如果說你想讓這段代碼運行 2 次后停止,那么你只需要一開始設置 Count 的值為 2 即可。

wKgaomTUOW2ARIoGAABEcivcu8A999.jpg

2、數據訪問

有些時候我們需要知道一些變量會在哪里被訪問,那么你可以設置該變量的訪問條件。比如魚鷹想知道 emOspery 變量會在哪里被讀取?那么你只需設置如下:

wKgaomTUOW6AdjfbAACbAAbgNpI179.jpg

定義之后就是這樣:

wKgZomTUOW6ARJF8AABfooKNIr8644.jpg

因為 Count 值設置為 1,所以每一次讀取 emOspery 的操作都將使程序停止。比如這段代碼:

wKgaomTUOW6AFuAEAABQrN0svLk696.jpg

還有后面的打印函數也使用 emOsprey 變量,所以也會導致程序運行停止。可能你會感到奇怪,為什么 emOsprey++這樣的操作也會涉及到讀取?事實上你理解了 CPU 寄存器存在的意義也就明白了。

而當你設置為寫(Write)訪問時,你會發現從復位程序開始運行后,程序會停止在某個地方,這是為什么?當你知道全局變量會在進入 main 函數之前被初始化時,你也就明白為什么了。

在這里我們選擇使用 Objects 訪問,即按整個變量對象進行訪問,上面的 emOsprey 變量實際上是 uint16_t,所以 len 為 2,即字節大小。也就說,如果你設置為 Objects 訪問,那么它會根據實際的情況設置訪問范圍。

為了更好的說明這一點,我構造一個結構體。

wKgZomTUOW6AfDgfAAA0XCgr-Z0927.jpg

這個結構體大小可以看出是 6 個字節。

然后設置訪問該結構體的條件:

wKgaomTUOW6ABk_nAACE4fJh0nM983.jpg

如果我們按 Objects 訪問的話,那么下面的每一條語句都會導致程序運行的停止。

wKgZomTUOW6AFzCUAABCWKobRZw667.jpg

這是因為這些數據都在 Osprey 結構體的范圍內(從這里也可以了解到,只要在 len 的范圍內的訪問都會導致程序停止運行,所以你可以試試將 Size 設置得更大)。

而如果設置為 Byte 訪問的話,那么就只有第一條語句才會導致程序停止運行:

wKgZomTUOW6AJGjVAABw9XSiy_M990.jpg

實際上如果你希望只在某個結構體成員變量被訪問時才停止,那么直接這么設置就可以:

wKgaomTUOW6AKpnnAADXNY-L8O4992.jpg

你會發現設置是如此之簡單。

實際上還有一種更為通用的訪問方式,即按地址訪問。

上面可以看出 Ospery.Ospery1 成員變量的地址為 0x20000016(由此我們知道也可以通過這個來看出一個結構體變量的地址是多少)。所以我們可以這樣設置:

wKgZomTUOW6AJEsuAAA_9cQntL8506.jpg

而代碼位置的斷點設置亦是如此。

斷點太多,怎么知道程序因何停止?看你的命令窗口就知道了:

wKgZomTUOW6ASMBJAABEd0BxxIQ378.jpg

3、數據匹配

有些時候,我們并不關注地址訪問情況,而對變量的數據內容感興趣。比如說魚鷹想讓變量emOspery 等于 1 時停下來,怎么設置?

wKgaomTUOW6AOB_cAAB0YfPdv5A746.jpg

只要簡單的設置 emOspery == 1 即可(注意必須設置訪問條件,并且 Size 設置正確)。

事實上你也可以設置兩個變量相等作為條件:

wKgZomTUOW6AastMAAA8cpaGO7s992.jpg

設置為不等也是可以的:

wKgaomTUOW6AEfPkAABfSk-GaRo987.jpg

當然還有其它支持的運算就靠你們自己去發現了(可支持運算:&,&&,<,<=,>,>= ,==,!=)。

注意:以上內容可以組合使用,比如讀、寫條件,計數器計數等可以同時設置。滿足條件時就會讓程序運行停止。

高級用法

以上為比較常規的調試功能,現在說說魚鷹剛學習的技能,這個技能的使用靈活性更大,而且對于解決疑難雜癥更是不二之選。

首先設置一個你需要的斷點:

wKgZomTUOW6AAzJ0AACANkrk4sk075.jpg

打開斷點窗口,并雙擊你之前設置的斷點:

wKgaomTUOW6AZxz8AABP5MTtsQ4358.jpg

設置 Command 為【printf(“USRAT_Init()\n”)】(注意\n,否則可能不能輸出,這個應該是 KEIL 的一個 bug)。最后【Define】

清空你之前的命令(如果你不嫌亂的話,也可以不清空):

wKgZomTUOW6ADvA8AAA6XfuEEEc746.jpg

那么你的程序每次運行到這個代碼位置都會在Command 窗口輸出一條信息:

wKgaomTUOW6AO6SwAAAc2GhrhMA432.jpg

但是你的程序并不會停止。

如果說你想讓斷點代碼位置運行多次之后才輸出一條信息也是可以的,只要設置Count 即可。

這里可能你會問,這 printf 不就是我們寫的打印函數嗎?事實上,是,也不是。

這個函數是打印函數沒錯,但是這是 KEIL 調用的打印函數,輸出位置是 Command 窗口,和你自己寫的代碼沒一點關系,每次觸發條件時KEIL 都會調用該函數進行打印,而不會讓你的程序暫停運行。事實上這個 Command 絕不僅僅只是設置 printf這么簡單,如果真是這樣我也不會如此推崇它了,感興趣的可以去官網查找關于調試命令的使用方法。

因為是利用 KEIL 去執行打印任務,所以對你的程序幾乎沒有任何影響,并且在你設置斷點后也不用擔心刪除代碼問題,可以放心飲用。還有一個額外的好處就是,對于所有能設置調試斷點的單片機都適用,因此對于調試器也就沒有過多的要求了,比如說,不管你是用JLINK、ST-LINK 還是CMSIS-DAP(CMSIS-DAP 不能使用ITM,所以魚鷹才會想著用別的方式替代。總算是找到了,而且它在某些方面更出色),都可以這么用。

現在摘錄官網一些關于斷點窗口的知識:

表達式定義斷點類型:

§當設置標志 **Read **或 **Write **或兩者時,訪問中斷(A被定義 。發生指定的內存訪問時會觸發斷點。以字節為單位指定內存訪問窗口的大小,或者以表達式的對象大小指定。對于此斷點類型,**Expression **必須解析為內存地址和內存類型。允許的運算符(&,&&,<。<=。>,> =,= =和!=)在程序執行暫停或執行**命令**之前比較變量值 。

§當Expression解析為代碼地址時,將執行執行中斷(E。到達指定的代碼地址時觸發斷點。代碼地址必須引用 CPU 指令的第一個字節。

§當Expression不能簡化為地址時,定義條件中斷(C。當條件表達式變為 TRUE 時,斷點將觸發。在每條 CPU 指令之后重新計算條件表達式,并且會大大減慢程序執行速度。

計數值指定的次數的斷點表達式必須計算為 TRUE 斷點觸發之前的數目。

命令被指定的μVision 執行語句,然后恢復執行程序。此處指定的命令可以是μVision 調試或信號功能。要從這些函數中暫停程序執行,請設置系統變量break

注意

當在模擬器中將訪問斷點(讀或寫)設置為外設寄存器(SFR)時,即使應用程序未訪問外設寄存器,斷點也可能觸發。發生這種情況是因為μVision 模擬器在應用程序驅動和模擬器內部訪問之間沒有區別。

里面有一個比較關鍵的就是關于條件中斷(C,如果你設置的表達式不是一個代碼地址,也沒有設置讀寫訪問條件,那么就會被設置為條件中斷,一旦設置為條件中斷,那么會在每條匯編指令后計算表達式,這會影響程序正常運行速度,所以沒有必要的話,不要設置為條件中斷。

設置斷點的一般錯誤總結:

當彈出以下窗口時,說明斷點設置錯誤,需要查看命令窗口才能知道具體錯誤信息。

wKgZomTUOW6AAvbWAABLxpmJx9o596.jpg

**a) **斷點太多

刪除一些斷點即可

**b) **重復定義斷點

這是因為之前你已經定義了這個斷點,而現在你又定義了這個斷點,這個時候你可以選擇覆蓋之前的斷點或者保留之前的斷點

**c) **不允許對同一個資源設置不同類型斷點

這個是由于對同一個資源準備設置不同斷點導致的,需要刪除之前的設置的斷點才行。

**d) **表達式錯誤

檢查你的表達式是否正確,注意如果你使用了運算符,那么對于浮點變量的支持好像并不正常,不管你怎么設置,都說表達式錯誤。

到此,斷點窗口(前期我叫它數據觀察點,我也不知道從哪看到的這個詞,后來覺得還是斷點窗口比較準確)的內容就結束了。這個小節內容對于調試而言絕對是一大利器,也是魚鷹決定寫這個KEIL 調試系列文章的主要原因。但是以上所有的調試內容都有一個很大的局限性,就是它只能定格在某一刻(如果你使用Command 命令就不一樣了),而這一刻前面的所有信息都無法知曉。這個時候就要了解另一個調試技能,ITM,它能將程序從出生(復位程序開始)到死亡(死循環或者斷電)的大部分信息記錄下來。這個章節內容早已發布,感興趣的就去前面看一看咯。






審核編輯:劉清

聲明:本文內容及配圖由入駐作者撰寫或者入駐合作網站授權轉載。文章觀點僅代表作者本人,不代表電子發燒友網立場。文章及其配圖僅供工程師學習之用,如有內容侵權或者其他違規問題,請聯系本站處理。 舉報投訴
  • 寄存器
    +關注

    關注

    31

    文章

    5343

    瀏覽量

    120368
  • 計數器
    +關注

    關注

    32

    文章

    2256

    瀏覽量

    94569
  • CMSIS
    +關注

    關注

    0

    文章

    40

    瀏覽量

    11905
  • 調試器
    +關注

    關注

    1

    文章

    305

    瀏覽量

    23741
  • 串口驅動
    +關注

    關注

    2

    文章

    82

    瀏覽量

    18652

原文標題:數據被篡改了,無法在線調試該怎么定位?

文章出處:【微信號:eOsprey,微信公眾號:魚鷹談Linux】歡迎添加關注!文章轉載請注明出處。

收藏 人收藏

    評論

    相關推薦

    CAN發送接收過程中收到到的數據過大怎么辦??

    本人在做CAN庫函數。現在遇到一個問題:要是接收的數據包過大,接收緩沖區放不下,怎么辦?同樣在發送過程中也是這樣的問題,要是發送過程中數據
    發表于 07-18 13:05

    認證名字寫錯了 無法修改了 怎么辦

    認證名字寫錯了 無法修改了 怎么辦 改了幾次 都改不過來求救了
    發表于 04-18 14:16

    MSP430F149 芯片運行過程中Flash更改

    ,會直接跳到boot程序后首先更改0xffe2處的內容從而轉入boot程序執行。現在的情況是,在運行過程中也就一晚上吧,有些芯片的Flash
    發表于 05-24 10:05

    STM32F407VG內部程序區域修改了

    大家好!!STM32F407VG的主控在工作一段時間后,偶爾會有多片出現程序丟失的問題,重新燒好程序之后可以正常工作了。拿回板卡對STM32F407VG內部Flash讀取,發現內部程序區域有些
    發表于 04-17 07:36

    STM32在調試程序過程中發現按鍵切換會導致程序卡死怎么辦

    STM32在調試程序過程中發現按鍵切換會導致程序卡死怎么辦
    發表于 11-16 08:28

    嵌入式的變量意外修改怎么辦

    如何將實時操作系統移植到一款RISC-V架構芯片上去呢?嵌入式變量意外修改怎么辦呢?
    發表于 12-24 07:43

    程序運行過程中可以直接調用rt_device_control修改串口屬性嗎

    程序運行過程中可以直接調用 rt_device_control 修改串口屬性嗎?如串口波特率
    發表于 08-01 10:49

    RT-Thread studio在仿真過程中想實現程序的復位該怎么辦

    studio我一直以為下面這個這個按鈕是這個功能,但是每次按了之后都沒有達到復位的效果,想問一下大家,這個按鈕是干嘛的?如果在仿真過程中想實現程序的復位該怎么辦?謝謝!
    發表于 08-30 14:21

    Python如何防止數據修改Python的深拷貝與淺拷貝的問題說明

    在平時工作,經常涉及到數據的傳遞。在數據傳遞使用過程中,可能會發生數據
    的頭像 發表于 03-30 09:54 ?3056次閱讀
    Python如何防止<b class='flag-5'>數據</b><b class='flag-5'>被</b><b class='flag-5'>修改</b>Python<b class='flag-5'>中</b>的深拷貝與淺拷貝的問題說明

    內存修改了怎么辦

    內存又稱主存,是CPU能直接尋址的存儲空間,內存是電腦中的主要部件,它是相對于外存而言的。內存是電腦中的主要部件,它是相對于外存而言的。
    發表于 06-02 09:36 ?1310次閱讀

    內存卡寫保護怎么辦

    現在生活我們總是會使用到各種各樣的智能設備,內存卡則是我們經常會使用到的,但是在內存卡使用過程中也會遇到一些問題,這就需要我們進行專業的處理,以幫助解決所遇到的問題,那么內存卡寫保護怎么辦
    發表于 01-01 16:53 ?7833次閱讀

    原來單片機還隱藏了這些調試技巧 | 顛覆認知

    1.1.1.1 斷點窗口參考鏈接:http://www.keil.com/support/man/docs/uv4cl/uv4cl_db_dbg_breakpnts.htm程序運行過程中有些
    發表于 12-22 19:50 ?5次下載
    原來單片機<b class='flag-5'>中</b>還隱藏了這些調試技巧 | 顛覆認知

    水泵控制使用過程中出現信號報警怎么辦

    過控制方式我們發現,水泵一般設置在自動狀態比較合適。從而實現便捷、自動化的功能需求。 水泵控制使用過程中,出現信號報警怎么辦?有哪些排查步驟?
    發表于 07-25 11:34 ?3104次閱讀
    水泵控制使用<b class='flag-5'>過程中</b>出現信號報警<b class='flag-5'>怎么辦</b>

    在焊接過程中發現錫膏太稀怎么辦

    近日有客戶咨詢在焊接過程中發現錫膏太稀怎么辦,今天佳金源錫膏廠家來為大家簡單分析一下,如果錫膏太稀,可能會導致在焊接過程中無法獲得良好的焊點質量。以下是發現錫膏太稀怎么辦的幾種可能的臨
    的頭像 發表于 11-24 17:31 ?796次閱讀
    在焊接<b class='flag-5'>過程中</b>發現錫膏太稀<b class='flag-5'>怎么辦</b>?

    電力補償電容器運行過程中缺相怎么辦

    電力補償電容器是現代電力系統不可或缺的重要設備之一。它能夠有效改善電力系統的功率因數,提高電能利用率,降低電力系統的損耗。然而,在電力補償電容器的運行過程中,有時會出現缺相的問題。那么,當電力補償電容器
    的頭像 發表于 02-20 14:28 ?988次閱讀
    電力補償電容器<b class='flag-5'>運行</b><b class='flag-5'>過程中</b>缺相<b class='flag-5'>怎么辦</b>?
    主站蜘蛛池模板: 日本特黄特色特爽大片老鸭| 久久综合色综合| 国产一级特黄aa大片爽爽| 天天干天天拍| 亚洲欧洲一区二区三区在线| 特级做a爰片毛片免费看 | 色性网| 国产色网| 把小嫩嫩曰出白浆| 亚洲 欧美 自拍 卡通 综合| 狠狠色丁香婷婷综合欧美| 国内精品久久影视| 曰本又色又爽又黄三级视频| 黄色三级在线看| 国产理论最新国产精品视频| 亚洲三级在线免费观看| 国产精品网址你懂的| 日本69xxⅹxxxxxx19| 深爱五月网| 亚洲第一视频在线播放| 亚洲国产一区二区在线| 能看的黄色网址| 久久久久久久久久免免费精品| 永久在线免费| 天天操 夜夜操| 欧美满足你的丝袜高跟ol| 欧美人与动欧交视频| 天天噜天天干| 国产福利vr专区精品| 国产亚洲精品自在久久77| 3344在线| 日本三级a| 理论片亚洲| 性欧美高清视频| 中国一级特黄视频| 日本一区二区三区视频在线观看| 欧美feer| www激情| 欧美一卡二三卡四卡不卡| 国内精品久久久久影院免费| 天堂精品在线|