| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967 | //// The multimedia graphics platform GLScene https://github.com/glscene//unit GLS.ROAMPatch;(*  Class for managing a ROAM (square) patch *)interface{$I GLScene.inc}uses  Winapi.OpenGL,  Winapi.OpenGLext,  System.SysUtils,    GLS.OpenGLTokens,   GLS.XOpenGL,   GLS.VectorGeometry,  GLS.HeightData,   GLS.VectorLists,   GLS.Context,  GLS.VectorTypes,  GLS.Isolines,  GLS.Strings;type  // Exception use by splitter for SafeTesselation  EGLROAMException = class(Exception);  PROAMTriangleNode = ^TROAMTriangleNode;  TROAMTriangleNode = packed record    Base, Left, Right: PROAMTriangleNode;    LeftChild, RightChild: PROAMTriangleNode;  end;  TROAMRenderPoint = packed record    X, Y: Integer;    Idx: Integer;  end;  TGLROAMPatch = class(TObject)  private    FID: Integer;    FHeightData: TGLHeightData; // Referred, not owned    FHeightRaster: PSmallIntRaster;    FTLNode, FBRNode: PROAMTriangleNode;     FTLVariance, FBRVariance: array of cardinal;    FPatchSize, FTriangleCount: Integer;    FListHandle: TGLListHandle;    FTag: Integer;    FObserverPosition: TAffineVector;    FNorth, FSouth, FWest, FEast: TGLROAMPatch; // neighbours    FHighRes: Boolean;    FMaxDepth: Integer;    FVertexScale, FVertexOffset: TAffineVector;    FTextureScale, FTextureOffset: TAffineVector;    FMaxTLVarianceDepth, FMaxBRVarianceDepth: Integer;    FOcclusionQuery: TGLOcclusionQueryHandle;    FOcclusionSkip, FOcclusionCounter: Integer;    FLastOcclusionTestPassed: Boolean;    FContourInterval: Integer;    FContourWidth: Integer;  protected    procedure SetHeightData(Val: TGLHeightData);    procedure SetOcclusionSkip(Val: Integer);    procedure RenderROAM(Vertices: TGLAffineVectorList;      VertexIndices: TGLIntegerList; TexCoords: TGLTexPointList);    procedure RenderAsStrips(Vertices: TGLAffineVectorList;      VertexIndices: TGLIntegerList; TexCoords: TGLTexPointList);  public    constructor Create;    destructor Destroy; override;    procedure ComputeVariance(Variance: Integer);    procedure ResetTessellation;    procedure ConnectToTheWest(WestPatch: TGLROAMPatch);    procedure ConnectToTheNorth(NorthPatch: TGLROAMPatch);    // Returns false if MaxCLODTriangles limit is reached    function Tesselate: Boolean;    (*  AV free version of Tesselate.      When IncreaseTrianglesCapacity is called, all PROAMTriangleNode      values in higher function became invalid due to the memory shifting.      Recursivity is the main problem, that's why SafeTesselate is calling      Tesselate in a try..except . *)    function SafeTesselate: Boolean;    (*  Render the patch in high-resolution.      The lists are assumed to have enough capacity to allow AddNC calls      (additions without capacity check). High-resolution renders use      display lists, and are assumed to be made together. *)    procedure RenderHighRes(Vertices: TGLAffineVectorList;      VertexIndices: TGLIntegerList; TexCoords: TGLTexPointList; ForceROAM: Boolean);    (*  Render the patch by accumulating triangles.      The lists are assumed to have enough capacity to allow AddNC calls      (additions without capacity check).      Once at least autoFlushVertexCount vertices have been accumulated,      perform a FlushAccum *)    procedure RenderAccum(Vertices: TGLAffineVectorList;      VertexIndices: TGLIntegerList; TexCoords: TGLTexPointList;      AutoFlushVertexCount: Integer);    // Render all vertices accumulated in the arrays and set their count back to zero.    class procedure FlushAccum(Vertices: TGLAffineVectorList;      VertexIndices: TGLIntegerList; TexCoords: TGLTexPointList);    property HeightData: TGLHeightData read FHeightData write SetHeightData;    property VertexScale: TAffineVector read FVertexScale write FVertexScale;    property VertexOffset: TAffineVector read FVertexOffset write FVertexOffset;    property ObserverPosition: TAffineVector read FObserverPosition  write FObserverPosition;    property TextureScale: TAffineVector read FTextureScale write FTextureScale;    property TextureOffset: TAffineVector read FTextureOffset  write FTextureOffset;    property HighRes: Boolean read FHighRes write FHighRes;    //  Number of frames to skip after an occlusion test returned zero pixels.    property OcclusionSkip: Integer read FOcclusionSkip write SetOcclusionSkip;    //  Number of frames remaining to next occlusion test.    property OcclusionCounter: Integer read FOcclusionCounter write FOcclusionCounter;    (*  Result for the last occlusion test.      Note that this value is updated upon rendering the tile in      non-high-res mode only. *)    property LastOcclusionTestPassed: Boolean read FLastOcclusionTestPassed;    property ID: Integer read FID;    property TriangleCount: Integer read FTriangleCount;    property Tag: Integer read FTag write FTag;    // Distance between contours - zero (default) for no contours    property ContourInterval: Integer read FContourInterval write FContourInterval default 0;    // Width of contours    property ContourWidth: Integer read FContourWidth write FContourWidth default 1;  end;// Specifies the maximum number of ROAM triangles that may be allocated.procedure SetROAMTrianglesCapacity(nb: Integer);function GetROAMTrianglesCapacity: Integer;//  Draw contours on rendering terrain patchesprocedure DrawContours(Vertices: TGLAffineVectorList; VertexIndices: TGLIntegerList;  ContourInterval: Integer; ContourWidth: Integer; DecVal: Integer);// ------------------------------------------------------------------implementation// ------------------------------------------------------------------var  FVBOVertHandle, FVBOTexHandle: TGLVBOArrayBufferHandle;  FVBOIndicesHandle: TGLVBOElementArrayHandle;  vNextPatchID: Integer;  vNbTris, vTriangleNodesCapacity: Integer;  vTriangleNodes: array of TROAMTriangleNode;  RenderRaster: PSmallIntRaster;  RenderIndices: PIntegerArray;  RenderVertices: TGLAffineVectorList;  RenderTexCoords: TGLTexPointList;  TessMaxVariance: Cardinal;  TessMaxDepth: Cardinal;  TessCurrentVariance: PIntegerArray;  TessObserverPosX, TessObserverPosY: Integer;type  TROAMVariancePoint = packed record    X, Y, Z: Integer;  end;procedure SetROAMTrianglesCapacity(nb: Integer);begin  vNbTris := 0;  if vTriangleNodesCapacity <> nb then  begin    SetLength(vTriangleNodes, nb);    vTriangleNodesCapacity := nb;  end;end;function GetROAMTrianglesCapacity: Integer;begin  Result := vTriangleNodesCapacity;end;procedure DrawContours(Vertices: TGLAffineVectorList; VertexIndices: TGLIntegerList;  ContourInterval: Integer; ContourWidth: Integer; DecVal: Integer);var  i: Integer;  Isolines: TGLAffineVectorList;  CurColor: TGLVector;begin  if ContourInterval > 0 then  begin    gl.PolygonOffset(1, 1);    gl.Enable(GL_POLYGON_OFFSET_FILL);    i := VertexIndices.Count - 3;    Isolines := TGLAffineVectorList.Create;    while i >= 0 do    begin      TriangleElevationSegments(Vertices[VertexIndices[i]],        Vertices[VertexIndices[i + 1]], Vertices[VertexIndices[i + 2]],        ContourInterval, Isolines);      Dec(i, DecVal);    end;    gl.PushAttrib(GL_ENABLE_BIT or GL_CURRENT_BIT);    gl.Disable(GL_TEXTURE_2D);    gl.LineWidth(ContourWidth);    gl.GetFloatv(GL_CURRENT_COLOR, @CurColor);    gl.Color4f(0, 0, 0, 1);    gl.Begin_(GL_LINES);     for i := 0 to Isolines.Count - 1 do       gl.Vertex3fv(@Isolines.List[i]);    gl.End_;    gl.Color4fv(@CurColor);    gl.PopAttrib;    Isolines.Free;  end;end;// The result is the delta between the old address of the array and the new onefunction IncreaseTrianglesCapacity(NewCapacity: Integer): int64;  procedure FixNodePtr(var p: PROAMTriangleNode; const delta: int64);  begin    if p = nil then      exit;    Inc(PByte(p), delta);  end;var  oldbase, newbase: pointer;  node: PROAMTriangleNode;  i, oldsize: Integer;begin  Result := 0;  if NewCapacity <= vTriangleNodesCapacity then    exit;  oldsize := vTriangleNodesCapacity;  oldbase := pointer(vTriangleNodes);  SetLength(vTriangleNodes, NewCapacity);  vTriangleNodesCapacity := NewCapacity;  newbase := pointer(vTriangleNodes);  // Array has not been relocated, no need to fix  if oldbase = newbase then    exit;  // go through all the old nodes and fix the pointers  // YP: Delphi needs int64 dual casting to avoid overflow exception  Result := int64(Cardinal(newbase)) - int64(Cardinal(oldbase));  for i := 0 to oldsize - 1 do  begin    node := @vTriangleNodes[i];    FixNodePtr(node^.Base, Result);    FixNodePtr(node^.Left, Result);    FixNodePtr(node^.Right, Result);    FixNodePtr(node^.LeftChild, Result);    FixNodePtr(node^.RightChild, Result);  end;end;function AllocTriangleNode: PROAMTriangleNode; var  nilNode: PROAMTriangleNode;begin  if vNbTris >= vTriangleNodesCapacity then  begin    // grow by 50%   IncreaseTrianglesCapacity(vTriangleNodesCapacity + (vTriangleNodesCapacity shr 1));  end;  Result := @vTriangleNodes[vNbTris]; ///Result := vNbTris;  with Result^ do  begin    nilNode := nil;    Left := nilNode;    Right := nilNode;    LeftChild := nilNode;    RightChild := nilNode;  end;  Inc(vNbTris);end;function Split(tri: PROAMTriangleNode): Boolean;var  n: Integer;  lc, rc: PROAMTriangleNode;  Shift: int64;begin  Result := Assigned(tri.LeftChild);  if Result then    exit; // dont split if tri already has a left child  with tri^ do  begin    // If this triangle is not in a proper diamond, force split our base neighbor    if Assigned(Base) and (Base.Base <> tri) then      Split(Base);    n := vNbTris;  end;  if n >= vTriangleNodesCapacity - 1 then  begin    // grow by 50%    Shift := IncreaseTrianglesCapacity(vTriangleNodesCapacity +      (vTriangleNodesCapacity shr 1));    if Shift <> 0 then    begin      raise EGLROAMException.Create        ('PROAMTriangleNode addresses are invalid now');    end;  end;  with tri^ do  begin    // Creates children and cross-link them    lc := @vTriangleNodes[n]; // left child    rc := @vTriangleNodes[n + 1]; // right child    LeftChild := lc;    RightChild := rc;    rc.Base := Right; // right child    rc.LeftChild := nil;    rc.RightChild := LeftChild;    rc.Right := LeftChild;    lc.Base := Left; // left child    lc.LeftChild := nil;    lc.RightChild := LeftChild;    lc.Left := RightChild;    Inc(vNbTris, 2);    // Link our Left Neighbor to the new children    if Assigned(Left) then      if Left.Base = tri then        Left.Base := lc      else if Left.Left = tri then        Left.Left := lc      else        Left.Right := lc;    // Link our Right Neighbor to the new children    if Assigned(Right) then      if Right.Base = tri then        Right.Base := rc      else if Right.Left = tri then        Right.Left := rc      else        Right.Right := rc;    // Link our Base Neighbor to the new children    if Assigned(Base) then    begin      if Assigned(Base.LeftChild) then      begin        Base.LeftChild.Right := RightChild;        RightChild.Left := Base.LeftChild;        Base.RightChild.Left := LeftChild;        LeftChild.Right := Base.RightChild;      end      else        Split(Base);    end    else    begin // An edge triangle, trivial case.      LeftChild.Right := nil;      RightChild.Left := nil;    end;  end;  Result := True;end;// ------------------// ------------------ TGLROAMPatch ------------------// ------------------constructor TGLROAMPatch.Create;begin  inherited Create;  FID := vNextPatchID;  Inc(vNextPatchID);  FListHandle := TGLListHandle.Create;  FContourInterval := 0;  FOcclusionQuery := TGLOcclusionQueryHandle.Create;end;destructor TGLROAMPatch.Destroy;begin  FListHandle.Free;  FOcclusionQuery.Free;  inherited Destroy;end;procedure TGLROAMPatch.SetHeightData(Val: TGLHeightData);begin  FHeightData := Val;  FPatchSize := FHeightData.Size - 1;  FHeightRaster := Val.SmallIntRaster;end;procedure TGLROAMPatch.SetOcclusionSkip(Val: Integer);begin  if Val < 0 then    Val := 0;  if FOcclusionSkip <> Val then  begin    FOcclusionSkip := Val;    FOcclusionQuery.DestroyHandle;  end;end;procedure TGLROAMPatch.ConnectToTheWest(WestPatch: TGLROAMPatch);begin  if Assigned(WestPatch) then  begin    if not(WestPatch.HighRes or HighRes) then    begin       FTLNode.left := westPatch.FBRNode;       westPatch.FBRNode.left := FTLNode;    end;    FWest := WestPatch;    WestPatch.FEast := Self;  end;end;procedure TGLROAMPatch.ConnectToTheNorth(NorthPatch: TGLROAMPatch);begin  if Assigned(NorthPatch) then  begin    if not(NorthPatch.HighRes or HighRes) then    begin      FTLNode.right := northPatch.FBRNode;      northPatch.FBRNode.right := FTLNode;    end;    FNorth := NorthPatch;    NorthPatch.FSouth := Self;  end;end;procedure TGLROAMPatch.ComputeVariance(Variance: Integer);var  raster: PSmallIntRaster;  currentVariance: PIntegerArray;  maxVarianceDepth: Integer;  maxNonNullIndex: Integer;  invVariance: Single;  function ROAMVariancePoint(anX, anY: Integer): TROAMVariancePoint;  begin    Result.X := anX;    Result.Y := anY;    Result.Z := (Integer(FHeightRaster[anY][anX]) shl 8);  end;  function RecursComputeVariance(const Left, Right, apex: TROAMVariancePoint;    node: Integer): cardinal;  var    half: TROAMVariancePoint;    v: cardinal;    n2: Integer;  begin    with half do    begin      X := (Left.X + Right.X) shr 1;      Y := (Left.Y + Right.Y) shr 1;      Z := Integer(raster[Y][X]) shl 8;      Result := ScaleAndRound(Abs(((Left.Z + Right.Z) div 2) - Z), invVariance);    end;    n2 := node shl 1;    if n2 < maxVarianceDepth then    begin      v := RecursComputeVariance(apex, Left, half, n2);      if v > Result then        Result := v;      v := RecursComputeVariance(Right, apex, half, 1 + n2);      if v > Result then        Result := v;    end;    currentVariance[node] := Result;  end;  procedure ScaleVariance(n, d: Integer);  var    newVal: Integer;  begin    if d >= 0 then      newVal := (currentVariance[n] shl (d shr 1))    else      newVal := (currentVariance[n] shr (-d shr 1));    currentVariance[n] := newVal;    if newVal > 0 then      if n > maxNonNullIndex then        maxNonNullIndex := n;    n := n shl 1;    if n < maxVarianceDepth then    begin      Dec(d);      ScaleVariance(n, d);      ScaleVariance(n + 1, d);    end;  end;var  s, p: Integer;begin  invVariance := 1 / Variance;  s := Sqr(FPatchSize);  raster := FHeightRaster;  FMaxDepth := 1;  p := -1 - 8;  repeat    FMaxDepth := FMaxDepth shl 2;    Inc(p);  until FMaxDepth >= s;  maxVarianceDepth := FMaxDepth;  SetLength(FTLVariance, maxVarianceDepth);  SetLength(FBRVariance, maxVarianceDepth);  s := FPatchSize;  currentVariance := @FTLVariance[0];  maxNonNullIndex := 1;  RecursComputeVariance(ROAMVariancePoint(0, s), ROAMVariancePoint(s, 0),    ROAMVariancePoint(0, 0), 1);  ScaleVariance(1, p);  FMaxTLVarianceDepth := maxNonNullIndex + 1;  SetLength(FTLVariance, FMaxTLVarianceDepth);  currentVariance := @FBRVariance[0];  maxNonNullIndex := 1;  RecursComputeVariance(ROAMVariancePoint(s, 0), ROAMVariancePoint(0, s),    ROAMVariancePoint(s, s), 1);  ScaleVariance(1, p);  FMaxBRVarianceDepth := maxNonNullIndex + 1;  SetLength(FBRVariance, FMaxBRVarianceDepth);end;procedure TGLROAMPatch.ResetTessellation;begin  FTLNode := AllocTriangleNode;  FBRNode := AllocTriangleNode;  FTLNode.Base := FBRNode;  FTLNode.Left := nil;  FTLNode.Right := nil;  FBRNode.Base := FTLNode;  FBRNode.Left := nil;  FBRNode.Right := nil;  FNorth := nil;  FSouth := nil;  FWest := nil;  FEast := nil;end;// returns false if tessellation failed due to MaxCLODTriangles limitfunction RecursTessellate(tri: PROAMTriangleNode; n: Cardinal;  const Left, Right, apex: Cardinal): Boolean;var  d: Integer;begin  Result := True;  d := ((Left + Right) shr 1);  if TessCurrentVariance[n] > d then  begin    Result := False;    if Split(tri) then    begin      n := n shl 1;      if n < TessMaxVariance then      begin        RecursTessellate(Tri.LeftChild, n, apex, Left, d);        Result := RecursTessellate(Tri.RightChild, n + 1, Right, apex, d);      end;    end;  end;end;function TGLROAMPatch.Tesselate: Boolean;var  tessFrameVarianceDelta: Integer;  function VertexDist(X, Y: Integer): Cardinal;  var    f: Single;  const    c1Div100: Single = 0.01;  begin    if HighRes then      f := 0.2 * Sqr(FPatchSize)    else      f := Sqr(X - TessObserverPosX) + Sqr(Y - TessObserverPosY) +        tessFrameVarianceDelta;    Result := Round(Sqrt(f) + f * c1Div100);  end;procedure FullBaseTess(tri: PROAMTriangleNode; n: Cardinal); forward;  procedure FullLeftTess(tri: PROAMTriangleNode; n: Cardinal);  begin    if Split(tri) then    begin      n := n shl 1;      if n < TessMaxDepth then        FullBaseTess(tri.LeftChild, n);    end;  end;  procedure FullRightTess(tri: PROAMTriangleNode; n: Cardinal);  begin    if Split(tri) then    begin      n := n shl 1;      if n < TessMaxDepth then        FullBaseTess(tri.RightChild, n);    end;  end;  procedure FullBaseTess(tri: PROAMTriangleNode; n: Cardinal);  begin    if Split(tri) then    begin      n := n shl 1;      if n < TessMaxDepth then      begin        FullRightTess(tri.LeftChild, n);        FullLeftTess(tri.RightChild, n);      end;    end;  end;var  s: Integer;begin  TessMaxDepth := FMaxDepth;  TessObserverPosX := Round(FObserverPosition.X);  TessObserverPosY := Round(FObserverPosition.Y);  if HighRes then  begin    FullRightTess(FTLNode, 1);    FullRightTess(FBRNode, 1);    FullLeftTess(FBRNode, 1);    FullLeftTess(FTLNode, 1);    tessFrameVarianceDelta := 0;  end  else  begin    if Assigned(FNorth) and FNorth.HighRes then      FullRightTess(FTLNode, 1);    if Assigned(FSouth) and FSouth.HighRes then      FullRightTess(FBRNode, 1);    if Assigned(FEast) and FEast.HighRes then      FullLeftTess(FBRNode, 1);    if Assigned(FWest) and FWest.HighRes then      FullLeftTess(FTLNode, 1);    if FObserverPosition.Z > 0 then      tessFrameVarianceDelta := Round(Sqr(FObserverPosition.Z * (1 / 16)))    else      tessFrameVarianceDelta := 0;  end;  s := FPatchSize;  TessCurrentVariance := @FTLVariance[0];  TessMaxVariance := FMaxTLVarianceDepth;  Result := RecursTessellate(FTLNode, 1, VertexDist(0, s), VertexDist(s, 0), VertexDist(0, 0));  TessCurrentVariance := @FBRVariance[0];  TessMaxVariance := FMaxBRVarianceDepth;  if Result then    Result := RecursTessellate(FBRNode, 1, VertexDist(s, 0), VertexDist(0, s), VertexDist(s, s));end;function TGLROAMPatch.SafeTesselate: Boolean;var  Fail: Boolean;begin  Result := False;  Fail := True;  repeat    try      Result := Tesselate;      Fail := False;    except      on e: EGLROAMException do      begin        // Nothing to do, just wait the next iteration        Fail := True;      end;    end;  until not Fail;end;procedure TGLROAMPatch.RenderHighRes(Vertices: TGLAffineVectorList;  VertexIndices: TGLIntegerList; TexCoords: TGLTexPointList; ForceROAM: Boolean);var  Primitive: Cardinal;begin  // Prepare display list if needed  if FListHandle.Handle = 0 then  begin    // either use brute-force strips or a high-res static tesselation    if ForceROAM then    begin      ResetTessellation;      SafeTesselate;      RenderROAM(Vertices, VertexIndices, TexCoords);      Primitive := GL_TRIANGLES;      FTriangleCount := VertexIndices.Count div 3;    end    else    begin      RenderAsStrips(Vertices, VertexIndices, TexCoords);      Primitive := GL_TRIANGLE_STRIP;      FTriangleCount := VertexIndices.Count - 2 * FPatchSize;    end;    Vertices.Translate(VertexOffset);    TexCoords.ScaleAndTranslate(PTexPoint(@TextureScale)^,      PTexPoint(@TextureOffset)^);    gl.VertexPointer(3, GL_FLOAT, 0, Vertices.List);    xgl.TexCoordPointer(2, GL_FLOAT, 0, TexCoords.List);    FListHandle.AllocateHandle;    gl.NewList(FListHandle.Handle, GL_COMPILE);      gl.DrawElements(Primitive, VertexIndices.Count, GL_UNSIGNED_INT, VertexIndices.List);    gl.EndList;    DrawContours(Vertices, VertexIndices, FContourInterval, FContourWidth, 1);    Vertices.Count := 0;    TexCoords.Count := 0;    VertexIndices.Count := 0;  end;  // perform the render  gl.CallList(FListHandle.Handle);end;procedure TGLROAMPatch.RenderAccum(Vertices: TGLAffineVectorList;  VertexIndices: TGLIntegerList; TexCoords: TGLTexPointList;  AutoFlushVertexCount: Integer);var  OcclusionPassed: Boolean;  n, nb, nvi: Integer;begin  // CLOD tiles are rendered via ROAM  if (FOcclusionSkip > 0) and FOcclusionQuery.IsSupported then  begin    if FOcclusionQuery.Handle = 0 then    begin      FOcclusionQuery.AllocateHandle;      FOcclusionCounter := -(ID mod (FOcclusionSkip));    end;    OcclusionPassed := (FOcclusionCounter <= 0) or (FOcclusionQuery.PixelCount > 0);    Dec(FOcclusionCounter);    if OcclusionPassed then    begin      if FOcclusionCounter <= 0 then        Inc(FOcclusionCounter, FOcclusionSkip);      FOcclusionQuery.BeginQuery;    end;  end  else    OcclusionPassed := True;  FLastOcclusionTestPassed := OcclusionPassed;  if OcclusionPassed then  begin    nvi := VertexIndices.Count;    n := Vertices.Count;    RenderROAM(Vertices, VertexIndices, TexCoords);    nb := Vertices.Count - n;    FTriangleCount := (VertexIndices.Count - nvi) div 3;    Vertices.Translate(VertexOffset, n, nb);    TexCoords.ScaleAndTranslate(PTexPoint(@TextureScale)^,      PTexPoint(@TextureOffset)^, n, nb);    DrawContours(Vertices, VertexIndices, FContourInterval, FContourWidth, 3);    if FOcclusionQuery.Active then    begin      FlushAccum(Vertices, VertexIndices, TexCoords);      FOcclusionQuery.EndQuery;    end    else if VertexIndices.Count > AutoFlushVertexCount then      FlushAccum(Vertices, VertexIndices, TexCoords);  end  else    FTriangleCount := 0;end;class procedure TGLROAMPatch.FlushAccum(Vertices: TGLAffineVectorList;  VertexIndices: TGLIntegerList; TexCoords: TGLTexPointList);begin  if VertexIndices.Count = 0 then    Exit;  if gl.ARB_vertex_buffer_object then  begin    FVBOVertHandle.AllocateHandle;    FVBOVertHandle.BindBufferData(Vertices.List, Vertices.DataSize,      GL_STREAM_DRAW_ARB);    gl.VertexPointer(3, GL_FLOAT, 0, nil);    FVBOTexHandle.AllocateHandle;    FVBOTexHandle.BindBufferData(TexCoords.List, TexCoords.DataSize,      GL_STREAM_DRAW_ARB);    xgl.TexCoordPointer(2, GL_FLOAT, 0, nil);    gl.DrawRangeElements(GL_TRIANGLES, 0, Vertices.Count - 1,      VertexIndices.Count, GL_UNSIGNED_INT, VertexIndices.List);    gl.BindBuffer(GL_ARRAY_BUFFER_ARB, 0);    gl.BindBuffer(GL_ELEMENT_ARRAY_BUFFER_ARB, 0);  end  else if gl.EXT_compiled_vertex_array and gl.EXT_draw_range_elements then  begin    gl.LockArrays(0, Vertices.Count);    gl.DrawRangeElements(GL_TRIANGLES, 0, Vertices.Count - 1,      VertexIndices.Count, GL_UNSIGNED_INT, VertexIndices.List);    gl.UnLockArrays;  end  else  begin    gl.DrawElements(GL_TRIANGLES, VertexIndices.Count, GL_UNSIGNED_INT, VertexIndices.List);  end;  Vertices.Count := 0;  TexCoords.Count := 0;  VertexIndices.Count := 0;end;procedure RecursRender(const tri: PROAMTriangleNode;  const Left, Right, Apex: TROAMRenderPoint);var  Half: TROAMRenderPoint;  LocalIndices: PIntegerArray;begin  if Assigned(tri.LeftChild) then  begin // = if node is split    Half.X := (Left.X + Right.X) shr 1;    Half.Y := (Left.Y + Right.Y) shr 1;    RenderTexCoords.AddNC(@half.X);    Half.Idx := RenderVertices.AddNC(@half.X, RenderRaster[half.Y][half.X]);    RecursRender(Tri.LeftChild, Apex, Left, Half);    RecursRender(Tri.RightChild, Right, Apex, Half);  end  else  begin    LocalIndices := RenderIndices;    LocalIndices[0] := Left.Idx;    LocalIndices[1] := Apex.Idx;    LocalIndices[2] := Right.Idx;    RenderIndices := PIntegerArray(@LocalIndices[3]);  end;end;procedure TGLROAMPatch.RenderROAM(Vertices: TGLAffineVectorList;  VertexIndices: TGLIntegerList; TexCoords: TGLTexPointList);  procedure ROAMRenderPoint(var p: TROAMRenderPoint; anX, anY: Integer);  begin    p.X := anX;    p.Y := anY;    p.Idx := Vertices.Add(anX, anY, RenderRaster[anY][anX]);    TexCoords.Add(anX, anY);  end;var  rtl, rtr, rbl, rbr: TROAMRenderPoint;begin  RenderVertices := Vertices;  RenderTexCoords := TexCoords;  VertexIndices.AdjustCapacityToAtLeast(Sqr(FPatchSize) * 6 + 15000);  // this is required, the actual item count is maintained out of the list scope  VertexIndices.SetCountResetsMemory := False;  RenderIndices := @VertexIndices.List[VertexIndices.Count];  RenderRaster := FHeightData.SmallIntRaster;  ROAMRenderPoint(rtl, 0, 0);  ROAMRenderPoint(rtr, FPatchSize, 0);  ROAMRenderPoint(rbl, 0, FPatchSize);  ROAMRenderPoint(rbr, FPatchSize, FPatchSize);  RecursRender(FTLNode, rbl, rtr, rtl);  RecursRender(FBRNode, rtr, rbl, rbr);  VertexIndices.Count := (Cardinal(RenderIndices) - Cardinal(VertexIndices.List)) div SizeOf(Integer);end;procedure TGLROAMPatch.RenderAsStrips(Vertices: TGLAffineVectorList;  VertexIndices: TGLIntegerList; TexCoords: TGLTexPointList);var  X, Y, baseTop, rowLength: Integer;  p: TAffineVector;  Row: PSmallIntArray;  raster: PSmallIntRaster;  Tex: TTexPoint;  VerticesList: PAffineVector;  TexCoordsList: PTexPoint;  IndicesList: PInteger;begin  raster := FHeightData.SmallIntRaster;  rowLength := FPatchSize + 1;  // prepare vertex data  Vertices.Count := Sqr(rowLength);  VerticesList := PAffineVector(Vertices.List);  TexCoords.Count := Sqr(rowLength);  TexCoordsList := PTexPoint(TexCoords.List);  for Y := 0 to FPatchSize do  begin    p.Y := Y;    Tex.T := p.Y;    Row := raster[Y];    for X := 0 to FPatchSize do    begin      p.X := X;      Tex.s := p.X;      p.Z := Row[X];      VerticesList^ := p;      Inc(VerticesList);      TexCoordsList^ := Tex;      Inc(TexCoordsList);    end;  end;  // build indices list  baseTop := 0;  VertexIndices.Count := (rowLength * 2 + 2) * FPatchSize - 1;  IndicesList := PInteger(VertexIndices.List);  Y := 0;  while Y < FPatchSize do  begin    if Y > 0 then    begin      IndicesList^ := baseTop + FPatchSize;      Inc(IndicesList);    end;    for X := baseTop + FPatchSize downto baseTop do    begin      IndicesList^ := X;      PIntegerArray(IndicesList)[1] := rowLength + X;      Inc(IndicesList, 2);    end;    IndicesList^ := baseTop + rowLength;    Inc(baseTop, rowLength);    PIntegerArray(IndicesList)[1] := baseTop + rowLength;    Inc(IndicesList, 2);    for X := baseTop to baseTop + FPatchSize do    begin      IndicesList^ := rowLength + X;      PIntegerArray(IndicesList)[1] := X;      Inc(IndicesList, 2);    end;    IndicesList^ := baseTop + FPatchSize;    Inc(IndicesList);    Inc(baseTop, rowLength);    Inc(Y, 2);  end;  VertexIndices.Count := VertexIndices.Count - 1;end;// ------------------------------------------------------------------initialization// ------------------------------------------------------------------FVBOVertHandle := TGLVBOArrayBufferHandle.Create;FVBOTexHandle := TGLVBOArrayBufferHandle.Create;FVBOIndicesHandle := TGLVBOElementArrayHandle.Create;//---------------------------------------finalization//---------------------------------------FVBOVertHandle.Free;FVBOTexHandle.Free;FVBOIndicesHandle.Free;SetROAMTrianglesCapacity(0);end.
 |