2022-03-08 12:25:51 -05:00
|
|
|
{
|
|
|
|
Apophysis Copyright (C) 2001-2004 Mark Townsend
|
|
|
|
Apophysis Copyright (C) 2005-2006 Ronald Hordijk, Piotr Borys, Peter Sdobnov
|
|
|
|
Apophysis Copyright (C) 2007-2008 Piotr Borys, Peter Sdobnov
|
|
|
|
|
|
|
|
Apophysis "3D hack" Copyright (C) 2007-2008 Peter Sdobnov
|
|
|
|
Apophysis "7X" Copyright (C) 2009-2010 Georg Kiehne
|
2022-06-23 06:22:32 -04:00
|
|
|
Apophysis AV "Phoenix Edition" Copyright (C) 2021-2022 Alice V. Koryagina
|
2022-03-08 12:25:51 -05:00
|
|
|
|
|
|
|
This program is free software; you can redistribute it and/or modify
|
|
|
|
it under the terms of the GNU General Public License as published by
|
|
|
|
the Free Software Foundation; either version 2 of the License, or
|
|
|
|
(at your option) any later version.
|
|
|
|
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
GNU General Public License for more details.
|
|
|
|
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
|
|
along with this program; if not, write to the Free Software
|
|
|
|
Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
|
|
|
|
}
|
|
|
|
unit RenderThread;
|
|
|
|
|
|
|
|
interface
|
|
|
|
|
|
|
|
uses
|
|
|
|
Classes, Windows, Messages, Graphics,
|
|
|
|
ControlPoint, RenderingInterface,
|
|
|
|
Global, RenderingCommon, PngImage,
|
|
|
|
RenderingImplementation;
|
|
|
|
|
|
|
|
//Disabled:
|
|
|
|
|
|
|
|
//Render64, Render64MT,
|
|
|
|
//Render48, Render48MT,
|
|
|
|
//Render32f, Render32fMT;
|
|
|
|
|
|
|
|
const
|
|
|
|
WM_THREAD_COMPLETE = WM_APP + 5437;
|
|
|
|
WM_THREAD_TERMINATE = WM_APP + 5438;
|
|
|
|
|
|
|
|
type
|
|
|
|
TRenderThread = class(TThread)
|
|
|
|
private
|
|
|
|
FRenderer: TBaseRenderer;
|
|
|
|
|
|
|
|
FOnProgress: TOnProgress;
|
|
|
|
FCP: TControlPoint;
|
|
|
|
FMaxMem: int64;
|
|
|
|
FNrThreads: Integer;
|
|
|
|
FBitsPerSample: integer;
|
|
|
|
FMinDensity: double;
|
|
|
|
FOutput: TStrings;
|
|
|
|
FExportBuffer: boolean;
|
|
|
|
FCommentStr: String; // AV
|
|
|
|
|
|
|
|
procedure CreateRenderer;
|
|
|
|
function GetNrSlices: integer;
|
|
|
|
function GetSlice: integer;
|
|
|
|
procedure SetBitsPerSample(const bits: Integer);
|
|
|
|
|
|
|
|
function GetExportBuffer: boolean;
|
|
|
|
procedure SetExportBuffer(value: boolean);
|
|
|
|
|
|
|
|
procedure Trace(const str: string);
|
|
|
|
|
|
|
|
public
|
|
|
|
TargetHandle: HWND;
|
|
|
|
WaitForMore, More: boolean;
|
|
|
|
|
|
|
|
constructor Create;
|
|
|
|
destructor Destroy; override;
|
|
|
|
|
|
|
|
procedure SetCP(CP: TControlPoint);
|
|
|
|
function GetImage: TBitmap;
|
|
|
|
function GetTransparentImage: TPngObject;
|
|
|
|
procedure SaveImage(const FileName: String);
|
|
|
|
|
|
|
|
procedure Execute; override;
|
|
|
|
function GetRenderer: TBaseRenderer;
|
|
|
|
|
|
|
|
procedure Terminate;
|
|
|
|
procedure Suspend;
|
|
|
|
procedure Resume;
|
|
|
|
procedure BreakRender;
|
|
|
|
procedure HibernateRender(filePath: string);
|
|
|
|
procedure ResumeFromHibernation(filePath: string);
|
|
|
|
function EmbedText(CommentStr: String): boolean; //AV
|
|
|
|
function DoSnapshot: boolean; // AV: to make mid-render snapshots
|
|
|
|
|
|
|
|
// procedure GetBucketStats(var Stats: TBucketStats);
|
|
|
|
procedure ShowBigStats;
|
|
|
|
procedure ShowSmallStats;
|
|
|
|
|
|
|
|
property OnProgress: TOnProgress
|
|
|
|
// read FOnProgress
|
|
|
|
write FOnProgress;
|
|
|
|
|
|
|
|
property Slice: integer
|
|
|
|
read GetSlice;
|
|
|
|
property NrSlices: integer
|
|
|
|
read GetNrSlices;
|
|
|
|
property MaxMem: int64
|
|
|
|
read FMaxMem
|
|
|
|
write FMaxMem;
|
|
|
|
// property compatibility: Integer read Fcompatibility write Fcompatibility;
|
|
|
|
property NrThreads: Integer
|
|
|
|
read FNrThreads
|
|
|
|
write FNrThreads;
|
|
|
|
property BitsPerSample: Integer
|
|
|
|
read FBitsPerSample
|
|
|
|
write SetBitsPerSample;
|
|
|
|
property Output: TStrings
|
|
|
|
write FOutput;
|
|
|
|
property MinDensity: double
|
|
|
|
write FMinDensity;
|
|
|
|
property ExportBuffer: boolean
|
|
|
|
read GetExportBuffer
|
|
|
|
write SetExportBuffer;
|
|
|
|
end;
|
|
|
|
|
|
|
|
implementation
|
|
|
|
|
|
|
|
uses
|
|
|
|
Math, SysUtils,Tracer;
|
|
|
|
{ TRenderThread }
|
|
|
|
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
|
|
destructor TRenderThread.Destroy;
|
|
|
|
begin
|
|
|
|
if assigned(FRenderer) then
|
|
|
|
FRenderer.Free;
|
|
|
|
FRenderer := nil;
|
|
|
|
|
|
|
|
if assigned(FCP) then FCP.Free;
|
|
|
|
|
|
|
|
inherited;
|
|
|
|
end;
|
|
|
|
|
|
|
|
function TRenderThread.DoSnapshot: boolean; // AV
|
|
|
|
begin
|
|
|
|
Result := false;
|
|
|
|
if assigned(FRenderer) then
|
|
|
|
Result := FRenderer.DoSnapshot;
|
|
|
|
end;
|
|
|
|
|
|
|
|
function TRenderThread.GetExportBuffer: boolean;
|
|
|
|
begin
|
|
|
|
if assigned(FRenderer) then
|
|
|
|
Result := FRenderer.ExportBuffer
|
|
|
|
else Result := FExportBuffer;
|
|
|
|
end;
|
|
|
|
|
|
|
|
procedure TRenderThread.SetExportBuffer(value: boolean);
|
|
|
|
begin
|
|
|
|
if assigned(FRenderer) then
|
|
|
|
FRenderer.ExportBuffer := value;
|
|
|
|
FExportBuffer := value;
|
|
|
|
end;
|
|
|
|
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
|
|
function TRenderThread.GetImage: TBitmap;
|
|
|
|
begin
|
|
|
|
Result := nil;
|
|
|
|
if assigned(FRenderer) then
|
|
|
|
Result := FRenderer.GetImage;
|
|
|
|
end;
|
|
|
|
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
|
|
function TRenderThread.GetTransparentImage: TPngObject;
|
|
|
|
begin
|
|
|
|
Result := nil;
|
|
|
|
if assigned(FRenderer) then
|
|
|
|
Result := FRenderer.GetTransparentImage;
|
|
|
|
end;
|
|
|
|
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
|
|
procedure TRenderThread.SetCP(CP: TControlPoint);
|
|
|
|
begin
|
|
|
|
|
|
|
|
FCP := CP.Clone;
|
|
|
|
end;
|
|
|
|
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
|
|
constructor TRenderThread.Create;
|
|
|
|
begin
|
|
|
|
MaxMem := 0;
|
|
|
|
// BitsPerSample := InternalBitsPerSample; // AV: now unused
|
|
|
|
FreeOnTerminate := false;
|
|
|
|
WaitForMore := false;
|
|
|
|
|
|
|
|
inherited Create(True); // Create Suspended;
|
|
|
|
end;
|
|
|
|
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
|
|
procedure TRenderThread.CreateRenderer;
|
|
|
|
begin
|
|
|
|
if assigned(FRenderer) then begin
|
|
|
|
Trace('Destroying previous renderer (?)');
|
|
|
|
FRenderer.Free;
|
|
|
|
end;
|
|
|
|
Trace('Creating renderer');
|
|
|
|
|
|
|
|
if NrThreads <= 1 then begin
|
|
|
|
if MaxMem = 0 then begin
|
|
|
|
FRenderer := TRenderWorkerST.Create;
|
|
|
|
end else begin
|
|
|
|
FRenderer := TRenderWorkerST_MM.Create;
|
|
|
|
FRenderer.MaxMem := MaxMem;
|
|
|
|
end;
|
|
|
|
end
|
|
|
|
else begin
|
|
|
|
if MaxMem = 0 then begin
|
|
|
|
FRenderer := TRenderWorkerMT.Create;
|
|
|
|
end else begin
|
|
|
|
FRenderer := TRenderWorkerMT_MM.Create;
|
|
|
|
FRenderer.MaxMem := MaxMem;
|
|
|
|
end;
|
|
|
|
FRenderer.NumThreads := NrThreads;
|
|
|
|
end;
|
|
|
|
|
|
|
|
FRenderer.ExportBuffer := FExportbuffer;
|
|
|
|
FRenderer.SetCP(FCP);
|
|
|
|
FRenderer.MinDensity := FMinDensity;
|
|
|
|
FRenderer.OnProgress := FOnProgress;
|
|
|
|
FRenderer.Output := FOutput;
|
|
|
|
end;
|
|
|
|
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
|
|
procedure TRenderThread.Execute;
|
|
|
|
label RenderMore;
|
|
|
|
begin
|
|
|
|
CreateRenderer;
|
|
|
|
|
|
|
|
RenderMore:
|
|
|
|
assert(assigned(FRenderer));
|
|
|
|
|
|
|
|
Trace('Rendering');
|
|
|
|
FRenderer.Render;
|
|
|
|
|
|
|
|
if Terminated or FRenderer.Failed then begin
|
|
|
|
Trace('Sending WM_THREAD_TERMINATE');
|
|
|
|
PostMessage(TargetHandle, WM_THREAD_TERMINATE, 0, ThreadID);
|
|
|
|
Trace('Terminated');
|
|
|
|
exit;
|
|
|
|
end
|
|
|
|
else begin
|
|
|
|
Trace('Sending WM_THREAD_COMPLETE');
|
|
|
|
PostMessage(TargetHandle, WM_THREAD_COMPLETE, 0, ThreadID);
|
|
|
|
end;
|
|
|
|
|
|
|
|
if WaitForMore and (FRenderer <> nil) then begin
|
|
|
|
FRenderer.RenderMore := true;
|
|
|
|
|
|
|
|
Trace('Waiting for more');
|
|
|
|
inherited Suspend;
|
|
|
|
|
|
|
|
if WaitForMore then goto RenderMore;
|
|
|
|
end;
|
|
|
|
|
|
|
|
Trace('Finished');
|
|
|
|
end;
|
|
|
|
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
|
|
procedure TRenderThread.Terminate;
|
|
|
|
begin
|
|
|
|
try
|
|
|
|
if assigned(FRenderer) then
|
|
|
|
FRenderer.Stop;
|
|
|
|
except on EAccessViolation do
|
|
|
|
// nothing
|
|
|
|
end;
|
|
|
|
|
|
|
|
WaitForMore := false;
|
|
|
|
|
|
|
|
inherited Terminate;
|
|
|
|
end;
|
|
|
|
|
|
|
|
procedure TRenderThread.Suspend;
|
|
|
|
begin
|
|
|
|
if assigned(FRenderer) then FRenderer.Pause;
|
|
|
|
|
|
|
|
inherited;
|
|
|
|
end;
|
|
|
|
|
|
|
|
procedure TRenderThread.Resume;
|
|
|
|
begin
|
|
|
|
if assigned(FRenderer) then FRenderer.UnPause;
|
|
|
|
|
|
|
|
inherited;
|
|
|
|
end;
|
|
|
|
|
|
|
|
procedure TRenderThread.BreakRender;
|
|
|
|
begin
|
|
|
|
if assigned(FRenderer) then
|
|
|
|
FRenderer.BreakRender;
|
|
|
|
end;
|
|
|
|
|
|
|
|
procedure TRenderThread.HibernateRender(filePath: string);
|
|
|
|
begin
|
|
|
|
if assigned(FRenderer) then
|
|
|
|
FRenderer.Hibernate(filePath);
|
|
|
|
end;
|
|
|
|
|
|
|
|
procedure TRenderThread.ResumeFromHibernation(filePath: string);
|
|
|
|
begin
|
|
|
|
if assigned(FRenderer) then
|
|
|
|
FRenderer.Stop;
|
|
|
|
FRenderer.Resume(filePath);
|
|
|
|
FRenderer.UnPause;
|
|
|
|
end;
|
|
|
|
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
|
|
function TRenderThread.GetNrSlices: integer;
|
|
|
|
begin
|
|
|
|
if assigned(FRenderer) then
|
|
|
|
Result := FRenderer.NrSlices
|
|
|
|
else
|
|
|
|
Result := 1;
|
|
|
|
end;
|
|
|
|
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
|
|
function TRenderThread.GetSlice: integer;
|
|
|
|
begin
|
|
|
|
if assigned(FRenderer) then
|
|
|
|
Result := FRenderer.Slice
|
|
|
|
else
|
|
|
|
Result := 1;
|
|
|
|
end;
|
|
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////////
|
|
|
|
function TRenderThread.GetRenderer: TBaseRenderer;
|
|
|
|
begin
|
|
|
|
Result := FRenderer;
|
|
|
|
FRenderer := nil;
|
|
|
|
end;
|
|
|
|
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
|
|
procedure TRenderThread.SetBitsPerSample(const bits: Integer);
|
|
|
|
begin
|
|
|
|
if FRenderer = nil then FBitsPerSample := bits
|
|
|
|
else assert(false);
|
|
|
|
end;
|
|
|
|
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
|
|
procedure TRenderThread.SaveImage(const FileName: String);
|
|
|
|
begin
|
|
|
|
if assigned(FRenderer) then
|
|
|
|
begin
|
|
|
|
if FCommentStr <> '' then
|
|
|
|
FRenderer.EmbedText(FCommentStr);
|
|
|
|
FRenderer.SaveImage(FileName);
|
|
|
|
end;
|
|
|
|
end;
|
|
|
|
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
|
|
function TRenderThread.EmbedText(CommentStr: String): boolean; // AV: for parameters embedding
|
|
|
|
begin
|
|
|
|
Result := False;
|
|
|
|
FCommentStr := CommentStr;
|
|
|
|
end;
|
|
|
|
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
|
|
procedure TRenderThread.Trace(const str: string);
|
|
|
|
begin
|
|
|
|
if assigned(FOutput) and (TraceLevel >= 2) then
|
|
|
|
FOutput.Add('. . > RenderThread #' + IntToStr(ThreadID) + ': ' + str);
|
|
|
|
end;
|
|
|
|
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
|
|
procedure TRenderThread.ShowBigStats;
|
|
|
|
begin
|
|
|
|
if assigned(FRenderer) then
|
|
|
|
FRenderer.ShowBigStats;
|
|
|
|
end;
|
|
|
|
|
|
|
|
procedure TRenderThread.ShowSmallStats;
|
|
|
|
begin
|
|
|
|
if assigned(FRenderer) then
|
|
|
|
FRenderer.ShowSmallStats;
|
|
|
|
end;
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
end.
|