轉譯 NG 架構

Chris Harrelson
Chris Harrelson

您可以在這裡瞭解 RenderingNG 元件的設定方式,以及轉譯管道的流程。

從最高層級開始,轉譯工作如下:

  1. 內容算繪至螢幕上的像素。
  2. 為內容加入視覺效果動畫,讓內容從一個狀態變換到另一個狀態。
  3. 根據輸入內容捲動
  4. 將輸入內容有效地導向正確位置,以便開發人員指令碼和其他子系統做出回應。

要轉譯的內容是每個瀏覽器分頁的框架樹狀結構,以及瀏覽器介面。以及來自觸控螢幕、滑鼠、鍵盤和其他硬體裝置的原始輸入事件串流。

每個畫面都包含:

  • DOM 狀態
  • CSS
  • 面板
  • 外部資源,例如圖片、影片、字型和 SVG

頁框是指 HTML 文件及其網址。 在瀏覽器分頁中載入的網頁含有頂層框架、頂層文件中每個 iframe 的子框架,以及遞迴 iframe 後代。

視覺效果是指套用至位圖的圖形運算,例如捲動、轉換、剪輯、濾鏡、不透明度或混合。

架構元件

在 RenderingNG 中,這些工作會以邏輯方式分散到多個階段和程式碼元件。這些元件最終會出現在各種 CPU 程序、執行緒,以及這些執行緒中的子元件。在網路內容的可靠性可擴充效能擴充性方面,每種類型都發揮了重要作用。

算繪管道結構

轉譯管道的圖表。
箭頭代表各階段的輸入和輸出。階段會以顏色標示,以示說明執行哪個執行緒或程序。在某些情況下,階段可在多個位置執行,具體取決於情況,這也是為什麼有些階段有兩種顏色的原因。綠色階段為轉譯程序主要執行緒;黃色階段為轉譯程序合成器;橘色階段則是視覺化程序。

算繪會在管道中進行,過程中會產生多個階段和構件。每個階段都代表程式碼,在轉譯作業中執行一項定義明確的工作。這些成果是資料結構,可做為階段的輸入或輸出內容。

這些階段如下:

  1. Animate:根據宣告式時間表,變更計算的樣式,並隨時間變更屬性樹狀結構
  2. 樣式:將 CSS 套用至 DOM,並建立計算樣式
  3. 版面配置:決定 DOM 元素在畫面上的大小和位置,並建立不可變動的片段樹狀結構
  4. 預先繪製:計算屬性樹狀結構,並視需要invalidate
  5. 捲動:變更屬性樹狀結構,藉此更新文件和可捲動 DOM 元素的捲動偏移。
  6. 繪圖:計算顯示清單,說明如何從 DOM 將 GPU 紋理圖塊轉為點陣圖。
  7. 提交:將屬性樹狀結構和顯示清單複製到合成器執行緒。
  8. 分層:將顯示清單分割為合成的圖層清單,以便獨立進行轉譯和動畫處理。
  9. 光柵、解碼和著色工作區:將顯示清單、已編碼圖片和著色工作區程式碼分別轉換為 GPU 紋理圖塊
  10. 啟用:建立合成器影格,代表如何在畫面上繪製及放置 GPU 圖塊,以及任何視覺效果。
  11. 匯總:將所有可見合成器影格合併為單一全域合成器影格。
  12. 繪製:在 GPU 上執行匯總合成器影格,在畫面上建立像素。

您可以略過不需要的轉譯管道階段。舉例來說,視覺效果和捲動的動畫可以略過版面配置、預先繪製和繪製動作。因此,動畫和捲動會在圖表中以黃色和綠色圓點標示。如果版面配置、預先繪製和繪畫作業可以略過,則可以在合成器執行緒上完全執行,並略過主執行緒。

這裡並未直接說明瀏覽器 UI 算繪作業,但可將其視為這項管道的簡化版本 (實際上,其實作會共用大部分程式碼)。影片 (也未直接描述) 一般會使用獨立的程式碼進行轉譯,將影格解碼為 GPU 紋理圖塊,然後插入合成器影格和繪圖步驟。

程序和執行緒結構

CPU 程序

使用多個 CPU 程序可在網站之間和瀏覽器狀態中實現效能和安全性隔離,以及從 GPU 硬體實現穩定性和安全性隔離。

CPU 處理程序各部分的圖表

  • 轉譯程序會為單一網站和分頁組合轉譯、製作動畫、捲動及路由輸入內容。您可以使用多種算繪程序。
  • 瀏覽器程序會為瀏覽器 UI (包括網址列、分頁標題和圖示) 轉譯、製作動畫和轉送輸入內容,並將所有剩餘輸入內容轉送至適當的轉譯程序。只有一個瀏覽器程序。
  • 「Viz 程序」匯總多個轉譯程序以及瀏覽器程序的組合。它會使用 GPU 進行算繪和繪製。只有一個 Viz 程序。

不同的網站最終都會進入不同的轉譯程序。

同一網站的多個瀏覽器分頁或視窗通常會進入不同的轉譯程序,除非分頁有相關的關聯,例如一個分頁開啟另一個分頁。在電腦上,如果記憶體壓力過大,Chromium 可能會將同一網站的多個分頁放入同一個轉譯程序,即使這些分頁不相關也一樣。

在單一瀏覽器分頁中,不同網站的框架一律會在不同的轉譯程序中,但同一個網站的框架一律會在同一個轉譯程序中。從轉譯的角度來看,多個轉譯程序的重要優點在於跨網站 iframe 和分頁可彼此達成效能隔離。此外,來源也可以選擇啟用更進階的隔離功能

所有 Chromium 都只有一個 Viz 程序,因為通常只有一個 GPU 和螢幕可繪製。

將 Viz 與其他處理程序進行區隔,會導致 GPU 驅動程式或硬體發生錯誤時,穩定性會有所提升。這也適用於安全性隔離,這對 Vulkan 等 GPU API 和一般安全性而言相當重要。

由於瀏覽器可以有許多分頁和視窗,且所有分頁和視窗都需要繪製瀏覽器 UI 像素,因此您可能會想知道:為何只有一個瀏覽器程序?原因是每次只會聚焦其中一個分頁;事實上,非顯示的瀏覽器分頁大多會停用,並放棄所有 GPU 記憶體。不過,複雜的瀏覽器 UI 算繪功能也越來越常在算繪程序中實作 (稱為 WebUI)。這並非出於效能隔離的原因,而是為了充分利用 Chromium 網路轉譯引擎的便利性。

在較舊的 Android 裝置上,在 WebView 中使用時,系統會共用轉譯和瀏覽器程序 (這通常不適用於 Android 上的 Chromium,僅適用於 WebView)。在 WebView 上,瀏覽器程序也會與嵌入應用程式共用,而 WebView 只有一個轉譯程序。

有時也會有用於解碼受保護影片內容的工具程式。上述圖表並未說明這項程序。

執行緒

不管是緩慢工作、管道平行處理和多個緩衝區,執行緒都有助於維持效能隔離和回應速度。

轉譯程序圖表。

  • 主執行緒會執行指令碼、轉譯事件迴圈、文件生命週期、命中測試、指令碼事件調度,以及剖析 HTML、CSS 和其他資料格式。
    • 主執行緒輔助程式會執行需要編碼或解碼的圖片位圖和 Blob 等工作。
    • Web Workers 執行指令碼,以及 OffscreenCanvas 的轉譯事件迴圈。
  • 合成器執行緒會處理輸入事件、執行網頁內容的捲動和動畫、計算網頁內容的最佳分層方式,以及協調圖片解碼、繪圖工作項和光柵工作。
    • 合成器執行緒輔助程式會協調 Viz 光柵工作,並執行圖片解碼工作、繪圖工作項和備用光柵。
  • 媒體、解複合器或音訊輸出執行緒會解碼、處理及同步處理影片和音訊串流。(請注意,影片會與主要轉譯管道並行執行)。

將主要和合成器執行緒區隔,對於動畫及捲動與主要執行緒作業之間的效能隔離而言相當重要。

每個轉譯程序只有一個主執行緒,即使來自同一個網站的多個分頁或影格可能會在同一個程序中結束。不過,在各種瀏覽器 API 中執行的工作,會受到效能隔離的影響。舉例來說,在 Canvas API 中產生圖片位元圖和 Blob 時,會在主執行緒輔助執行緒中執行。

同樣地,每個轉譯程序只有一個合成器執行緒。一般來說,只有一個工作緒並不會造成問題,因為合成器執行緒上所有非常耗費資源的作業都會委派給合成器 worker 執行緒或 Viz 程序,而這項工作可以與輸入路由、捲動或動畫並行執行。合成器工作站執行緒會協調在 Viz 處理程序中執行的工作,但隨處 GPU 加速可能會因 Chromium 無法控制的原因而失敗,例如驅動程式錯誤。在這種情況下,工作執行緒會在 CPU 的備用模式下執行工作。

合成器工作站執行緒數量取決於裝置功能。舉例來說,電腦通常會使用較多執行緒,因為執行緒的 CPU 核心更多,且電池受限的情況比行動裝置少。以下是放大和縮小的範例。

轉譯程序執行緒架構是套用三種不同最佳化模式的應用程式:

  • 輔助執行緒:將耗時較長的子工作傳送至其他執行緒,讓父執行緒能回應其他同時提出的要求。主執行緒輔助程式和轉譯器輔助程式是這項技巧的絕佳範例。
  • 多重緩衝:在轉譯新內容時顯示先前轉譯的內容,以隱藏轉譯延遲。合成器執行緒會使用這項技巧。
  • 管道平行處理:在多個位置同時執行轉譯管道。這就是捲動和動畫可以快速執行的原因;即使主執行緒正在進行轉譯更新,捲動和動畫仍可同時執行。

瀏覽器處理程序

瀏覽器程序圖表,顯示算繪和合成執行緒與算繪和合成執行緒輔助程式之間的關係。

  • 轉譯和合成的執行緒會回應瀏覽器 UI 中的輸入內容,並將其他輸入內容導向正確的轉譯程序;並會排版及繪製瀏覽器 UI。
  • 轉譯和組合執行緒輔助程式會執行圖片解碼工作,以及備用光柵或解碼。

瀏覽器程序的轉譯和合成執行緒與轉譯程序的程式碼和功能類似,差別在於主執行緒和合成器執行緒會合併為單一執行緒。在這種情況下,只需要一個執行緒,因為設計上沒有長時間主執行緒工作,因此不需要進行效能隔離。

Viz 處理程序

Viz 程序包含 GPU 主執行緒和顯示合成器執行緒。

  • GPU 主執行緒會將顯示清單和影片影格轉為 GPU 紋理圖塊,並將合成器影格繪製至螢幕。
  • 顯示器合成器執行緒會匯總並最佳化各個轉譯程序 (以及瀏覽器程序) 的合成作業,將其轉換為單一合成器影格,以便在螢幕上顯示。

一般來說,影像處理和繪圖作業會在同一個執行緒中執行,因為兩者都需要 GPU 資源,而要可靠地使用多執行緒的 GPU 相當困難 (開發新的 Vulkan 標準,就是為了讓多執行緒存取 GPU 更容易)。由於 WebView 的嵌入方式,Android WebView 會提供獨立的 OS 級轉譯執行緒來繪製畫面。其他平台日後也可能會提供這類執行緒。

顯示合成器位於不同的執行緒,因為它必須隨時回應,且不會阻斷 GPU 主執行緒的任何可能的減速來源。導致 GPU 主執行緒速度變慢的原因之一,是呼叫非 Chromium 程式碼 (例如供應商專屬的 GPU 驅動程式),這些程式碼可能會以難以預測的方式變慢。

元件結構

在每個轉譯程序主執行緒或轉譯器執行緒中,都有邏輯軟體元件,以結構化方式彼此互動。

轉譯程序主執行緒元件

閃爍轉譯器的圖表。

在 Blink 轉譯器中:

  • 本機影格樹狀結構片段代表本機影格樹狀結構,以及影格中的 DOM。
  • DOM 和 Canvas API 元件包含所有這些 API 的實作項目。
  • 文件生命週期執行器會執行轉譯管道步驟,直到執行修訂版本步驟為止。
  • 「input event hit testing and dispatching」元件會執行命中測試,找出事件指定的 DOM 元素,並執行輸入事件調度演算法和預設行為。

「轉譯事件迴圈排程器和執行器」會決定要在事件迴圈和時間執行什麼操作。並安排在與裝置螢幕相符的節奏進行轉譯。

頁框樹狀結構的圖表。

本地影格樹狀結構片段有點複雜。請注意,影格樹狀結構是主頁面及其子項 iframe 的遞迴。如果影格是在算繪程序中算繪,則該影格對該程序而言為本機;否則,該影格為遠端。

您可以根據渲染程序為影格著色。在上圖中,綠色圓圈是單一轉譯程序中的所有影格;橘色圓圈是第二個轉譯程序中的影格,藍色圓圈則是第三個轉譯程序中的影格。

本機影格樹狀結構片段是影格樹狀結構中同色連接元件。圖片中包含四個本機影格樹狀結構:兩個用於網站 A、一個用於網站 B,以及一個用於網站 C。每個本機頁框樹狀結構都有自己的 Blink 轉譯器元件。本機影格樹狀結構的 Blink 轉譯器可能會與其他本機影格樹狀結構位於相同轉譯程序中,也可能不會。這會根據選取算繪程序的方式決定,如前文所述。

算繪程序合成器執行緒結構

顯示轉譯程序合成器元件的圖表。

轉譯程序合成器元件包括:

  • 資料處理常式,可維護組合圖層清單、顯示清單和屬性樹狀結構。
  • 生命週期執行程式,可執行動畫、捲動、組合、光柵、解碼和啟用算繪管道的步驟。(請注意,動畫和捲動動作可能會同時發生在主執行緒和合成器中)。
  • 輸入和觸發測試處理常式會在合成圖層的解析度下執行輸入處理和觸發測試,以判斷是否可以在合成器執行緒上執行捲動手勢,以及應以哪個轉譯程序觸發測試為目標。

實際應用架構範例

本範例有三個分頁:

分頁 1:foo.com

<html>
  <iframe id=one src="foo.com/other-url"></iframe>
  <iframe  id=two src="bar.com"></iframe>
</html>

分頁 2:bar.com

<html>
 …
</html>

分頁 3:baz.com html <html> … </html>

這些分頁的程序、執行緒和元件結構如下所示:

分頁程序的流程圖。

讓我們逐一介紹算繪的四項主要工作。提醒:

  1. 將內容算繪為螢幕上的像素。
  2. Animate:在內容從一個狀態轉換為另一個狀態時,顯示視覺效果。
  3. 根據輸入內容捲動
  4. 路由輸入內容,以便開發人員指令碼和其他子系統回應。

如要算繪分頁 1 的變更 DOM:

  1. 開發人員指令碼會變更 foo.com 轉譯程序中的 DOM。
  2. Blink 轉譯器會告知合成器需要進行轉譯。
  3. 合成器會告訴 Viz 需要進行轉譯。
  4. Viz 會向合成器傳回算繪開始訊號。
  5. 合成器會將起始訊號轉送至 Blink 轉譯器。
  6. 主要執行緒事件迴圈執行程式會執行文件生命週期。
  7. 主執行緒會將結果傳送至轉譯器執行緒。
  8. 組合器事件迴圈執行程式會執行組合生命週期。
  9. 所有影像處理工作都會傳送至 Viz (通常會有多個這類工作)。
  10. 在 GPU 上將內容轉為點陣圖。
  11. Viz 確認已完成影像處理工作。注意:Chromium 通常不會等待光柵完成,而是使用稱為「同步權杖」的東西,該權杖必須在步驟 15 執行前由光柵工作項解析。
  12. 將合成器影格傳送至 Viz。
  13. Viz 會匯總 foo.com 轉譯程序、bar.com iframe 轉譯程序和瀏覽器 UI 的 compositor 影格。
  14. 維茲排定抽獎時間。
  15. Viz 會將匯總的轉譯器影格繪製至螢幕。

如何在第 2 分頁為 CSS 轉換轉換加上動畫效果

  1. bar.com 算繪程序的轉譯器執行緒會透過變更現有屬性樹狀結構,在轉譯器事件迴圈中標記動畫。然後重新執行合成器生命週期。(可能會執行轉譯和解碼工作,但這裡未加以顯示)。
  2. 將合成器影格傳送至 Viz。
  3. Viz 會匯總 foo.com 轉譯程序、bar.com 轉譯程序和瀏覽器 UI 的 compositor 影格。
  4. 維茲排定抽獎時間。
  5. Viz 會將匯總的轉譯器影格繪製至螢幕。

如要在第三個分頁中捲動網頁,請按照下列步驟操作:

  1. 一系列 input 事件 (滑鼠、觸控或鍵盤) 會傳送至瀏覽器程序。
  2. 每個事件都會路由至 baz.com 的轉譯程序轉換器執行緒。
  3. 合成器會判斷主執行緒是否需要瞭解事件。
  4. 必要時,系統會將事件傳送至主執行緒。
  5. 主執行緒會觸發 input 事件監聽器 (pointerdowntouchstarpointermovetouchmovewheel),以查看監聽器是否會在事件上呼叫 preventDefault
  6. 主執行緒會傳回 preventDefault 是否已呼叫至轉譯器。
  7. 否則系統會將輸入事件傳回瀏覽器程序。
  8. 瀏覽器程序會將其與其他近期事件結合,將其轉換為捲動手勢。
  9. 捲動手勢會再次傳送至 baz.com 的轉譯程序轉譯器執行緒,
  10. 捲動畫面會套用在該處,而 bar.com 轉譯程序的轉譯器執行緒會在轉譯器事件迴圈中標記動畫。接著,這會變更屬性樹狀結構中的捲動偏移量,並重新執行轉譯器生命週期。它也會告知主執行緒觸發 scroll 事件 (未在此顯示)。
  11. 將合成器影格傳送至 Viz。
  12. Viz 會匯總 foo.com 轉譯程序、bar.com 轉譯程序和瀏覽器 UI 的轉譯器影格。
  13. Viz 排定繪製時間。
  14. Viz 會將匯總的轉譯器影格繪製至螢幕。

如何路徑分頁 1 上 iframe #two 超連結上的 click 事件:

  1. 瀏覽器程序收到 input 事件 (滑鼠、觸控或鍵盤)。它會執行近似命中測試,判斷 bar.com iframe 轉譯程序應接收點擊,並將點擊傳送至該程序。
  2. bar.com 的轉譯器執行緒會將 click 事件轉送至 bar.com 的主要執行緒,並排程轉譯事件迴圈工作來處理該事件。
  3. bar.com 主執行緒的輸入事件處理器會執行命中測試,判斷點選的是 iframe 中的哪個 DOM 元素,並觸發 click 事件,供指令碼觀察。由於沒有 preventDefault,因此系統會前往超連結。
  4. 載入超連結的目的網頁後,系統會轉譯新狀態,步驟與前述「轉譯變更的 DOM」範例相似。(這裡未顯示後續的變更)。

重點摘要

您可能需要花費大量時間,才能記住及內化算繪的運作方式。

最重要的收穫是,經過仔細的模組化和細節處理,算繪管道已分割為多個獨立的元件。接著,這些元件會在平行處理程序和執行緒中分割,以便盡可能提升可擴充的效能可擴充性機會。

每個元件都扮演著關鍵角色,可提供現代網頁應用程式的效能和功能。

請繼續閱讀瞭解金鑰資料結構,這些結構對於 RenderingNG 作為程式碼元件的重要性。


插圖由 Una Kravets 繪製。