/usr/src/castle-game-engine-5.2.0/base/castlelog.pas is in castle-game-engine-src 5.2.0-3.
This file is owned by root:root, with mode 0o644.
The actual contents of the file can be viewed below.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 | {
Copyright 2006-2014 Michalis Kamburelis.
This file is part of "Castle Game Engine".
"Castle Game Engine" is free software; see the file COPYING.txt,
included in this distribution, for details about the copyright.
"Castle Game Engine" 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.
----------------------------------------------------------------------------
}
{ Logging. Log has to be activated in your program (nothing in the
Castle Game Engine activates it automatically) by InitializeLog.
Various units of the engine print some logging info when @link(Log) is true. }
unit CastleLog;
interface
uses Classes;
{ Is logging active. Initially no. Activate by InitializeLog. }
function Log: boolean;
{ Initialize logging.
If you leave ALogStream as @nil (default),
then we will prints log messages to StdOut, not to some external file.
This is most useful and common behavior on Unixes, where most programs
log to StdOut, and StdOut is always available.
It also removes any problems with users asking "where can I find
the log file?".
The downside is that Windows users
have to explicitly redirect StdOut of the GUI program to get the log,
i.e. run your program from command-line like "program.exe > log.txt".
Otherwise, GUI programs (with apptype GUI) do not have StdOut available
under Windows. }
procedure InitializeLog(const ProgramVersion: string = 'Unknown';
const ALogStream: TStream = nil);
{ Log message. Ignored when log is not initialized (@link(Log) is @false).
Although we check @link(Log) here, you can also check it yourself
before even calling this procedure. This way you can avoid spending time
on constructing LogMessage. }
procedure WritelnLog(const Title: string; const LogMessage: string);
{ Format and log message.
Ignored when log is not initialized (@link(Log) is @false).
This is a shortcut for @code(WritelnLog(Title, Format(LogMessageBase, Args))). }
procedure WritelnLog(const Title: string; const LogMessageBase: string;
const Args: array of const);
{ Log message, without appending newline at the end (given LogMessage
should already contain a final newline). }
procedure WriteLog(const Title: string; const LogMessage: string);
{ Log multiline message.
LogMessage may be multiline and must be terminated by final newline. }
procedure WriteLogMultiline(const Title: string; const LogMessage: string);
{ Log multiline message.
LogMessage may be multiline and must @italic(not) be terminated by
a final newline, because we will add final newline ourselves. }
procedure WritelnLogMultiline(const Title: string; const LogMessage: string);
implementation
uses CastleUtils, CastleClassUtils, CastleTimeUtils,
SysUtils {$ifdef ANDROID}, CastleAndroidLog {$endif};
{ Dump backtrace (always to StdErr for now, regardless of LogStream)
of each log.
Displaying line info requires compiling your program with -gl.
Unfortunately line info is not 100% reliable (sometimes it only works in gdb;
sometimes it does not even work in gdb, but still you can use gdb's "info symbol xxx"
to resolve addresses to method names).
Depends very much on OS, debug info type, and FPC version. }
{ $define BACKTRACE_ON_LOG}
var
FLog: boolean = false;
LogStream: TStream;
LogStreamOwned: boolean;
function Log: boolean;
begin
Result := FLog;
end;
procedure InitializeLog(const ProgramVersion: string;
const ALogStream: TStream);
var
LogFileName: string;
begin
if Log then Exit; { ignore 2nd call to InitializeLog }
LogStreamOwned := false;
if ALogStream = nil then
begin
if not IsConsole then
begin
{ Under Windows GUI program, by default write to file .log
in the current directory.
Do not try to use StdOutStream anymore. In some cases, GUI program
may have an stdout, when it is explicitly run like
"xxx.exe --debug-log > xxx.log". But do not depend on it.
Simply writing to xxx.log is more what people expect. }
LogFileName := ExpandFileName(ApplicationName + '.log');
try
LogStream := TFileStream.Create(LogFileName, fmCreate);
except
on E: EFCreateError do
{ special exception message when LogFileName non-empty
(usual case on Windows). }
raise EWithHiddenClassName.Create('Cannot create log file "' + LogFileName + '". To dump log of GUI application on Windows, you have to run the application in a directory where you have write access, for example your user or Desktop directory.') else
end;
LogStreamOwned := true;
end else
LogStream := StdOutStream;
end else
LogStream := ALogStream;
WritelnStr(LogStream, 'Log for "' + ApplicationName +
'", version ' + ProgramVersion +
'. Started on ' + DateTimeToAtStr(Now) + '.');
{ Set Log to true only once we succeded.
Otherwise (when FLog := true would be done at the beginning of
InitializeLog), if something is done in finally..end clauses surrounding
InitializeLog, and it does "WritelnLog..." then it would
try to write something to uninitialized log. }
FLog := true;
end;
procedure WriteLogRaw(const S: string); inline;
begin
if Log then
begin
WriteStr(LogStream, S); // we know that LogStream <> nil when FLog = true
{$ifdef BACKTRACE_ON_LOG}
Dump_Stack(StdErr, Get_Frame);
{$endif}
{$ifdef ANDROID}
AndroidLog(alInfo, S);
{$endif}
end;
end;
procedure WriteLog(const Title: string; const LogMessage: string);
begin
if Log then
WriteLogRaw(Title + ': ' + LogMessage);
end;
procedure WritelnLog(const Title: string; const LogMessage: string);
begin
WriteLog(Title, LogMessage + NL);
end;
procedure WritelnLog(const Title: string; const LogMessageBase: string;
const Args: array of const);
begin
WritelnLog(Title, Format(LogMessageBase, Args));
end;
procedure WriteLogMultiline(const Title: string; const LogMessage: string);
begin
if Log then
WriteLogRaw(
'-------------------- ' + Title + ' begin' + NL +
LogMessage +
'-------------------- ' + Title + ' end' + NL);
end;
procedure WritelnLogMultiline(const Title: string; const LogMessage: string);
begin
WriteLogMultiline(Title, LogMessage + NL);
end;
finalization
if LogStreamOwned then
begin
FreeAndNil(LogStream);
FLog := false;
end;
end.
|