/usr/src/castle-game-engine-5.2.0/base/castleutils_miscella.inc 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 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302 303 304 305 306 307 308 309 310 | {
Copyright 2002-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.
----------------------------------------------------------------------------
}
{ Some miscellaneous things, too small to be included in separate files. }
{$ifdef read_interface}
{ OS constants ---------------------------------------- }
const
{ New line. Short name for LineEnding. }
NL = LineEnding;
{ basic types ---------------------------------------- }
type
{$ifdef FPC}
{ }
PByteArray = ^TByteArray;
TByteArray = array[0..MaxInt div SizeOf(Byte)-1] of Byte;
{$endif}
TArray_PChar = array[0..MaxInt div SizeOf(PChar)-1]of PChar;
PArray_PChar=^TArray_PChar;
TArray_TObject = array[0..MaxInt div SizeOf(Pointer)-1]of TObject;
PArray_TObject=^TArray_TObject;
PString = ^AnsiString;
{ Pointer to TObject.
Don't call this PObject or PTObject to avoid possible name clashes
with other units (pointers are often used in situations that prevent
good type-checking, so better to avoid name clashes to avoid some
nasty errors). }
PtrObject = ^TObject;
(*Class to be raised (like an exception) and caught, to exit from some
code blocks.
When deriving new TCodeBreaker descendant, specify always where it's caught.
It should be caught and silenced (without causing any errors) in appropriate
place. This way doing @code(raise BreakXxx.Create) is a way to exit from
given code part.
This class does not inherit from @code(Exception) because it does not
signal some errorneous situation (it must always be caught and silenced),
also it doesn't need a Message property. *)
TCodeBreaker = class(TObject);
{ Internal error in the program. Something that definitely should not
happen and indicates a bug that should be reported.
This exception must be shown to user (never silently ignored),
and usually the safest thing to do is to close the program at this point.
We will add to your error message (passed to the constructor) some
instructive message, like @code('Internal error occurred, please
submit this to the author : '). }
EInternalError = class(Exception)
constructor Create(const s: string);
constructor CreateFmt(const s: string; const Args: array of const);
end;
ECheckFailed = class(Exception);
{ Check condition.
@raises ECheckFailed Raised with ErrMessage if condition TrueValue if false. }
procedure Check(TrueValue: boolean; const ErrMessage: string = 'Check failed');
{ Arrays searching ----------------------------------------------------------- }
{ Search the array for a given value.
Returns index (zero-based) or -1 if not found.
Useful for writing case as:
@longCode(#
case ArrayPosStr(variable, [val1, val2]) of
0 : Something1;
1 : Something2;
else SomethingElse;
end;
#) }
function ArrayPosStr(const A: string; const Arr: array of string): Integer; overload;
function ArrayPosText(const A: string; const Arr: array of string;
IgnoreCase: boolean = true): Integer; overload;
function PArrayPosStr(const A: string; Arr: PString; ArrCount: Integer): Integer; overload;
function PArrayPosText(const A: string; Arr: PString; ArrCount: Integer;
IgnoreCase: boolean = true): Integer; overload;
{ Iff functions ------------------------------------------------------------- }
{ }
function Iff(boolval: boolean; trueval, falseval: string) : string; overload;
function Iff(boolval: boolean; trueval, falseval: Integer) : Integer; overload;
function Iff(boolval: boolean; trueval, falseval: Float) : Float; overload;
function Iff(boolval: boolean; trueval, falseval: Cardinal): Cardinal; overload;
function Iff(boolval: boolean; trueval, falseval: char) : char; overload;
{ Some helpful consts -------------------------------------------------------- }
{$ifdef FPC}
{ Describe FPC version. In the form 'version.release.patch'.
This is actually a constant (for every run of a program it has
always the same value) but I can't declare it as a Pascal constant
because it must use "Format" function that is not allowed in constant
expressions. }
function SFPCVersion: string;
{$endif}
{ Short name and version of Pascal compiler used
to compile this unit. It is a constant, actually, but I cannot
declare it as a constant because it must call SFPCVersion that
is not declared as a constant. }
function SCompilerDescription: string;
{ Print some common info for programs released on
[http://castle-engine.sourceforge.net/].
This is useful only for programs released on this WWW page by Michalis.
Resulting string is multiline.
@param(DisplayApplicationName Usually ApplicationName, but you can give
here something else if you want.)
@param(Version For my programs this usually looks like '%d.%d.%d'
and conforms to [http://castle-engine.sourceforge.net/versioning.php])
@param(WrapLines If true then resulting string will not have lines
longer than 80 characters. Suitable for printing program
help message on stdout, e.g. in response to @--help option.) }
function SCastleEngineProgramHelpSuffix(const DisplayApplicationName: string;
const Version: string; WrapLines: boolean): string;
{ If not TrueValue then RaiseLastOSError. }
procedure OSCheck(TrueValue: boolean); overload;
procedure OSCheck(TrueValue: boolean; const Place: string); overload;
{$endif read_interface}
{$ifdef read_implementation}
constructor EInternalError.Create(const s: string);
begin
inherited Create('Internal error: something that should not ever happen... happened. ' +
'This indicates a bug in the program''s code (or something related, ' +
'like a shared library, that should be workarounded from the program then). ' +
{ 'Don''t panic - if you see this note then at least my code noticed the '+
'bug in some way, so it may be something simple and I can probably fix it '+
'(or it''s external bug and I am not responsible for this...). '+}
'We definitely want you to report this as a bug! Please send this error-specific info: ' + S);
end;
constructor EInternalError.CreateFmt(const s: string; const Args: array of const);
begin
Create(Format(S, Args));
end;
procedure Check(TrueValue: boolean; const errMessage: string);
begin
if not TrueValue then raise ECheckFailed.Create(errMessage);
end;
{ arrays searching ---------------------------------------- }
function ArrayPosStr(const A: string; const Arr: array of string): Integer;
begin
for Result := 0 to High(Arr) do
if Arr[result] = A then Exit;
Result := -1;
end;
function ArrayPosText(const A: string; const Arr: array of string;
IgnoreCase: boolean): Integer;
begin
Result := PArrayPosText(A, @Arr, High(Arr)+1, IgnoreCase);
end;
function PArrayPosStr(const A: string; Arr: PString; ArrCount: Integer): Integer;
begin
for result := 0 to ArrCount-1 do
begin
if Arr^=A then exit;
Inc(Arr);
end;
result := -1;
end;
function PArrayPosText(const A: string; Arr: PString; ArrCount: Integer;
IgnoreCase: boolean): Integer;
var AA: string;
begin
if IgnoreCase then
begin
AA := AnsiUpperCase(A);
for result := 0 to ArrCount-1 do
begin if AnsiUpperCase(Arr^) = AA then exit; Inc(Arr) end;
result := -1;
end else
begin
for result := 0 to ArrCount-1 do
begin if Arr^=A then exit; Inc(Arr) end;
result := -1;
end;
end;
{ Iff functions ------------------------------------------------------------- }
function Iff(boolval: boolean; trueval, falseval: string) : string; begin if boolval then result := trueval else result := falseval end;
function Iff(boolval: boolean; trueval, falseval: Integer) : Integer; begin if boolval then result := trueval else result := falseval end;
function Iff(boolval: boolean; trueval, falseval: Float) : Float; begin if boolval then result := trueval else result := falseval end;
function Iff(boolval: boolean; trueval, falseval: Cardinal): Cardinal; begin if boolval then result := trueval else result := falseval end;
function Iff(boolval: boolean; trueval, falseval: char) : char; begin if boolval then result := trueval else result := falseval end;
{ Helpful consts ------------------------------------------------------------ }
{$ifdef FPC}
function SFPCVersion: string;
begin
Result := Format('%d.%d.%d', [FPC_VERSION, FPC_RELEASE, FPC_PATCH]);
(*
Notes about old ideas how to implement it to be able to declare it
as a constant:
As some time, I used this code:
{$ifdef VER1_0_6} '1.0.6' {$endif}
{$ifdef VER1_0_10} '1.0.10' {$endif}
{$ifdef VER1_9_4} '1.9.4' {$endif}
{$ifdef VER1_9_5} '1.9.5' {$endif}
{$ifdef VER1_9_6} '1.9.6' {$endif}
{$ifdef VER1_9_7} '1.9.7' {$endif};
but this obviously had to be fixed for each new FPC version.
There's also this trick to define it as a constant,
but it works only when there are only single digits in version
(e.g. it wouldn't work for FPC 1.0.10):
Chr(Ord(FPC_VERSION)+Ord('0')) +'.' +
Chr(Ord(FPC_RELEASE)+Ord('0')) +'.' +
Chr(Ord(FPC_PATCH)+Ord('0'));
After some time, I decided to implement SFPCVersion cleanly
as a function. *)
end;
{$endif FPC}
function SCompilerDescription: string;
begin
Result :=
{$ifdef FPC}
'Free Pascal Compiler ' + SFPCVersion
{$else} {$ifdef DELPHI}
{$ifdef WIN32} 'Delphi'
{$ifdef VER140} + ' 6' {$endif}
{$ifdef VER150} + ' 7' {$endif}
{$else} 'Kylix'
{$endif}
{$else} {$ifdef __GPC__}
'GNU Pascal'
{$else}
Undefined compiler.
{$endif} {$endif} {$endif} ;
end;
function SCastleEngineProgramHelpSuffix(const DisplayApplicationName: string;
const Version: string; WrapLines: boolean): string;
begin
Result :=
DisplayApplicationName + ' version ' + Version + '.' +nl+
'Author: Michalis Kamburelis, aka Kambi <michalis.kambi@gmail.com>' +nl+
'See http://castle-engine.sourceforge.net/ for latest versions' +
Iff(WrapLines, nl + ' ', '') +
' of this program, sources, documentation etc.' +nl+
'Compiled with ' + SCompilerDescription +'.';
end;
procedure OSCheck(TrueValue: boolean);
begin
if not TrueValue then RaiseLastOSError;
end;
procedure OSCheck(TrueValue: boolean; const Place: string);
begin
if not TrueValue then
try
RaiseLastOSError;
except on E: EOSError do
begin
E.Message := Place + ': ' + E.Message;
raise;
end;
end;
end;
{$endif read_implementation}
|