首先來看看C語言溢位小程式
data:image/s3,"s3://crabby-images/ad9ca/ad9cad573b29c5edb65849ecdde262fb68fadcdf" alt="緩衝區溢位小程式分析"
這個程式看著好像是沒有任何問題,執行是這樣的:
data:image/s3,"s3://crabby-images/a4421/a4421fc471b6b37ae1e93086717f818cd767b61b" alt="緩衝區溢位小程式分析"
我們主要關注這個check函式,首先定義了一個int型別的陣列且其中個數為8個,下一行將fun函式地址存入arr[9]中,其實這裡我們可以看到定義陣列的時候個數也就是8,實際到達的下標是7,所以我們最多能取陣列的值為arr[7],連8都不到,這裡我們獲取fun函式地址的時候卻存到了下標為9的地址,跳了一個下標。首先來了解下陣列的儲存方式
data:image/s3,"s3://crabby-images/638f8/638f8cde9d77d995abb12b8075766b751f8dac01" alt="緩衝區溢位小程式分析"
高地址 低地址
這是我簡單畫的陣列儲存的堆疊圖,陣列是使用連續的記憶體空間來儲存的,說白了和區域性變數的儲存差異不大,我們現在開始分析程式碼:
data:image/s3,"s3://crabby-images/2d7ad/2d7adfa6605a84d62bafef178d34f0357258d8be" alt="緩衝區溢位小程式分析"
現在我們記住ebp和esp的值
data:image/s3,"s3://crabby-images/fd583/fd583a1ada5f5d7bc8c8871e5aac5713f1b7ab56" alt="緩衝區溢位小程式分析"
現在我們來把這個堆疊圖畫一下
data:image/s3,"s3://crabby-images/a994f/a994f9eb11fe044a1ee658ee277c12676215dc5a" alt="緩衝區溢位小程式分析"
這裡按F11進入函式體中,call進函式會將它下一行地址壓入棧中,然後調到函式體,這時再來看看堆疊的變化:
data:image/s3,"s3://crabby-images/093e1/093e187dc0e6d070ab4d319458e0f1b913b8d8b5" alt="緩衝區溢位小程式分析"
接著往下走,我們看看函式體中的程式碼:
data:image/s3,"s3://crabby-images/471b9/471b953e323c048a35d2f6e6d9b56fda4f7a43d0" alt="緩衝區溢位小程式分析"
進入函式後將ebp壓入棧中,然後將esp中的值移動到ebp中,之後esp-60h(60h/4h=18h=24)提升棧空間,再來看看此時的堆疊:
接下來將ebx,esi,edi依次推入棧中,下面的程式碼到rep指令是將緩衝區填充為CCCCCCCCCh,再來看看堆疊的變化:
data:image/s3,"s3://crabby-images/d3bcc/d3bcc36e74b863a54a34191a3dd74351d3f9b3db" alt="緩衝區溢位小程式分析"
接著往下走
data:image/s3,"s3://crabby-images/50277/5027733d77b9220339dbbbf56efaa5c3ab7f7c01" alt="緩衝區溢位小程式分析"
這裡是將陣列存到緩衝區中,從低地址向高地址儲存,20h/4h=8h=8從esp的位置向上數八格就是儲存1的位置,依次儲存下來,再來看看棧中的變化:
data:image/s3,"s3://crabby-images/2e9e2/2e9e22ffad006f9ff94bcda40f70b4d99970a4aa" alt="緩衝區溢位小程式分析"
data:image/s3,"s3://crabby-images/f0cd0/f0cd06e1f77a76d3dbe7a7a6868696bf825a1814" alt="緩衝區溢位小程式分析"
這裡是取fun的地址出來並轉換為int型別,然後存入陣列下標為9的地方,我們看看ebp+4這個地址執行前後的變化
執行前
data:image/s3,"s3://crabby-images/817aa/817aafdb5cab57f7b3d9436096d41a02ce430039" alt="緩衝區溢位小程式分析"
經過這個對比之後,ebp+4剛好是這個儲存返回地址的地方,我們繼續往下走
data:image/s3,"s3://crabby-images/ed859/ed859209064ba24eca660e7c86e9738514e5c420" alt="緩衝區溢位小程式分析"
異或eax之後,依次取出edi,esi,ebx,然後將ebp的值移動到esp中,隨後取出ebp,再來看看堆疊變化:
data:image/s3,"s3://crabby-images/8e0b2/8e0b215aa2dfcee1475fa5085345bc9fe8bd8127" alt="緩衝區溢位小程式分析"
上面的操作是使堆疊平衡,接著來看
data:image/s3,"s3://crabby-images/87e9d/87e9d8e67b9107e67e52f2271d75de87e30ced0d" alt="緩衝區溢位小程式分析"
接著執行了一個ret,ret的操作相當於push eip,F11一下,看看eip
data:image/s3,"s3://crabby-images/629e2/629e27656da794201fda8057fe54d0094fbcd1fc" alt="緩衝區溢位小程式分析"
就這樣進入了fun函式
data:image/s3,"s3://crabby-images/e94f5/e94f5f78401a1b7ebbabbd4e5b7e73f7d1e06309" alt="緩衝區溢位小程式分析"
data:image/s3,"s3://crabby-images/a1952/a1952d8bb96cc9bd9a95fc79cd7d1e75814fa12f" alt="緩衝區溢位小程式分析"
本次分析就是分析緩衝區都溢位了為何還能執行fun函式,到這裡我們可以知道溢位之後覆蓋掉了程式的返回地址,所以導致程式進入了fun函式中。溢位並不可怕,可怕的是不會除錯啊!