Condividi tramite


Panoramica del rendering del volume

Per i volumi di mrI o ingegneria medici, vedere Rendering del volume su Wikipedia. Queste "immagini volumetriche" contengono informazioni ricche con opacità e colore nel volume che non possono essere facilmente espresse come superfici come mesh poligonali.

Soluzioni chiave per migliorare le prestazioni

  1. BAD: Approccio naïve: Mostra volume intero, in genere viene eseguito troppo lentamente
  2. GOOD: Piano di taglio: mostra solo una singola sezione del volume
  3. GOOD: Taglio sotto volume: Mostra solo alcuni livelli del volume
  4. GOOD: ridurre la risoluzione del rendering del volume (vedere "Rendering scena di risoluzione mista")

Esiste solo una certa quantità di informazioni che possono essere trasferite dall'applicazione allo schermo in qualsiasi frame specifico, ovvero la larghezza di banda totale di memoria. Inoltre, qualsiasi elaborazione (o "ombreggiatura") necessaria per trasformare i dati per la presentazione richiede tempo. Le considerazioni principali quando si esegue il rendering del volume sono, ad esempio:

  • Screen-Width * Screen-Height * Screen-Count * Volume-Layer-On-That-Pixel = Total-Volume-Samples-Per-Frame
  • 1028 * 720 * 2 * 2 * 256 = 378961920 (100%) (volume di res completo: troppi campioni)
  • 1028 * 720 * 2 * 1 = 1480320 (0,3% di pieno) (sezione sottile: 1 campione per pixel, viene eseguito senza problemi)
  • 1028 * 720 * 2 * 10 = 14803200 (3,9% di pieno) (sezione subvolume: 10 campioni per pixel, viene eseguito abbastanza uniformemente, sembra 3d)
  • 200 * 200 * 2 * 256 = 20480000 (5% di pieno) (volume di res inferiore: meno pixel, volume pieno, sembra 3d ma un po'sfocato)

Rappresentazione di trame 3D

Nella CPU:

public struct Int3 { public int X, Y, Z; /* ... */ }
 public class VolumeHeader  {
   public readonly Int3 Size;
   public VolumeHeader(Int3 size) { this.Size = size;  }
   public int CubicToLinearIndex(Int3 index) {
     return index.X + (index.Y * (Size.X)) + (index.Z * (Size.X * Size.Y));
   }
   public Int3 LinearToCubicIndex(int linearIndex)
   {
     return new Int3((linearIndex / 1) % Size.X,
       (linearIndex / Size.X) % Size.Y,
       (linearIndex / (Size.X * Size.Y)) % Size.Z);
   }
   /* ... */
 }
 public class VolumeBuffer<T> {
   public readonly VolumeHeader Header;
   public readonly T[] DataArray;
   public T GetVoxel(Int3 pos)        {
     return this.DataArray[this.Header.CubicToLinearIndex(pos)];
   }
   public void SetVoxel(Int3 pos, T val)        {
     this.DataArray[this.Header.CubicToLinearIndex(pos)] = val;
   }
   public T this[Int3 pos] {
     get { return this.GetVoxel(pos); }
     set { this.SetVoxel(pos, value); }
   }
   /* ... */
 }

Nella GPU:

float3 _VolBufferSize;
 int3 UnitVolumeToIntVolume(float3 coord) {
   return (int3)( coord * _VolBufferSize.xyz );
 }
 int IntVolumeToLinearIndex(int3 coord, int3 size) {
   return coord.x + ( coord.y * size.x ) + ( coord.z * ( size.x * size.y ) );
 }
 uniform StructuredBuffer<float> _VolBuffer;
 float SampleVol(float3 coord3 ) {
   int3 intIndex3 = UnitVolumeToIntVolume( coord3 );
   int index1D = IntVolumeToLinearIndex( intIndex3, _VolBufferSize.xyz);
   return __VolBuffer[index1D];
 }

Ombreggiatura e sfumature

Come ombreggiatura di un volume, ad esempio MRI, per una visualizzazione utile. Il metodo primario consiste nell'avere una "finestra di intensità" (un min e max) che si desidera visualizzare le intensità all'interno e semplicemente scalare in tale spazio per visualizzare l'intensità nera e bianca. Una "rampa di colore" può quindi essere applicata ai valori all'interno di tale intervallo e archiviata come trama, in modo che diverse parti dello spettro di intensità possano essere ombreggiate in colori diversi:

float4 ShadeVol( float intensity ) {
   float unitIntensity = saturate( intensity - IntensityMin / ( IntensityMax - IntensityMin ) );
   // Simple two point black and white intensity:
   color.rgba = unitIntensity;
   // Color ramp method:
   color.rgba = tex2d( ColorRampTexture, float2( unitIntensity, 0 ) );

In molte delle nostre applicazioni vengono archiviati nel volume sia un valore di intensità non elaborato che un "indice di segmentazione" (per segmentare parti diverse, ad esempio pelle e osso; questi segmenti vengono creati dagli esperti negli strumenti dedicati). Questo può essere combinato con l'approccio precedente per inserire un colore diverso o anche una rampa di colore diversa per ogni indice di segmento:

// Change color to match segment index (fade each segment towards black):
 color.rgb = SegmentColors[ segment_index ] * color.a; // brighter alpha gives brighter color

Slicing del volume in uno shader

Un primo passo è quello di creare un "piano di slicing" che può spostarsi attraverso il volume, 'slicing it', e come i valori di analisi a ogni punto. Ciò presuppone che sia presente un cubo "VolumeSpace", che rappresenta la posizione in cui il volume si trova nello spazio mondiale, che può essere usato come riferimento per inserire i punti:

// In the vertex shader:
 float4 worldPos = mul(_Object2World, float4(input.vertex.xyz, 1));
 float4 volSpace = mul(_WorldToVolume, float4(worldPos, 1));
// In the pixel shader:
 float4 color = ShadeVol( SampleVol( volSpace ) );

Traccia del volume in Shader

Come usare la GPU per eseguire la traccia sottovolume (illustra alcuni voxel profondi, quindi livelli sui dati da indietro a front):

float4 AlphaBlend(float4 dst, float4 src) {
   float4 res = (src * src.a) + (dst - dst * src.a);
   res.a = src.a + (dst.a - dst.a*src.a);
   return res;
 }
 float4 volTraceSubVolume(float3 objPosStart, float3 cameraPosVolSpace) {
   float maxDepth = 0.15; // depth in volume space, customize!!!
   float numLoops = 10; // can be 400 on nice PC
   float4 curColor = float4(0, 0, 0, 0);
   // Figure out front and back volume coords to walk through:
   float3 frontCoord = objPosStart;
   float3 backCoord = frontPos + (normalize(cameraPosVolSpace - objPosStart) * maxDepth);
   float3 stepCoord = (frontCoord - backCoord) / numLoops;
   float3 curCoord = backCoord;
   // Add per-pixel random offset, avoids layer aliasing:
   curCoord += stepCoord * RandomFromPositionFast(objPosStart);
   // Walk from back to front (to make front appear in-front of back):
   for (float i = 0; i < numLoops; i++) {
     float intensity = SampleVol(curCoord);
     float4 shaded = ShadeVol(intensity);
     curColor = AlphaBlend(curColor, shaded);
     curCoord += stepCoord;
   }
   return curColor;
 }
// In the vertex shader:
 float4 worldPos = mul(_Object2World, float4(input.vertex.xyz, 1));
 float4 volSpace = mul(_WorldToVolume, float4(worldPos.xyz, 1));
 float4 cameraInVolSpace = mul(_WorldToVolume, float4(_WorldSpaceCameraPos.xyz, 1));
// In the pixel shader:
 float4 color = volTraceSubVolume( volSpace, cameraInVolSpace );

Rendering intero del volume

Modificando il codice subvolume precedente, si ottiene:

float4 volTraceSubVolume(float3 objPosStart, float3 cameraPosVolSpace) {
   float maxDepth = 1.73; // sqrt(3), max distance from point on cube to any other point on cube
   int maxSamples = 400; // just in case, keep this value within bounds
   // not shown: trim front and back positions to both be within the cube
   int distanceInVoxels = length(UnitVolumeToIntVolume(frontPos - backPos)); // measure distance in voxels
   int numLoops = min( distanceInVoxels, maxSamples ); // put a min on the voxels to sample

Rendering della scena di risoluzione mista

Come eseguire il rendering di una parte della scena con una risoluzione bassa e inserirla sul posto:

  1. Configurare due fotocamere fuori schermo, una per seguire ogni occhio che aggiorna ogni cornice
  2. Configurare due destinazioni di rendering a bassa risoluzione ( ovvero 200x200) che le telecamere vengono eseguite in
  3. Configurare un quad che si sposta davanti all'utente

Ogni cornice:

  1. Disegnare le destinazioni di rendering per ogni occhio a bassa risoluzione (dati del volume, shader costosi e così via)
  2. Disegna la scena normalmente come risoluzione completa (mesh, interfaccia utente e così via)
  3. Disegnare un quad davanti all'utente, sulla scena e proiettare il rendering a bassa res in tale
  4. Risultato: combinazione visiva di elementi a risoluzione completa con dati di volume a bassa risoluzione, ma ad alta densità

Vedere anche