主頁 >  其他 > [畢設] 用Rust實作一個3D渲染器 - 淺談MSAA抗鋸齒

[畢設] 用Rust實作一個3D渲染器 - 淺談MSAA抗鋸齒

2021-05-03 06:08:42 其他

摘要

寫作動機:畢業設計是從頭寫一個3D渲染器,編程語言為Rust,鑒于有幸聽過GAMES101相關課程,遂想為畢設加入4xMSAA抗鋸齒演算法,但踩了幾個大坑,差點自閉,解決后趁思路還算清晰,分享一下遇到的難點和對應方案,

這三篇博文給了我較大啟發:

  1. 博文一

  2. 博文二

  3. 博文三

另外,本文并不會教你MSAA的原理,討論僅從我在實作時應用的方案來切入,


問題的產生

接觸過GAMES101 CG課作業二的伙伴應該熟悉,課程定義的MSAA流程以及我們在網上看到的實作如下:

// 't'為當前三角形面片
seg_pos = {{0.25, 0.25}, {0.75, 0.25},{0.25, 0.75}, {0.75, 0.75}};
for(int x=x_l;x<=x_r;x++)
    for(int y=y_b;y<=y_t;y++) 
        int count = 0
        float min_depth = FLT_MAX;
        for(auto & e : seg_pos) 
            if(insideTriangle((float)x + e[0], (float)y + e[1], t.v)) 
                count++
                ...用重心坐標插值出當前采樣點深度
                min_depth = std::min(min_depth, 當前采樣點深度)
        if(count != 0 && depth_buf[get_index(x, y)] > min_depth) 
            像素顏色 = t.getColor() * (count / 4)
            設定顏色(像素顏色)

這是一種簡化版MSAA:四個子采樣點設定在像素中心的右上角,判斷每個采樣點是否在當前考慮的三角形面片內,若是則命中次數加1,并維護一個變數min_depth,記錄四個采樣點中最小的深度值,如果命中次數為0,或者未通過深度測驗,則直接discard當前像素;否則當前像素設定顏色為 三角形顏色 * 命中率,說白了就是簡單的算術平均,

于是乎,我為畢設實作MSAA時便屁顛屁顛地直接將上述流程照搬,結果出了大問題 -- 每個三角形面片之間出現了黑色裂縫:

一開始我以為又是什么舍入帶來的精度損失,查閱資料后,好像在實作GAMES作業2的時候也有類似問題,

于是我重新回去運行作業2,確實在三角形交界處會出現黑線:

直接原因便是,這種演算法存在一個缺陷,具體請看圖:

(注意,作業2比較特殊,整個場景就定義了兩個三角形面片,也就是藍色和黃綠色三角形)

圖中,我們正在藍色三角形內進行光柵化,考慮一種最壞的情況,即邊緣像素僅有左下角的采樣點命中了藍色三角形內部,

按照上面偽代碼的思路,根本不考慮另外三個處于黃綠色三角形內的采樣點的貢獻,當前像素顏色簡單粗暴設定為 t.getColor() * (1/4) ,這里的 \(t\) 即為藍色三角形,它的顏色為 \(RGB(194,217,233)\),則當前像素會被設定為 \(RGB(48,54,58)\),呈現出一種類似于黑色的顏色,所以也不難想象當采樣點命中為2、3個的時候,顏色也只是會稍微淺一點罷了,這與右邊的黃綠色產生了一種較強的割裂感,從而產生裂縫的錯覺,


嘗試解決

意識到問題后,我便查閱資料,并確定要實作一種開銷較大的MSAA:

每個像素有4bit的coverage mask,以及4個深度值(每個sample各一個),另外還有其他attribute(最簡單的就是color),并且是可以拿一份color copy給所有sample(準確地說是只copy給那些mask非0的sample),而這份color可以來自像素中心或者某個sample,

策略

我選用的策略是,對于一個像素,算出其每個被命中sample(采樣點)的顏色、深度,最后再對相關值進行關于命中次數的算術平均:

(這個例子和上述作業2不太一樣,這里假設背景的藍色和黃綠色僅恰巧為某些材質的接縫處,并不是簡單的兩個三角形交界;圖中的三角形是很多三角面片的其中一個)

如圖,按照我選擇的策略,該像素最右的sample在該三角形之外,被discard(丟棄),那么該像素的顏色即為剩下三個sample顏色的算術平均,最終顏色在邊界處產生一種較為平滑的過渡,


實作

你可能會注意到我在講述策略時用了一種不太一樣的sample模式,也即采樣點并非在像素右上角,而是環繞像素中心并略微旋轉:

因為在應對三角形邊界情形時,順時針旋轉26.6度的算子被證明是一種比較有效的sample模式,下面來看如何通過代碼一步步實作:

為了得到目標模式,我們先確立一個正正方方的采樣模式,其各sample都是一個Vector2型別,為這些sample均乘以一個旋轉矩陣即可:

pub static MSAA_LEVEL: usize = 4;
pub static MSAA_OFFSET: f32 = 0.25;
pub static MSAA_SAMPLE_POS: Matrix2<Vector2<f32>> = Matrix2::new(
    Vector2::new(-MSAA_OFFSET,-MSAA_OFFSET),Vector2::new(MSAA_OFFSET,MSAA_OFFSET),
    Vector2::new(-MSAA_OFFSET,MSAA_OFFSET),Vector2::new(MSAA_OFFSET,-MSAA_OFFSET),
);
  • MSAA_LEVEL表征一個像素的sample個數
  • MSAA_OFFSET表征每個sample與像素中心x、y的偏移絕對值
  • MSAA_SAMPLE_POS表征還未旋轉的sample模式

計算旋轉后的模式:

pub fn calc_conv() -> Matrix2<Vector2<f32>> {
    let mut conv_tmp: Matrix2<Vector2<f32>> = Default::default();
    let rotate: Matrix2<f32> = rotate_matrix2d(-26.6);
    for i in 0..4 {
        conv_tmp[i] = rotate*MSAA_SAMPLE_POS[i];
    }
    conv_tmp
}

為了方便輸出單個像素四個sample的狀態,我定義了一個結構叫做 MsaaTensor

pub struct MsaaTensor {
    mask:  Vector4<bool>,
    dept:  Vector4<f32>,
    colo:  Vector4<Vector3<f32>>,
}
  • mask記錄各sample是否被激活(命中 / hit)
  • dept記錄各sample的深度
  • colo記錄各sample的顏色

進入渲染例程,通過三個變數奠基:

for 像素(x,y) in 當前三角形包圍盒 {
    let 像素坐標 = x + y * self.height;
    let 張量 = msaa_tensors[像素坐標];
    let hit = 0.0;
    ......
}

為每個像素計算其MsaaTensor四個角的資訊:

for 像素(x,y) in 當前三角形包圍盒 {
    ......
    for idx in 0..MSAA_LEVEL {
        let 采樣點重心坐標 = barycentric(三角形三頂點,像素.x+采樣模式[idx].x,像素.y+采樣模式[idx].y);
        if 在三角形內(采樣點重心坐標) {
            hit += 1.0;
            let 深度 = 深度插值(采樣點重心坐標,三角形三頂點);
            tensor.設定標志(idx,true);
            tensor.設定深度(idx,深度);
            tensor.設定顏色(idx,著色器.顏色(采樣點重心坐標));
        }else {
            tensor.設定標志(idx,false);
        }
    }
    ......
}

如果沒有子sample被hit,那么直接discard此像素:

for 像素(x,y) in 當前三角形包圍盒 {
    ......
    if hit == 0.0 {
        continue;
    }
    ......
}

只要有一個sample被hit,計算blend資訊并著色:

for 像素(x,y) in 當前三角形包圍盒 {
    .......
    else {
        let 漫反射_blend = Default::default();
        let 深度_blend = 0.0;
        for idx in 0..MSAA_LEVEL {
            if msaa_tensors[ipixel].標志位(idx) == true {
                漫反射_blend += msaa_tensors[ipixel].顏色(idx);
                深度_blend   += msaa_tensors[ipixel].深度(idx);
            }
        }

        // 當前pixel的深度由被hit的子sample混合得到
        深度_blend /= hit;
        if 深度快取(像素.x,像素.y)>深度_blend {
            continue;
        }

        深度快取(像素x, 像素.y, depth_blend);
        幀快取(像素.x, 像素.y, 漫反射_blend/hit);
    }
    ......
}

運行程式看看效果:

對比發現,好像除了一些地方變糊了,模型內部顏色交界處改善不太明顯,那么試著把MSAA_OFFSET的步長改大一點:

pub static MSAA_OFFSET: f32 = 0.5;

效果相對明顯了,但我發現除了顏色變換劇烈的邊緣,一些較為平滑的區域也被MSAA處理得糊成一片,原因是我們的策略會對一切sample hit數不為0的像素進行算術平均處理,這與MSAA的理念相違背了,

改進

若像素的sample hit數為4,則表明該像素并不處于三角形面片邊緣,也就沒必要進行MSAA平均處理,直接用改像素點本身的顏色著色就行了,有了這個覺悟,為代碼加一段判定:

for 像素(x,y) in 當前三角形包圍盒 {
    .......
    else if hit == 4.0 {
        let 像素重心坐標 = barycentric(三角形三頂點,像素.x,像素.y);
        let 深度 = 深度插值(像素重心坐標,三角形三頂點);
        if 深度快取(像素.x,像素.y)>深度 {
            continue;
        }
        深度快取(像素x, 像素.y, 深度);
        幀快取(像素.x, 像素.y, 著色器.顏色(像素重心坐標));
        continue;
    }
    ......
}

看看效果:

很明顯內部平滑部分未被MSAA影響,但似乎對比起來效果不顯著,這里我還是感到比較困惑,歡迎伙伴一起討論,

轉載請註明出處,本文鏈接:https://www.uj5u.com/qita/282226.html

標籤:其他

上一篇:求助 用python kivy開發的app的微信、支付寶登錄和支付的相關課程

下一篇:javase面試問題

標籤雲
其他(157675) Python(38076) JavaScript(25376) Java(17977) C(15215) 區塊鏈(8255) C#(7972) AI(7469) 爪哇(7425) MySQL(7132) html(6777) 基礎類(6313) sql(6102) 熊猫(6058) PHP(5869) 数组(5741) R(5409) Linux(5327) 反应(5209) 腳本語言(PerlPython)(5129) 非技術區(4971) Android(4554) 数据框(4311) css(4259) 节点.js(4032) C語言(3288) json(3245) 列表(3129) 扑(3119) C++語言(3117) 安卓(2998) 打字稿(2995) VBA(2789) Java相關(2746) 疑難問題(2699) 细绳(2522) 單片機工控(2479) iOS(2429) ASP.NET(2402) MongoDB(2323) 麻木的(2285) 正则表达式(2254) 字典(2211) 循环(2198) 迅速(2185) 擅长(2169) 镖(2155) 功能(1967) .NET技术(1958) Web開發(1951) python-3.x(1918) HtmlCss(1915) 弹簧靴(1913) C++(1909) xml(1889) PostgreSQL(1872) .NETCore(1853) 谷歌表格(1846) Unity3D(1843) for循环(1842)

熱門瀏覽
  • 網閘典型架構簡述

    網閘架構一般分為兩種:三主機的三系統架構網閘和雙主機的2+1架構網閘。 三主機架構分別為內端機、外端機和仲裁機。三機無論從軟體和硬體上均各自獨立。首先從硬體上來看,三機都用各自獨立的主板、記憶體及存盤設備。從軟體上來看,三機有各自獨立的作業系統。這樣能達到完全的三機獨立。對于“2+1”系統,“2”分為 ......

    uj5u.com 2020-09-10 02:00:44 more
  • 如何從xshell上傳檔案到centos linux虛擬機里

    如何從xshell上傳檔案到centos linux虛擬機里及:虛擬機CentOs下執行 yum -y install lrzsz命令,出現錯誤:鏡像無法找到軟體包 前言 一、安裝lrzsz步驟 二、上傳檔案 三、遇到的問題及解決方案 總結 前言 提示:其實很簡單,往虛擬機上安裝一個上傳檔案的工具 ......

    uj5u.com 2020-09-10 02:00:47 more
  • 一、SQLMAP入門

    一、SQLMAP入門 1、判斷是否存在注入 sqlmap.py -u 網址/id=1 id=1不可缺少。當注入點后面的引數大于兩個時。需要加雙引號, sqlmap.py -u "網址/id=1&uid=1" 2、判斷文本中的請求是否存在注入 從文本中加載http請求,SQLMAP可以從一個文本檔案中 ......

    uj5u.com 2020-09-10 02:00:50 more
  • Metasploit 簡單使用教程

    metasploit 簡單使用教程 浩先生, 2020-08-28 16:18:25 分類專欄: kail 網路安全 linux 文章標簽: linux資訊安全 編輯 著作權 metasploit 使用教程 前言 一、Metasploit是什么? 二、準備作業 三、具體步驟 前言 Msfconsole ......

    uj5u.com 2020-09-10 02:00:53 more
  • 游戲逆向之驅動層與用戶層通訊

    驅動層代碼: #pragma once #include <ntifs.h> #define add_code CTL_CODE(FILE_DEVICE_UNKNOWN,0x800,METHOD_BUFFERED,FILE_ANY_ACCESS) /* 更多游戲逆向視頻www.yxfzedu.com ......

    uj5u.com 2020-09-10 02:00:56 more
  • 北斗電力時鐘(北斗授時服務器)讓網路資料更精準

    北斗電力時鐘(北斗授時服務器)讓網路資料更精準 北斗電力時鐘(北斗授時服務器)讓網路資料更精準 京準電子科技官微——ahjzsz 近幾年,資訊技術的得了快速發展,互聯網在逐漸普及,其在人們生活和生產中都得到了廣泛應用,并且取得了不錯的應用效果。計算機網路資訊在電力系統中的應用,一方面使電力系統的運行 ......

    uj5u.com 2020-09-10 02:01:03 more
  • 【CTF】CTFHub 技能樹 彩蛋 writeup

    ?碎碎念 CTFHub:https://www.ctfhub.com/ 筆者入門CTF時時剛開始刷的是bugku的舊平臺,后來才有了CTFHub。 感覺不論是網頁UI設計,還是題目質量,賽事跟蹤,工具軟體都做得很不錯。 而且因為獨到的金幣制度的確讓人有一種想去刷題賺金幣的感覺。 個人還是非常喜歡這個 ......

    uj5u.com 2020-09-10 02:04:05 more
  • 02windows基礎操作

    我學到了一下幾點 Windows系統目錄結構與滲透的作用 常見Windows的服務詳解 Windows埠詳解 常用的Windows注冊表詳解 hacker DOS命令詳解(net user / type /md /rd/ dir /cd /net use copy、批處理 等) 利用dos命令制作 ......

    uj5u.com 2020-09-10 02:04:18 more
  • 03.Linux基礎操作

    我學到了以下幾點 01Linux系統介紹02系統安裝,密碼啊破解03Linux常用命令04LAMP 01LINUX windows: win03 8 12 16 19 配置不繁瑣 Linux:redhat,centos(紅帽社區版),Ubuntu server,suse unix:金融機構,證券,銀 ......

    uj5u.com 2020-09-10 02:04:30 more
  • 05HTML

    01HTML介紹 02頭部標簽講解03基礎標簽講解04表單標簽講解 HTML前段語言 js1.了解代碼2.根據代碼 懂得挖掘漏洞 (POST注入/XSS漏洞上傳)3.黑帽seo 白帽seo 客戶網站被黑帽植入劫持代碼如何處理4.熟悉html表單 <html><head><title>TDK標題,描述 ......

    uj5u.com 2020-09-10 02:04:36 more
最新发布
  • 2023年最新微信小程式抓包教程

    01 開門見山 隔一個月發一篇文章,不過分。 首先回顧一下《微信系結手機號資料庫被脫庫事件》,我也是第一時間得知了這個訊息,然后跟蹤了整件事情的經過。下面是這起事件的相關截圖以及近日流出的一萬條資料樣本: 個人認為這件事也沒什么,還不如關注一下之前45億快遞資料查詢渠道疑似在近日復活的訊息。 訊息是 ......

    uj5u.com 2023-04-20 08:48:24 more
  • web3 產品介紹:metamask 錢包 使用最多的瀏覽器插件錢包

    Metamask錢包是一種基于區塊鏈技術的數字貨幣錢包,它允許用戶在安全、便捷的環境下管理自己的加密資產。Metamask錢包是以太坊生態系統中最流行的錢包之一,它具有易于使用、安全性高和功能強大等優點。 本文將詳細介紹Metamask錢包的功能和使用方法。 一、 Metamask錢包的功能 數字資 ......

    uj5u.com 2023-04-20 08:47:46 more
  • vulnhub_Earth

    前言 靶機地址->>>vulnhub_Earth 攻擊機ip:192.168.20.121 靶機ip:192.168.20.122 參考文章 https://www.cnblogs.com/Jing-X/archive/2022/04/03/16097695.html https://www.cnb ......

    uj5u.com 2023-04-20 07:46:20 more
  • 從4k到42k,軟體測驗工程師的漲薪史,給我看哭了

    清明節一過,盲猜大家已經無心上班,在數著日子準備過五一,但一想到銀行卡里的余額……瞬間心情就不美麗了。最近,2023年高校畢業生就業調查顯示,本科畢業月平均起薪為5825元。調查一出,便有很多同學表示自己又被平均了。看著這一資料,不免讓人想到前不久中國青年報的一項調查:近六成大學生認為畢業10年內會 ......

    uj5u.com 2023-04-20 07:44:00 more
  • 最新版本 Stable Diffusion 開源 AI 繪畫工具之中文自動提詞篇

    🎈 標簽生成器 由于輸入正向提示詞 prompt 和反向提示詞 negative prompt 都是使用英文,所以對學習母語的我們非常不友好 使用網址:https://tinygeeker.github.io/p/ai-prompt-generator 這個網址是為了讓大家在使用 AI 繪畫的時候 ......

    uj5u.com 2023-04-20 07:43:36 more
  • 漫談前端自動化測驗演進之路及測驗工具分析

    隨著前端技術的不斷發展和應用程式的日益復雜,前端自動化測驗也在不斷演進。隨著 Web 應用程式變得越來越復雜,自動化測驗的需求也越來越高。如今,自動化測驗已經成為 Web 應用程式開發程序中不可或缺的一部分,它們可以幫助開發人員更快地發現和修復錯誤,提高應用程式的性能和可靠性。 ......

    uj5u.com 2023-04-20 07:43:16 more
  • CANN開發實踐:4個DVPP記憶體問題的典型案例解讀

    摘要:由于DVPP媒體資料處理功能對存放輸入、輸出資料的記憶體有更高的要求(例如,記憶體首地址128位元組對齊),因此需呼叫專用的記憶體申請介面,那么本期就分享幾個關于DVPP記憶體問題的典型案例,并給出原因分析及解決方法。 本文分享自華為云社區《FAQ_DVPP記憶體問題案例》,作者:昇騰CANN。 DVPP ......

    uj5u.com 2023-04-20 07:43:03 more
  • msf學習

    msf學習 以kali自帶的msf為例 一、msf核心模塊與功能 msf模塊都放在/usr/share/metasploit-framework/modules目錄下 1、auxiliary 輔助模塊,輔助滲透(埠掃描、登錄密碼爆破、漏洞驗證等) 2、encoders 編碼器模塊,主要包含各種編碼 ......

    uj5u.com 2023-04-20 07:42:59 more
  • Halcon軟體安裝與界面簡介

    1. 下載Halcon17版本到到本地 2. 雙擊安裝包后 3. 步驟如下 1.2 Halcon軟體安裝 界面分為四大塊 1. Halcon的五個助手 1) 影像采集助手:與相機連接,設定相機引數,采集影像 2) 標定助手:九點標定或是其它的標定,生成標定檔案及內參外參,可以將像素單位轉換為長度單位 ......

    uj5u.com 2023-04-20 07:42:17 more
  • 在MacOS下使用Unity3D開發游戲

    第一次發博客,先發一下我的游戲開發環境吧。 去年2月份買了一臺MacBookPro2021 M1pro(以下簡稱mbp),這一年來一直在用mbp開發游戲。我大致分享一下我的開發工具以及使用體驗。 1、Unity 官網鏈接: https://unity.cn/releases 我一般使用的Apple ......

    uj5u.com 2023-04-20 07:40:19 more