深入了解Unprot

一、概述

Unprot是一個輕量級的內存保護工具。它通過在可執行文件中插入一些技巧性的代碼,來保證內存的安全性。與傳統的內存保護工具不同,Unprot不依賴於任何特定的平台或編譯器,因此同樣可以在不同的操作系統和編譯器上使用。

二、Unprot可實現的內存保護功能

Unprot主要用於保護內存的讀、寫、執行許可權不被越權操作。具體來說,它可以實現以下幾個粒度的內存保護功能:

1)頁面級別的內存保護。Unprot插入的代碼可以將整個頁面(通常是4KB)的內存標記為只讀或不可執行。

void page_protect(void *addr);

2)區域級別的內存保護。Unprot可以把整個內存區域(可以是幾個頁面)標記為只讀或不可執行。

void region_protect(void *addr, size_t size);

3)指針級別的內存保護。Unprot可以將某個指針指向的內存區域標記為只讀或不可執行。

void pointer_protect(void *pointer);

三、Unprot的實現原理

Unprot通過在可執行文件中插入特定的代碼,來對內存進行保護。相比於其他內存保護工具,Unprot的實現原理比較巧妙,同時也比較複雜。

1、頁面級別的保護

Unprot將頁面標記為只讀或不可執行,主要是通過修改該頁面對應的頁表項。當程序試圖寫該頁面的內存時,CPU會檢查該頁表項是否可寫,如果不可寫,則會拋出一個頁錯誤<Page Fault>異常,然後交由操作系統處理。操作系統通常會將該頁面標記為可寫,並重新執行該指令。

#define PT_PAGE_PROT (0x1 << 1)
#define PT_PAGE_NO_EXEC (0x1 << 63)

void page_protect(void *addr) {
  uintptr_t pte = get_pt_entry(addr);
  set_pt_entry(addr, pte & ~PT_PAGE_PROT & ~PT_PAGE_NO_EXEC);
}

void page_unprotect(void *addr) {
  uintptr_t pte = get_pt_entry(addr);
  set_pt_entry(addr, pte | PT_PAGE_PROT | PT_PAGE_NO_EXEC);
}

2、區域級別的保護

對於區域級別的保護,Unprot需要找到包含該區域的所有頁面,並將它們標記為只讀或不可執行。

void region_protect(void *addr, size_t size) {
  void *start_page = round_down(addr, PAGE_SIZE);
  void *end_page = round_up(addr + size, PAGE_SIZE);

  while (start_page < end_page) {
    page_protect(start_page);
    start_page += PAGE_SIZE;
  }
}

3、指針級別的保護

對於指針級別的保護,Unprot會先解引用該指針,然後再對指向的內存區域進行保護。如果該指針指向了一塊只讀或不可執行的內存,Unprot會拋出一個頁錯誤異常。

void pointer_protect(void *pointer) {
  uintptr_t addr = *((uintptr_t *)pointer);
  page_protect(round_down(addr, PAGE_SIZE));
}

四、Unprot的應用場景

Unprot可以被廣泛應用於安全領域。例如:

1)防止攻擊者通過緩衝區溢出等方式改變程序流程。

2)防止攻擊者利用ROP(Return-Oriented Programming)技術執行惡意代碼。

3)防止代碼注入和代碼修改。

五、Unprot的局限與優化

儘管Unprot能夠有效地保護內存,但它也存在一些局限和優化空間:

1)Unprot只能保護可執行代碼段和全局數據段,對於棧和堆等動態分配的內存無法保護。

2)Unprot的代碼實現比較複雜,可能會對程序運行性能產生一定的影響。

3)Unprot只能保護內存不被write和execute越權寫入,但無法保護read的越權讀取。

未來,我們可以考慮優化Unprot的代碼實現,使其更加輕量級和高效。

原創文章,作者:小藍,如若轉載,請註明出處:https://www.506064.com/zh-tw/n/195431.html

(0)
打賞 微信掃一掃 微信掃一掃 支付寶掃一掃 支付寶掃一掃
小藍的頭像小藍
上一篇 2024-12-02 20:35
下一篇 2024-12-02 20:35

相關推薦

  • 深入解析Vue3 defineExpose

    Vue 3在開發過程中引入了新的API `defineExpose`。在以前的版本中,我們經常使用 `$attrs` 和` $listeners` 實現父組件與子組件之間的通信,但…

    編程 2025-04-25
  • 深入理解byte轉int

    一、位元組與比特 在討論byte轉int之前,我們需要了解位元組和比特的概念。位元組是計算機存儲單位的一種,通常表示8個比特(bit),即1位元組=8比特。比特是計算機中最小的數據單位,是…

    編程 2025-04-25
  • 深入理解Flutter StreamBuilder

    一、什麼是Flutter StreamBuilder? Flutter StreamBuilder是Flutter框架中的一個內置小部件,它可以監測數據流(Stream)中數據的變…

    編程 2025-04-25
  • 深入探討OpenCV版本

    OpenCV是一個用於計算機視覺應用程序的開源庫。它是由英特爾公司創建的,現已由Willow Garage管理。OpenCV旨在提供一個易於使用的計算機視覺和機器學習基礎架構,以實…

    編程 2025-04-25
  • 深入了解scala-maven-plugin

    一、簡介 Scala-maven-plugin 是一個創造和管理 Scala 項目的maven插件,它可以自動生成基本項目結構、依賴配置、Scala文件等。使用它可以使我們專註於代…

    編程 2025-04-25
  • 深入了解LaTeX的腳註(latexfootnote)

    一、基本介紹 LaTeX作為一種排版軟體,具有各種各樣的功能,其中腳註(footnote)是一個十分重要的功能之一。在LaTeX中,腳註是用命令latexfootnote來實現的。…

    編程 2025-04-25
  • 深入剖析MapStruct未生成實現類問題

    一、MapStruct簡介 MapStruct是一個Java bean映射器,它通過註解和代碼生成來在Java bean之間轉換成本類代碼,實現類型安全,簡單而不失靈活。 作為一個…

    編程 2025-04-25
  • 深入了解Python包

    一、包的概念 Python中一個程序就是一個模塊,而一個模塊可以引入另一個模塊,這樣就形成了包。包就是有多個模塊組成的一個大模塊,也可以看做是一個文件夾。包可以有效地組織代碼和數據…

    編程 2025-04-25
  • 深入探討馮諾依曼原理

    一、原理概述 馮諾依曼原理,又稱「存儲程序控制原理」,是指計算機的程序和數據都存儲在同一個存儲器中,並且通過一個統一的匯流排來傳輸數據。這個原理的提出,是計算機科學發展中的重大進展,…

    編程 2025-04-25
  • 深入理解Python字元串r

    一、r字元串的基本概念 r字元串(raw字元串)是指在Python中,以字母r為前綴的字元串。r字元串中的反斜杠(\)不會被轉義,而是被當作普通字元處理,這使得r字元串可以非常方便…

    編程 2025-04-25

發表回復

登錄後才能評論