DEADSOFTWARE

b65ab8a4af7b13fb8eb25f445f01442175050ca4
[d2df-sdl.git] / src / shared / xdynrec.pas
1 (* Copyright (C) DooM 2D:Forever Developers
2 *
3 * This program is free software: you can redistribute it and/or modify
4 * it under the terms of the GNU General Public License as published by
5 * the Free Software Foundation, either version 3 of the License, or
6 * (at your option) any later version.
7 *
8 * This program is distributed in the hope that it will be useful,
9 * but WITHOUT ANY WARRANTY; without even the implied warranty of
10 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
11 * GNU General Public License for more details.
12 *
13 * You should have received a copy of the GNU General Public License
14 * along with this program. If not, see <http://www.gnu.org/licenses/>.
15 *)
16 {$INCLUDE a_modes.inc}
17 {.$DEFINE XDYNREC_USE_FIELDHASH} // actually, it is SLOWER with this
18 unit xdynrec;
20 interface
22 uses
23 SysUtils, Variants, Classes,
24 xparser, xstreams, utils, hashtable;
27 // ////////////////////////////////////////////////////////////////////////// //
28 type
29 TDynRecException = class(Exception)
30 public
31 constructor Create (const amsg: AnsiString);
32 constructor CreateFmt (const afmt: AnsiString; const args: array of const);
33 end;
35 TDynParseException = class(TDynRecException)
36 public
37 tokLine, tokCol: Integer;
39 public
40 constructor Create (pr: TTextParser; const amsg: AnsiString);
41 constructor CreateFmt (pr: TTextParser; const afmt: AnsiString; const args: array of const);
42 end;
45 // ////////////////////////////////////////////////////////////////////////// //
46 type
47 TDynMapDef = class;
48 TDynRecord = class;
49 TDynField = class;
50 TDynEBS = class;
52 TDynFieldList = specialize TSimpleList<TDynField>;
53 TDynRecList = specialize TSimpleList<TDynRecord>;
54 TDynEBSList = specialize TSimpleList<TDynEBS>;
56 // this is base type for all scalars (and arrays)
57 TDynField = class
58 public
59 type
60 TType = (TBool, TChar, TByte, TUByte, TShort, TUShort, TInt, TUInt, TString, TPoint, TSize, TColor, TList, TTrigData);
61 // TPoint: pair of Integers
62 // TSize: pair of UShorts
63 // TList: actually, array of records
64 // TTrigData: array of mMaxDim bytes, but internally a record (mRecRef)
65 // in binary: arrays of chars are pascal shortstrings (with counter in the first byte)
67 private
68 type
69 TEBS = (TNone, TRec, TEnum, TBitSet);
71 private
72 mOwner: TDynRecord; // owner record
73 mName: AnsiString; // field name
74 mTip: AnsiString; // short tip
75 mHelp: AnsiString; // long help
76 mType: TType; // field type
77 mIVal: Integer; // for all integer types
78 mIVal2: Integer; // for point and size
79 mIVal3: Integer; // for TColor
80 mIVal4: Integer; // for TColor
81 mSVal: AnsiString; // string; for byte and char arrays
82 mRVal: TDynRecList; // for list
83 mRHash: THashStrInt; // id -> index in mRVal
84 mRecRef: TDynRecord; // for TEBS.TRec
85 mMaxDim: Integer; // for byte and char arrays; <0: not an array; 0: impossible value
86 mBinOfs: Integer; // offset in binary; <0 - none
87 mSepPosSize: Boolean; // for points and sizes, use separate fields
88 mAsT: Boolean; // for points and sizes, use separate fields, names starts with `t`
89 mDefined: Boolean;
90 mHasDefault: Boolean;
91 mWriteDef: Boolean;
92 mInternal: Boolean;
93 mNegBool: Boolean;
94 mBitSetUnique: Boolean; // bitset can contain only one value
95 mAsMonsterId: Boolean; // special hack for triggers: monster record number+1 in binary (so 0 means "none")
96 // default value
97 mDefUnparsed: AnsiString;
98 mDefSVal: AnsiString; // default string value
99 mDefIVal, mDefIVal2, mDefIVal3, mDefIVal4: Integer; // default integer values
100 mDefRecRef: TDynRecord;
101 mEBS: TEBS; // complex type type
102 mEBSTypeName: AnsiString; // name of enum, bitset or record
103 mEBSType: TObject; // either TDynRecord or TDynEBS; nil means "simple type"; nil for `TTrigData` too
105 // for binary parser
106 mRecRefId: AnsiString;
108 // for userdata
109 mTagInt: Integer;
110 mTagPtr: Pointer;
112 // for pasgen
113 mAlias: AnsiString;
115 private
116 procedure cleanup ();
118 procedure parseDefaultValue (); // parse `mDefUnparsed` to `mDefSVal`, `mDefIVal`, `mDefIVal2`, `mDefRecRef`
119 procedure fixDefaultValue (); // this will NOT clone `mDefRecRef`
120 function isDefaultValue (): Boolean;
122 function getListCount (): Integer; inline;
123 function getListItem (idx: Integer): TDynRecord; inline; overload;
124 function getListItem (const aname: AnsiString): TDynRecord; inline; overload;
126 function getRecRefIndex (): Integer;
128 function getVar (): Variant;
129 procedure setVar (val: Variant);
131 procedure setRecRef (arec: TDynRecord);
133 procedure parseDef (pr: TTextParser); // parse mapdef definition
134 function definition (): AnsiString; // generate mapdef definition
136 protected
137 // returns `true` for duplicate record id
138 function addListItem (rec: TDynRecord): Boolean; inline;
139 function removeListItem (const aid: AnsiString): TDynRecord; // returns nil or removed record
141 public
142 // get string name for the given type
143 class function getTypeName (t: TType): AnsiString;
145 public
146 constructor Create (const aname: AnsiString; atype: TType);
147 constructor Create (const aname: AnsiString; val: Variant);
148 constructor Create (pr: TTextParser);
149 destructor Destroy (); override;
151 // clone this field; register all list records in `registerIn`
152 // "registration" is required to manage record lifetime; use header record if in doubt
153 // owner will be set to `newOwner`, if it is not `nil`, or to `owner`
154 // for lists, cloning will clone all list members
155 function clone (newOwner: TDynRecord=nil; registerIn: TDynRecord=nil): TDynField;
157 // compare field values (including trigdata)
158 // WARNING: won't work for lists
159 function isSimpleEqu (fld: TDynField): Boolean;
161 // parse string value to appropriate type and set new field value
162 procedure setValue (const s: AnsiString);
164 // supports `for rec in field do` (for lists)
165 function GetEnumerator (): TDynRecList.TEnumerator; inline;
167 function getRed (): Integer; inline;
168 procedure setRed (v: Integer); inline;
170 function getGreen (): Integer; inline;
171 procedure setGreen (v: Integer); inline;
173 function getBlue (): Integer; inline;
174 procedure setBlue (v: Integer); inline;
176 function getAlpha (): Integer; inline;
177 procedure setAlpha (v: Integer); inline;
179 public
180 // text parser and writer
181 procedure parseValue (pr: TTextParser);
182 procedure writeTo (wr: TTextWriter);
184 // binary parser and writer (DO NOT USE!)
185 procedure parseBinValue (st: TStream);
186 procedure writeBinTo (var hasLostData: Boolean; st: TStream);
188 public
189 // the following functions are here only for 'mapgen'! DO NOT USE!
190 // build "alias name" for pascal code
191 function palias (firstUp: Boolean=false): AnsiString;
193 public
194 property owner: TDynRecord read mOwner;
195 property name: AnsiString read mName; // field name
196 property baseType: TType read mType; // field type (base for arrays)
197 property defined: Boolean read mDefined; // was field value set to something by external code?
198 property internal: Boolean read mInternal write mInternal; // internal field?
199 property ival: Integer read mIVal; // integer value for int field (for speed), first field (x/w) for `TPoint` and `TSize`
200 property ival2: Integer read mIVal2; // for `TPoint` and `TSize`, this is second field (y/h)
201 property ival3: Integer read mIVal3; // for `TColor`: blue
202 property ival4: Integer read mIVal4; // for `TColor`: alpha
203 property red: Integer read getRed write setRed; // for `TColor`: red
204 property green: Integer read getGreen write setGreen; // for `TColor`: green
205 property blue: Integer read getBlue write setBlue; // for `TColor`: blue
206 property alpha: Integer read getAlpha write setAlpha; // for `TColor`: alpha
207 property sval: AnsiString read mSVal; // string value for string field (for speed)
208 property hasDefault: Boolean read mHasDefault; // `true` if this field has default value in mapdef
209 property defsval: AnsiString read mDefSVal; // string representation of default value
210 property ebs: TEBS read mEBS; // what kind of reference is this? none, enum, bitset, record
211 property ebstype: TObject read mEBSType; // reference type (nil, TDynRecord, TDynEBS); WARNING: don't modify type!
212 property ebstypename: AnsiString read mEBSTypeName; // enum/bitset name
213 property recref: TDynRecord read mRecRef write setRecRef; // referenced record (actual one, you can modify it)
214 property recrefIndex: Integer read getRecRefIndex; // index of referenced record in header; -1: not found
215 // for record lists
216 property count: Integer read getListCount;
217 property itemAt[idx: Integer]: TDynRecord read getListItem;
218 property item[const aname: AnsiString]: TDynRecord read getListItem; default; // alas, FPC 3+ lost property overloading feature
219 // field value as Variant
220 property value: Variant read getVar write setVar;
222 property tip: AnsiString read mTip;
223 property help: AnsiString read mHelp;
225 public
226 // userdata (you can use these properties as you want to; they won't be written or read to files)
227 property tagInt: Integer read mTagInt write mTagInt;
228 property tagPtr: Pointer read mTagPtr write mTagPtr;
230 public
231 // the following properties are here only for 'mapgen'! DO NOT USE!
232 property negbool: Boolean read mNegBool;
233 property hasTPrefix: Boolean read mAsT;
234 property separatePasFields: Boolean read mSepPosSize;
235 property binOfs: Integer read mBinOfs;
236 property equToDefault: Boolean read isDefaultValue;
237 end;
240 // record, either with actual values, or with type definitions
241 TDynRecord = class
242 private
243 mOwner: TDynMapDef;
244 mId: AnsiString;
245 mTypeName: AnsiString;
246 mTip: AnsiString; // short tip
247 mHelp: AnsiString; // long help
248 mSize: Integer;
249 mFields: TDynFieldList;
250 {$IF DEFINED(XDYNREC_USE_FIELDHASH)}
251 mFieldsHash: THashStrInt; // id -> index in mRVal
252 {$ENDIF}
253 mTrigTypes: array of AnsiString; // if this is triggerdata, we'll hold list of triggers here
254 mHeader: Boolean; // true for header record
255 mBinBlock: Integer; // -1: none
256 mHeaderRec: TDynRecord; // for "value" records this is header record with data, for "type" records this is header type record
258 // for userdata
259 mTagInt: Integer;
260 mTagPtr: Pointer;
262 mRec2Free: TDynRecList;
264 private
265 procedure parseDef (pr: TTextParser); // parse definition
266 function definition (): AnsiString;
268 function findByName (const aname: AnsiString): Integer; inline;
269 function hasByName (const aname: AnsiString): Boolean; inline;
270 function getFieldByName (const aname: AnsiString): TDynField; inline;
271 function getFieldAt (idx: Integer): TDynField; inline;
272 function getCount (): Integer; inline;
274 function getIsTrigData (): Boolean; inline;
275 function getIsForTrig (const aname: AnsiString): Boolean; inline;
277 function getForTrigCount (): Integer; inline;
278 function getForTrigAt (idx: Integer): AnsiString; inline;
280 procedure regrec (rec: TDynRecord);
282 function getUserVar (const aname: AnsiString): Variant;
283 procedure setUserVar (const aname: AnsiString; val: Variant);
285 procedure clearRefRecs (rec: TDynRecord);
287 protected
288 function findRecordByTypeId (const atypename, aid: AnsiString): TDynRecord;
289 function findRecordNumByType (const atypename: AnsiString; rc: TDynRecord): Integer;
290 function addRecordByType (const atypename: AnsiString; rc: TDynRecord): Boolean; // `true`: duplicate record id
292 procedure addField (fld: TDynField); inline;
293 function addFieldChecked (fld: TDynField): Boolean; inline; // `true`: duplicate name
295 public
296 constructor Create ();
297 constructor Create (pr: TTextParser); // parse definition
298 destructor Destroy (); override;
300 // clone this record; register all list records in `registerIn`
301 // "registration" is required to manage record lifetime; use header record if in doubt
302 // all fields are cloned too
303 function clone (registerIn: TDynRecord): TDynRecord;
305 // compare records (values of all fields, including trigdata)
306 // WARNING: won't work for records with list fields
307 function isSimpleEqu (rec: TDynRecord): Boolean;
309 // find field with `TriggerType` type
310 function trigTypeField (): TDynField;
312 // number of records of the given instance
313 function instanceCount (const atypename: AnsiString): Integer;
315 // only for headers: create new record with the given type
316 // will return cloned record ready for use, or `nil` on unknown type name
317 // `aid` must not be empty, and must be unique
318 function newTypedRecord (const atypename, aid: AnsiString): TDynRecord;
320 // remove record with the given type and id
321 // return `true` if record was successfully found and removed
322 // this will do all necessary recref cleanup too
323 // WARNING: not tested yet
324 function removeTypedRecord (const atypename, aid: AnsiString): Boolean;
326 //TODO:
327 // [.] API to create triggers
328 // [.] API to properly remove triggers (remove trigdata)
329 // [.] check if `removeTypedRecord()` does the right thing with inline records
330 // [.] for fields: assigning `recref` should remove previously assigned inline record (record without id)
331 // [.] other API i forgot
333 public
334 // text parser
335 // `beginEaten`: `true` if "{" was eaten
336 procedure parseValue (pr: TTextParser; beginEaten: Boolean=false);
338 // text writer
339 // `putHeader`: `true` to write complete header, otherwise only "{...}"
340 procedure writeTo (wr: TTextWriter; putHeader: Boolean=true);
342 // binary parser and writer (DO NOT USE!)
343 procedure parseBinValue (st: TStream; forceData: Boolean=false);
344 procedure writeBinTo (var hasLostData: Boolean; st: TStream; trigbufsz: Integer=-1; onlyFields: Boolean=false);
346 public
347 property mapdef: TDynMapDef read mOwner;
348 property id: AnsiString read mId; // record id in text map
349 property typeName: AnsiString read mTypeName; // record type name (like "panel", or "trigger")
350 property has[const aname: AnsiString]: Boolean read hasByName; // do we have field with the given name?
351 property count: Integer read getCount; // number of fields in this record
352 property field[const aname: AnsiString]: TDynField read getFieldByName; default; // get field by name
353 property fieldAt[idx: Integer]: TDynField read getFieldAt; // get field at the given index
354 property isTrigData: Boolean read getIsTrigData; // is this special "TriggerData" record?
355 property isForTrig[const aname: AnsiString]: Boolean read getIsForTrig; // can this "TriggerData" be used for the trigger with the given type?
356 property forTrigCount: Integer read getForTrigCount; // number of trigger type names for "TriggerData"
357 property forTrigAt[idx: Integer]: AnsiString read getForTrigAt; // trigger type name at the given index for "TriggerData"
358 property headerRec: TDynRecord read mHeaderRec; // get header record for this one (header contains all other records, enums, bitsets, etc.)
359 property isHeader: Boolean read mHeader; // is this a header record?
361 property tip: AnsiString read mTip;
362 property help: AnsiString read mHelp;
364 public
365 // user fields; user can add arbitrary custom fields
366 // by default, any user field will be marked as "internal"
367 // note: you can use this to manipulate non-user fields too
368 property user[const aname: AnsiString]: Variant read getUserVar write setUserVar;
370 public
371 // userdata (you can use these properties as you want to; they won't be written or read to files)
372 property tagInt: Integer read mTagInt write mTagInt;
373 property tagPtr: Pointer read mTagPtr write mTagPtr;
374 end;
377 // bitset/enum definition
378 TDynEBS = class
379 private
380 mOwner: TDynMapDef;
381 mIsEnum: Boolean;
382 mTypeName: AnsiString;
383 mTip: AnsiString; // short tip
384 mHelp: AnsiString; // long help
385 mIds: array of AnsiString;
386 mVals: array of Integer;
387 mMaxName: AnsiString; // MAX field
388 mMaxVal: Integer; // max value
390 private
391 procedure cleanup ();
393 procedure parseDef (pr: TTextParser); // parse definition
395 function findByName (const aname: AnsiString): Integer; inline;
396 function hasByName (const aname: AnsiString): Boolean; inline;
397 function getFieldByName (const aname: AnsiString): Integer; inline;
399 function definition (): AnsiString;
400 function pasdef (): AnsiString;
402 public
403 constructor Create (pr: TTextParser); // parse definition
404 destructor Destroy (); override;
406 // find name for the given value
407 // return empty string if not found
408 function nameByValue (v: Integer): AnsiString;
410 public
411 property mapdef: TDynMapDef read mOwner;
412 property typeName: AnsiString read mTypeName; // enum/bitset type name
413 property isEnum: Boolean read mIsEnum; // is this enum? `false` means "bitset"
414 property has[const aname: AnsiString]: Boolean read hasByName;
415 property field[const aname: AnsiString]: Integer read getFieldByName;
417 property tip: AnsiString read mTip;
418 property help: AnsiString read mHelp;
419 end;
422 // parsed "mapdef.txt"
423 TDynMapDef = class
424 public
425 recTypes: TDynRecList; // [0] is always header
426 trigTypes: TDynRecList; // trigdata
427 ebsTypes: TDynEBSList; // enums, bitsets
429 private
430 procedure parseDef (pr: TTextParser);
432 function getHeaderRecType (): TDynRecord; inline;
434 function getRecTypeCount (): Integer; inline;
435 function getRecTypeAt (idx: Integer): TDynRecord; inline;
437 function getEBSTypeCount (): Integer; inline;
438 function getEBSTypeAt (idx: Integer): TDynEBS; inline;
440 function getTrigTypeCount (): Integer; inline;
441 function getTrigTypeAt (idx: Integer): TDynRecord; inline;
443 // creates new header record
444 function parseTextMap (pr: TTextParser): TDynRecord;
446 // creates new header record
447 function parseBinMap (st: TStream): TDynRecord;
449 public
450 constructor Create (pr: TTextParser); // parses data definition
451 destructor Destroy (); override;
453 function findRecType (const aname: AnsiString): TDynRecord;
454 function findTrigFor (const aname: AnsiString): TDynRecord;
455 function findEBSType (const aname: AnsiString): TDynEBS;
457 public
458 // parse text or binary map, return new header record
459 // WARNING! stream must be seekable
460 function parseMap (st: TStream; wasBinary: PBoolean=nil): TDynRecord;
462 // returns `true` if the given stream can be a map file
463 // stream position is 0 on return
464 // WARNING! stream must be seekable
465 class function canBeMap (st: TStream): Boolean;
467 public
468 // the following functions are here only for 'mapgen'! DO NOT USE!
469 function pasdefconst (): AnsiString;
471 public
472 property headerType: TDynRecord read getHeaderRecType;
473 // for record types
474 property recTypeCount: Integer read getRecTypeCount;
475 property recTypeAt[idx: Integer]: TDynRecord read getRecTypeAt;
476 property recType[const aname: AnsiString]: TDynRecord read findRecType;
477 // for enum/bitset types
478 property ebsTypeCount: Integer read getEBSTypeCount;
479 property ebsTypeAt[idx: Integer]: TDynEBS read getEBSTypeAt;
480 property ebsType[const aname: AnsiString]: TDynEBS read findEBSType;
481 // for trigtypes
482 property trigTypeCount: Integer read getTrigTypeCount;
483 property trigTypeAt[idx: Integer]: TDynRecord read getTrigTypeAt;
484 property trigTypeFor[const aname: AnsiString]: TDynRecord read findTrigFor;
485 end;
488 {$IF DEFINED(D2D_DYNREC_PROFILER)}
489 procedure xdynDumpProfiles ();
490 {$ENDIF}
493 implementation
495 uses
496 e_log
497 {$IF DEFINED(D2D_DYNREC_PROFILER)},xprofiler{$ENDIF};
500 // ////////////////////////////////////////////////////////////////////////// //
501 function StrEqu (const a, b: AnsiString): Boolean; inline; begin result := (a = b); end;
504 // ////////////////////////////////////////////////////////////////////////// //
505 constructor TDynRecException.Create (const amsg: AnsiString);
506 begin
507 inherited Create(amsg);
508 end;
510 constructor TDynRecException.CreateFmt (const afmt: AnsiString; const args: array of const);
511 begin
512 inherited Create(formatstrf(afmt, args));
513 end;
516 // ////////////////////////////////////////////////////////////////////////// //
517 constructor TDynParseException.Create (pr: TTextParser; const amsg: AnsiString);
518 begin
519 if (pr <> nil) then begin tokLine := pr.tokLine; tokCol := pr.tokCol; end else begin tokLine := 0; tokCol := 0; end;
520 inherited Create(amsg);
521 end;
523 constructor TDynParseException.CreateFmt (pr: TTextParser; const afmt: AnsiString; const args: array of const);
524 begin
525 if (pr <> nil) then begin tokLine := pr.tokLine; tokCol := pr.tokCol; end else begin tokLine := 0; tokCol := 0; end;
526 inherited Create(formatstrf(afmt, args));
527 end;
530 // ////////////////////////////////////////////////////////////////////////// //
531 function TDynField.GetEnumerator (): TDynRecList.TEnumerator; inline;
532 begin
533 //result := TListEnumerator.Create(mRVal);
534 if (mRVal <> nil) then result := mRVal.GetEnumerator else result := TDynRecList.TEnumerator.Create(nil, 0);
535 end;
538 // ////////////////////////////////////////////////////////////////////////// //
539 constructor TDynField.Create (const aname: AnsiString; atype: TType);
540 begin
541 mRVal := nil;
542 mRecRef := nil;
543 mRHash := nil;
544 cleanup();
545 mName := aname;
546 mType := atype;
547 if (mType = TType.TList) then
548 begin
549 mRVal := TDynRecList.Create();
550 mRHash := hashNewStrInt();
551 end;
552 end;
555 constructor TDynField.Create (pr: TTextParser);
556 begin
557 cleanup();
558 parseDef(pr);
559 end;
562 constructor TDynField.Create (const aname: AnsiString; val: Variant);
563 procedure setInt32 (v: LongInt);
564 begin
565 case mType of
566 TType.TBool:
567 if (v = 0) then mIVal := 0
568 else if (v = 1) then mIVal := 1
569 else raise TDynRecException.Create('cannot convert shortint variant to field value');
570 TType.TByte:
571 if (v >= -128) and (v <= 127) then mIVal := v
572 else raise TDynRecException.Create('cannot convert shortint variant to field value');
573 TType.TUByte:
574 if (v >= 0) and (v <= 255) then mIVal := v
575 else raise TDynRecException.Create('cannot convert shortint variant to field value');
576 TType.TShort:
577 if (v >= -32768) and (v <= 32767) then mIVal := v
578 else raise TDynRecException.Create('cannot convert shortint variant to field value');
579 TType.TUShort:
580 if (v >= 0) and (v <= 65535) then mIVal := v
581 else raise TDynRecException.Create('cannot convert shortint variant to field value');
582 TType.TInt:
583 mIVal := v;
584 TType.TUInt:
585 mIVal := v;
586 TType.TString:
587 mSVal := formatstrf('%s', [v]);
588 else
589 raise TDynRecException.Create('cannot convert integral variant to field value');
590 end;
591 end;
592 begin
593 mRVal := nil;
594 mRecRef := nil;
595 mRHash := nil;
596 cleanup();
597 mName := aname;
598 case varType(val) of
599 varEmpty: raise TDynRecException.Create('cannot convert empty variant to field value');
600 varNull: raise TDynRecException.Create('cannot convert null variant to field value');
601 varSingle: raise TDynRecException.Create('cannot convert single variant to field value');
602 varDouble: raise TDynRecException.Create('cannot convert double variant to field value');
603 varDecimal: raise TDynRecException.Create('cannot convert decimal variant to field value');
604 varCurrency: raise TDynRecException.Create('cannot convert currency variant to field value');
605 varDate: raise TDynRecException.Create('cannot convert date variant to field value');
606 varOleStr: raise TDynRecException.Create('cannot convert olestr variant to field value');
607 varStrArg: raise TDynRecException.Create('cannot convert stdarg variant to field value');
608 varString: mType := TType.TString;
609 varDispatch: raise TDynRecException.Create('cannot convert dispatch variant to field value');
610 varBoolean: mType := TType.TBool;
611 varVariant: raise TDynRecException.Create('cannot convert variant variant to field value');
612 varUnknown: raise TDynRecException.Create('cannot convert unknown variant to field value');
613 varByte: mType := TType.TUByte;
614 varWord: mType := TType.TUShort;
615 varShortInt: mType := TType.TByte;
616 varSmallint: mType := TType.TShort;
617 varInteger: mType := TType.TInt;
618 varInt64: raise TDynRecException.Create('cannot convert int64 variant to field value');
619 varLongWord: raise TDynRecException.Create('cannot convert longword variant to field value');
620 varQWord: raise TDynRecException.Create('cannot convert uint64 variant to field value');
621 varError: raise TDynRecException.Create('cannot convert error variant to field value');
622 else raise TDynRecException.Create('cannot convert undetermined variant to field value');
623 end;
624 value := val;
625 end;
628 destructor TDynField.Destroy ();
629 begin
630 cleanup();
631 inherited;
632 end;
635 procedure TDynField.cleanup ();
636 begin
637 mName := '';
638 mTip := '';
639 mHelp := '';
640 mType := TType.TInt;
641 mIVal := 0;
642 mIVal2 := 0;
643 mIVal3 := 0;
644 mIVal4 := 0; // default alpha value
645 mSVal := '';
646 mRVal.Free();
647 mRVal := nil;
648 mRHash.Free();
649 mRHash := nil;
650 mRecRef := nil;
651 mMaxDim := -1;
652 mBinOfs := -1;
653 mSepPosSize := false;
654 mAsT := false;
655 mHasDefault := false;
656 mDefined := false;
657 mWriteDef := false;
658 mInternal := true;
659 mDefUnparsed := '';
660 mDefSVal := '';
661 mDefIVal := 0;
662 mDefIVal2 := 0;
663 mDefIVal3 := 0;
664 mDefIVal4 := 0; // default value for alpha
665 mDefRecRef := nil;
666 mEBS := TEBS.TNone;
667 mEBSTypeName := '';
668 mEBSType := nil;
669 mBitSetUnique := false;
670 mAsMonsterId := false;
671 mNegBool := false;
672 mRecRefId := '';
673 mTagInt := 0;
674 mTagPtr := nil;
675 mAlias := '';
676 end;
679 function TDynField.clone (newOwner: TDynRecord=nil; registerIn: TDynRecord=nil): TDynField;
680 var
681 rec: TDynRecord;
682 begin
683 result := TDynField.Create(mName, mType);
684 result.mOwner := mOwner;
685 if (newOwner <> nil) then result.mOwner := newOwner else result.mOwner := mOwner;
686 result.mName := mName;
687 result.mTip := mTip;
688 result.mHelp := mHelp;
689 result.mType := mType;
690 result.mIVal := mIVal;
691 result.mIVal2 := mIVal2;
692 result.mIVal3 := mIVal3;
693 result.mIVal4 := mIVal4;
694 result.mSVal := mSVal;
695 if (mRVal <> nil) then
696 begin
697 if (result.mRVal = nil) then result.mRVal := TDynRecList.Create(mRVal.count);
698 if (result.mRHash = nil) then result.mRHash := hashNewStrInt();
699 for rec in mRVal do result.addListItem(rec.clone(registerIn));
700 end;
701 result.mRecRef := mRecRef;
702 result.mMaxDim := mMaxDim;
703 result.mBinOfs := mBinOfs;
704 result.mSepPosSize := mSepPosSize;
705 result.mAsT := mAsT;
706 result.mDefined := mDefined;
707 result.mHasDefault := mHasDefault;
708 result.mWriteDef := mWriteDef;
709 result.mInternal := mInternal;
710 result.mNegBool := mNegBool;
711 result.mBitSetUnique := mBitSetUnique;
712 result.mAsMonsterId := mAsMonsterId;
713 result.mDefUnparsed := mDefUnparsed;
714 result.mDefSVal := mDefSVal;
715 result.mDefIVal := mDefIVal;
716 result.mDefIVal2 := mDefIVal2;
717 result.mDefIVal3 := mDefIVal3;
718 result.mDefIVal4 := mDefIVal4;
719 result.mDefRecRef := mDefRecRef;
720 result.mEBS := mEBS;
721 result.mEBSTypeName := mEBSTypeName;
722 result.mEBSType := mEBSType;
723 result.mRecRefId := mRecRefId;
724 result.mTagInt := mTagInt;
725 result.mTagPtr := mTagPtr;
726 result.mAlias := mAlias;
727 end;
730 function TDynField.palias (firstUp: Boolean=false): AnsiString;
731 var
732 nextUp: Boolean;
733 ch: AnsiChar;
734 begin
735 if (Length(mAlias) > 0) then
736 begin
737 if firstUp then result := UpCase1251(mAlias[1])+Copy(mAlias, 2, Length(mAlias)-1) else result := mAlias;
738 end
739 else
740 begin
741 result := '';
742 nextUp := firstUp;
743 for ch in mName do
744 begin
745 if (ch = '_') then begin nextUp := true; continue; end;
746 if nextUp then result += UpCase1251(ch) else result += ch;
747 nextUp := false;
748 end;
749 end;
750 end;
753 procedure TDynField.setRecRef (arec: TDynRecord);
754 var
755 trc: TDynRecord = nil;
756 begin
757 case mEBS of
758 TEBS.TNone: raise TDynRecException.CreateFmt('cannot set refrec for non-reference field ''%s''', [mName]);
759 TEBS.TRec:
760 begin
761 if (arec <> nil) then
762 begin
763 if (mEBSType <> nil) and (mEBSType is TDynRecord) then trc := (mEBSType as TDynRecord);
764 if (trc = nil) then raise TDynRecException.CreateFmt('cannot set refrec for field ''%s'' (type conflict: improperly initialized field)', [mName]);
765 if (trc.typeName <> arec.typeName) then raise TDynRecException.CreateFmt('cannot set refrec for field ''%s'' (type conflict: expected ''%s'' got ''%s'')', [mName, trc.typeName, arec.typeName]);
766 end;
767 mRecRef := arec;
768 mDefined := true;
769 exit;
770 end;
771 TEBS.TEnum: raise TDynRecException.CreateFmt('cannot set refrec for enum field ''%s''', [mName]);
772 TEBS.TBitSet: raise TDynRecException.CreateFmt('cannot set refrec for bitset field ''%s''', [mName]);
773 else raise TDynRecException.Create('ketmar forgot to process some reftypes');
774 end;
775 end;
778 function TDynField.getVar (): Variant;
779 begin
780 if (mEBS = TEBS.TRec) then begin result := LongInt(getRecRefIndex); exit; end;
781 case mType of
782 TType.TBool: result := (mIVal <> 0);
783 TType.TChar: result := mSVal;
784 TType.TByte: result := ShortInt(mIVal);
785 TType.TUByte: result := Byte(mIVal);
786 TType.TShort: result := SmallInt(mIVal);
787 TType.TUShort: result := Word(mIVal);
788 TType.TInt: result := LongInt(mIVal);
789 TType.TUInt: result := LongWord(mIVal);
790 TType.TString: result := mSVal;
791 TType.TPoint: raise TDynRecException.Create('cannot convert point field to variant');
792 TType.TSize: raise TDynRecException.Create('cannot convert size field to variant');
793 TType.TColor: raise TDynRecException.Create('cannot convert color field to variant');
794 TType.TList: raise TDynRecException.Create('cannot convert list field to variant');
795 TType.TTrigData: raise TDynRecException.Create('cannot convert trigdata field to variant');
796 else result := Unassigned; raise TDynRecException.Create('ketmar forgot to handle some field type');
797 end;
798 end;
801 procedure TDynField.setVar (val: Variant);
802 procedure setInt32 (v: LongInt);
803 begin
804 case mType of
805 TType.TBool:
806 if (v = 0) then mIVal := 0
807 else if (v = 1) then mIVal := 1
808 else raise TDynRecException.Create('cannot convert shortint variant to field value');
809 TType.TByte:
810 if (v >= -128) and (v <= 127) then mIVal := v
811 else raise TDynRecException.Create('cannot convert shortint variant to field value');
812 TType.TUByte:
813 if (v >= 0) and (v <= 255) then mIVal := v
814 else raise TDynRecException.Create('cannot convert shortint variant to field value');
815 TType.TShort:
816 if (v >= -32768) and (v <= 32767) then mIVal := v
817 else raise TDynRecException.Create('cannot convert shortint variant to field value');
818 TType.TUShort:
819 if (v >= 0) and (v <= 65535) then mIVal := v
820 else raise TDynRecException.Create('cannot convert shortint variant to field value');
821 TType.TInt:
822 mIVal := v;
823 TType.TUInt:
824 mIVal := v;
825 TType.TString:
826 mSVal := formatstrf('%s', [v]);
827 else
828 raise TDynRecException.Create('cannot convert integral variant to field value');
829 end;
830 end;
831 begin
832 case varType(val) of
833 varEmpty: raise TDynRecException.Create('cannot convert empty variant to field value');
834 varNull: raise TDynRecException.Create('cannot convert null variant to field value');
835 varSingle: raise TDynRecException.Create('cannot convert single variant to field value');
836 varDouble: raise TDynRecException.Create('cannot convert double variant to field value');
837 varDecimal: raise TDynRecException.Create('cannot convert decimal variant to field value');
838 varCurrency: raise TDynRecException.Create('cannot convert currency variant to field value');
839 varDate: raise TDynRecException.Create('cannot convert date variant to field value');
840 varOleStr: raise TDynRecException.Create('cannot convert olestr variant to field value');
841 varStrArg: raise TDynRecException.Create('cannot convert stdarg variant to field value');
842 varString:
843 if (mType = TType.TChar) or (mType = TType.TString) then
844 begin
845 mSVal := val;
846 end
847 else
848 begin
849 raise TDynRecException.Create('cannot convert string variant to field value');
850 end;
851 varDispatch: raise TDynRecException.Create('cannot convert dispatch variant to field value');
852 varBoolean:
853 case mType of
854 TType.TBool,
855 TType.TByte,
856 TType.TUByte,
857 TType.TShort,
858 TType.TUShort,
859 TType.TInt,
860 TType.TUInt:
861 if val then mIVal := 1 else mIVal := 0;
862 TType.TString:
863 if val then mSVal := 'true' else mSVal := 'false';
864 else
865 raise TDynRecException.Create('cannot convert boolean variant to field value');
866 end;
867 varVariant: raise TDynRecException.Create('cannot convert variant variant to field value');
868 varUnknown: raise TDynRecException.Create('cannot convert unknown variant to field value');
869 varByte,
870 varWord,
871 varShortInt,
872 varSmallint,
873 varInteger:
874 setInt32(val);
875 varInt64:
876 if (val < Int64(LongInt($80000000))) or (val > LongInt($7FFFFFFF)) then
877 raise TDynRecException.Create('cannot convert boolean variant to field value')
878 else
879 mIVal := LongInt(val);
880 varLongWord:
881 if (val > LongWord($7FFFFFFF)) then raise TDynRecException.Create('cannot convert longword variant to field value')
882 else setInt32(Integer(val));
883 varQWord: raise TDynRecException.Create('cannot convert uint64 variant to field value');
884 varError: raise TDynRecException.Create('cannot convert error variant to field value');
885 else raise TDynRecException.Create('cannot convert undetermined variant to field value');
886 end;
887 mDefined := true;
888 end;
891 // won't work for lists
892 function TDynField.isSimpleEqu (fld: TDynField): Boolean;
893 begin
894 if (fld = nil) or (mType <> fld.mType) then begin result := false; exit; end;
895 case mType of
896 TType.TBool: result := ((mIVal <> 0) = (fld.mIVal <> 0));
897 TType.TChar: result := (mSVal = fld.mSVal);
898 TType.TByte,
899 TType.TUByte,
900 TType.TShort,
901 TType.TUShort,
902 TType.TInt,
903 TType.TUInt:
904 result := (mIVal = fld.mIVal);
905 TType.TString: result := (mSVal = fld.mSVal);
906 TType.TPoint,
907 TType.TSize:
908 result := ((mIVal = fld.mIVal) and (mIVal2 = fld.mIVal2));
909 TType.TColor:
910 result := ((mIVal = fld.mIVal) and (mIVal2 = fld.mIVal2) and (mIVal3 = fld.mIVal3) and (mIVal4 = fld.mIVal4));
911 TType.TList: result := false;
912 TType.TTrigData:
913 begin
914 if (mRecRef = nil) then begin result := (fld.mRecRef = nil); exit; end;
915 result := mRecRef.isSimpleEqu(fld.mRecRef);
916 end;
917 else raise TDynRecException.Create('ketmar forgot to handle some field type');
918 end;
919 end;
922 procedure TDynField.setValue (const s: AnsiString);
923 var
924 stp: TTextParser;
925 begin
926 stp := TStrTextParser.Create(s+';');
927 try
928 parseValue(stp);
929 finally
930 stp.Free();
931 end;
932 end;
935 function TDynField.getRed (): Integer; inline; begin result := mIVal; if (result < 0) then result := 0 else if (result > 255) then result := 255; end;
936 procedure TDynField.setRed (v: Integer); inline; begin if (v < 0) then v := 0 else if (v > 255) then v := 255; mIVal := v; end;
938 function TDynField.getGreen (): Integer; inline; begin result := mIVal2; if (result < 0) then result := 0 else if (result > 255) then result := 255; end;
939 procedure TDynField.setGreen (v: Integer); inline; begin if (v < 0) then v := 0 else if (v > 255) then v := 255; mIVal2 := v; end;
941 function TDynField.getBlue (): Integer; inline; begin result := mIVal3; if (result < 0) then result := 0 else if (result > 255) then result := 255; end;
942 procedure TDynField.setBlue (v: Integer); inline; begin if (v < 0) then v := 0 else if (v > 255) then v := 255; mIVal3 := v; end;
944 function TDynField.getAlpha (): Integer; inline; begin result := mIVal4; if (result < 0) then result := 0 else if (result > 255) then result := 255; end;
945 procedure TDynField.setAlpha (v: Integer); inline; begin if (v < 0) then v := 0 else if (v > 255) then v := 255; mIVal4 := v; end;
948 procedure TDynField.parseDefaultValue ();
949 var
950 stp: TTextParser = nil;
951 oSVal: AnsiString;
952 oIVal, oIVal2, oIVal3, oIVal4: Integer;
953 oRRef: TDynRecord;
954 oDef: Boolean;
955 begin
956 if not mHasDefault then
957 begin
958 mDefSVal := '';
959 mDefIVal := 0;
960 mDefIVal2 := 0;
961 mDefIVal3 := 0;
962 mDefIVal4 := 0; // default value for alpha
963 mDefRecRef := nil;
964 end
965 else
966 begin
967 oSVal := mSVal;
968 oIVal := mIVal;
969 oIVal2 := mIVal2;
970 oIVal3 := mIVal3;
971 oIVal4 := mIVal4;
972 oRRef := mRecRef;
973 oDef := mDefined;
974 try
975 stp := TStrTextParser.Create(mDefUnparsed+';');
976 parseValue(stp);
977 mDefSVal := mSVal;
978 mDefIVal := mIVal;
979 mDefIVal2 := mIVal2;
980 mDefIVal3 := mIVal3;
981 mDefIVal4 := mIVal4;
982 mDefRecRef := mRecRef;
983 finally
984 mSVal := oSVal;
985 mIVal := oIVal;
986 mIVal2 := oIVal2;
987 mIVal3 := oIVal3;
988 mIVal4 := oIVal4;
989 mRecRef := oRRef;
990 mDefined := oDef;
991 stp.Free();
992 end;
993 end;
994 end;
997 // default value should be parsed
998 procedure TDynField.fixDefaultValue ();
999 begin
1000 if mDefined then exit;
1001 if not mHasDefault then
1002 begin
1003 if mInternal then exit;
1004 raise TDynRecException.CreateFmt('field ''%s'' in record ''%s'' of record type ''%s'' is not set', [mName, mOwner.mId, mOwner.mTypeName]);
1005 end;
1006 if (mEBS = TEBS.TRec) then mRecRef := mDefRecRef;
1007 mSVal := mDefSVal;
1008 mIVal := mDefIVal;
1009 mIVal2 := mDefIVal2;
1010 mIVal3 := mDefIVal3;
1011 mIVal4 := mDefIVal4;
1012 mDefined := true;
1013 end;
1016 // default value should be parsed
1017 function TDynField.isDefaultValue (): Boolean;
1018 begin
1019 if not mHasDefault then begin result := false; exit; end;
1020 if (mEBS = TEBS.TRec) then begin result := (mRecRef = mDefRecRef); exit; end;
1021 case mType of
1022 TType.TChar, TType.TString: result := (mSVal = mDefSVal);
1023 TType.TPoint, TType.TSize: result := (mIVal = mDefIVal2) and (mIVal2 = mDefIVal2);
1024 TType.TColor: result := (mIVal = mDefIVal2) and (mIVal2 = mDefIVal2) and (mIVal3 = mDefIVal3) and (mIVal4 = mDefIVal4);
1025 TType.TList, TType.TTrigData: result := false; // no default values for those types
1026 else result := (mIVal = mDefIVal);
1027 end;
1028 end;
1031 function TDynField.getListCount (): Integer; inline;
1032 begin
1033 if (mRVal <> nil) then result := mRVal.count else result := 0;
1034 end;
1037 function TDynField.getListItem (idx: Integer): TDynRecord; inline; overload;
1038 begin
1039 if (mRVal <> nil) and (idx >= 0) and (idx < mRVal.count) then result := mRVal[idx] else result := nil;
1040 end;
1043 function TDynField.getListItem (const aname: AnsiString): TDynRecord; inline; overload;
1044 var
1045 idx: Integer;
1046 begin
1047 if (mRVal <> nil) and mRHash.get(aname, idx) then result := mRVal[idx] else result := nil;
1048 end;
1051 function TDynField.addListItem (rec: TDynRecord): Boolean; inline;
1052 begin
1053 result := false;
1054 if (mRVal <> nil) then
1055 begin
1056 mRVal.append(rec);
1057 if (Length(rec.mId) > 0) then result := mRHash.put(rec.mId, mRVal.count-1);
1058 end;
1059 end;
1062 function TDynField.removeListItem (const aid: AnsiString): TDynRecord;
1063 var
1064 f, idx: Integer;
1065 begin
1066 result := nil;
1067 if mRHash.get(aid, idx) then
1068 begin
1069 assert((idx >= 0) and (idx < mRVal.count));
1070 result := mRVal[idx];
1071 // fix hash and list
1072 for f := idx+1 to mRVal.count-1 do
1073 begin
1074 if (Length(mRVal[f].mId) > 0) then mRHash.put(mRVal[f].mId, f-1);
1075 end;
1076 mRHash.del(aid);
1077 mRVal.delete(idx);
1078 end;
1079 end;
1082 class function TDynField.getTypeName (t: TType): AnsiString;
1083 begin
1084 case t of
1085 TType.TBool: result := 'bool';
1086 TType.TChar: result := 'char';
1087 TType.TByte: result := 'byte';
1088 TType.TUByte: result := 'ubyte';
1089 TType.TShort: result := 'short';
1090 TType.TUShort: result := 'ushort';
1091 TType.TInt: result := 'int';
1092 TType.TUInt: result := 'uint';
1093 TType.TString: result := 'string';
1094 TType.TPoint: result := 'point';
1095 TType.TSize: result := 'size';
1096 TType.TColor: result := 'color';
1097 TType.TList: result := 'array';
1098 TType.TTrigData: result := 'trigdata';
1099 else raise TDynRecException.Create('ketmar forgot to handle some field type');
1100 end;
1101 end;
1104 function TDynField.definition (): AnsiString;
1105 begin
1106 result := quoteStr(mName)+' type ';
1107 result += getTypeName(mType);
1108 if (Length(mAlias) > 0) then result += ' alias '+mAlias;
1109 if (mMaxDim >= 0) then result += Format('[%d]', [mMaxDim]);
1110 if (mBinOfs >= 0) then result += Format(' offset %d', [mBinOfs]);
1111 case mEBS of
1112 TEBS.TNone: begin end;
1113 TEBS.TRec: result += ' '+mEBSTypeName;
1114 TEBS.TEnum: result += ' enum '+mEBSTypeName;
1115 TEBS.TBitSet: begin result += ' bitset '; if mBitSetUnique then result += 'unique '; result += mEBSTypeName; end;
1116 end;
1117 if mAsMonsterId then result += ' as monsterid';
1118 if mHasDefault and (Length(mDefUnparsed) > 0) then result += ' default '+mDefUnparsed;
1119 if mSepPosSize then
1120 begin
1121 if (mType = TType.TPoint) then begin if (mAsT) then result += ' as txy' else result += ' as xy'; end
1122 else if (mType = TType.TSize) then begin if (mAsT) then result += ' as twh' else result += ' as wh'; end;
1123 end;
1124 if mWriteDef then result += ' writedefault';
1125 if mInternal then result += ' internal';
1126 end;
1129 procedure TDynField.parseDef (pr: TTextParser);
1130 var
1131 fldname: AnsiString;
1132 fldtype: AnsiString;
1133 fldofs: Integer;
1134 fldrecname: AnsiString;
1135 asxy, aswh, ast: Boolean;
1136 ainternal: Boolean;
1137 writedef: Boolean;
1138 defstr: AnsiString;
1139 defint, defint2, defint3, defint4: Integer;
1140 hasdefStr: Boolean;
1141 hasdefInt: Boolean;
1142 hasdefId: Boolean;
1143 lmaxdim: Integer;
1144 lebs: TDynField.TEBS;
1145 unique: Boolean;
1146 asmonid: Boolean;
1147 defech: AnsiChar;
1148 xalias: AnsiString;
1149 atip, ahelp: AnsiString;
1150 begin
1151 fldname := '';
1152 fldtype := '';
1153 fldofs := -1;
1154 fldrecname := '';
1155 asxy := false;
1156 aswh := false;
1157 ast := false;
1158 ainternal := false;
1159 writedef := false;
1160 defstr := '';
1161 defint := 0;
1162 defint2 := 0;
1163 defint3 := 0;
1164 defint4 := 0;
1165 hasdefStr := false;
1166 hasdefInt := false;
1167 hasdefId := false;
1168 unique := false;
1169 asmonid := false;
1170 lmaxdim := -1;
1171 lebs := TDynField.TEBS.TNone;
1172 xalias := '';
1173 atip := '';
1174 ahelp := '';
1176 // field name
1177 fldname := pr.expectStrOrId();
1179 while (pr.tokType <> pr.TTSemi) do
1180 begin
1181 if pr.eatId('type') then
1182 begin
1183 if (Length(fldtype) > 0) then raise TDynParseException.CreateFmt(pr, 'duplicate type definition for field ''%s''', [fldname]);
1184 // field type
1185 fldtype := pr.expectId();
1186 // fixed-size array?
1187 if pr.eatDelim('[') then
1188 begin
1189 lmaxdim := pr.expectInt();
1190 // arbitrary limits
1191 if (lmaxdim < 1) or (lmaxdim > 32768) then raise TDynParseException.CreateFmt(pr, 'invalid field ''%s'' array size', [fldname]);
1192 pr.expectDelim(']');
1193 end;
1194 continue;
1195 end;
1197 if pr.eatId('alias') then
1198 begin
1199 if (Length(xalias) > 0) then raise TDynParseException.CreateFmt(pr, 'duplicate alias definition for field ''%s''', [fldname]);
1200 xalias := pr.expectId();
1201 continue;
1202 end;
1204 if pr.eatId('tip') then
1205 begin
1206 if (Length(atip) > 0) then raise TDynParseException.CreateFmt(pr, 'duplicate tip definition for field ''%s''', [fldname]);
1207 atip := pr.expectStr(false);
1208 continue;
1209 end;
1211 if pr.eatId('help') then
1212 begin
1213 if (Length(ahelp) > 0) then raise TDynParseException.CreateFmt(pr, 'duplicate tip definition for field ''%s''', [fldname]);
1214 ahelp := pr.expectStr(false);
1215 continue;
1216 end;
1218 if pr.eatId('offset') then
1219 begin
1220 if (fldofs >= 0) then raise TDynParseException.CreateFmt(pr, 'duplicate field ''%s'' offset', [fldname]);
1221 fldofs := pr.expectInt();
1222 if (fldofs < 0) then raise TDynParseException.CreateFmt(pr, 'invalid field ''%s'' offset', [fldname]);
1223 continue;
1224 end;
1226 if pr.eatId('as') then
1227 begin
1228 if pr.eatId('xy') then asxy := true
1229 else if pr.eatId('wh') then aswh := true
1230 else if pr.eatId('txy') then begin asxy := true; ast := true; end
1231 else if pr.eatId('twh') then begin aswh := true; ast := true; end
1232 else if pr.eatId('monsterid') then begin asmonid := true; end
1233 else raise TDynParseException.CreateFmt(pr, 'invalid field ''%s'' as what?', [fldname]);
1234 continue;
1235 end;
1237 if pr.eatId('enum') then
1238 begin
1239 lebs := TDynField.TEBS.TEnum;
1240 if (Length(fldrecname) <> 0) then raise TDynParseException.CreateFmt(pr, 'field ''%s'' already typed as ''%s''', [fldname, fldrecname]);
1241 fldrecname := pr.expectId();
1242 continue;
1243 end;
1245 if pr.eatId('bitset') then
1246 begin
1247 lebs := TDynField.TEBS.TBitSet;
1248 if (Length(fldrecname) <> 0) then raise TDynParseException.CreateFmt(pr, 'field ''%s'' already typed as ''%s''', [fldname, fldrecname]);
1249 unique := pr.eatId('unique');
1250 fldrecname := pr.expectId();
1251 continue;
1252 end;
1254 if pr.eatId('default') then
1255 begin
1256 if hasdefStr or hasdefInt or hasdefId then raise TDynParseException.CreateFmt(pr, 'field ''%s'' has duplicate default', [fldname]);
1257 case pr.tokType of
1258 pr.TTStr:
1259 begin
1260 hasdefStr := true;
1261 defstr := pr.expectStr(true); // allow empty strings
1262 end;
1263 pr.TTId:
1264 begin
1265 hasdefId := true;
1266 defstr := pr.expectId();
1267 end;
1268 pr.TTInt:
1269 begin
1270 hasdefInt := true;
1271 defint := pr.expectInt();
1272 end;
1273 pr.TTDelim:
1274 begin
1275 hasdefInt := true;
1276 if pr.eatDelim('[') then defech := ']' else begin pr.expectDelim('('); defech := ')'; end;
1277 defint := pr.expectInt();
1278 defint2 := pr.expectInt();
1279 if (pr.tokType = pr.TTInt) then
1280 begin
1281 defint3 := pr.expectInt();
1282 if (pr.tokType = pr.TTInt) then defint4 := pr.expectInt();
1283 end;
1284 pr.expectDelim(defech);
1285 end;
1286 else
1287 raise TDynParseException.CreateFmt(pr, 'field ''%s'' has invalid default', [fldname]);
1288 end;
1289 continue;
1290 end;
1292 if pr.eatId('writedefault') then
1293 begin
1294 writedef := true;
1295 continue;
1296 end;
1298 if pr.eatId('internal') then
1299 begin
1300 ainternal := true;
1301 continue;
1302 end;
1304 // record type, no special modifiers
1305 if (pr.tokType <> pr.TTId) then raise TDynParseException.CreateFmt(pr, 'field ''%s'' has something unexpected in definition', [fldname]);
1307 if (Length(fldrecname) <> 0) then raise TDynParseException.CreateFmt(pr, 'field ''%s'' already typed as ''%s''', [fldname, fldrecname]);
1308 fldrecname := pr.expectId();
1309 lebs := TDynField.TEBS.TRec;
1310 end;
1312 pr.expectTT(pr.TTSemi);
1314 // create field
1315 mName := fldname;
1316 if (fldtype = 'bool') then mType := TType.TBool
1317 else if (fldtype = 'negbool') then begin mType := TType.TBool; mNegBool := true; end
1318 else if (fldtype = 'char') then mType := TType.TChar
1319 else if (fldtype = 'byte') then mType := TType.TByte
1320 else if (fldtype = 'ubyte') then mType := TType.TUByte
1321 else if (fldtype = 'short') then mType := TType.TShort
1322 else if (fldtype = 'ushort') then mType := TType.TUShort
1323 else if (fldtype = 'int') then mType := TType.TInt
1324 else if (fldtype = 'uint') then mType := TType.TUInt
1325 else if (fldtype = 'string') then mType := TType.TString
1326 else if (fldtype = 'point') then mType := TType.TPoint
1327 else if (fldtype = 'size') then mType := TType.TSize
1328 else if (fldtype = 'color') then mType := TType.TColor
1329 else if (fldtype = 'trigdata') then mType := TType.TTrigData
1330 else
1331 begin
1332 // record types defaults to int
1333 if (Length(fldrecname) > 0) then
1334 begin
1335 mType := TType.TInt;
1336 end
1337 else
1338 begin
1339 if (Length(fldtype) = 0) then raise TDynParseException.CreateFmt(pr, 'field ''%s'' has no type', [fldname])
1340 else raise TDynParseException.CreateFmt(pr, 'field ''%s'' has invalid type ''%s''', [fldname, fldtype]);
1341 end;
1342 end;
1344 // check for valid arrays
1345 if (lmaxdim > 0) and (mType <> TType.TChar) and (mType <> TType.TTrigData) then raise TDynParseException.CreateFmt(pr, 'field ''%s'' of type ''%s'' cannot be array', [fldname, fldtype]);
1347 // check for valid trigdata or record type
1348 if (mType = TType.TTrigData) then
1349 begin
1350 // trigdata
1351 if (lmaxdim < 1) then raise TDynParseException.CreateFmt(pr, 'field ''%s'' of type ''%s'' cannot be non-array', [fldname, 'trigdata']);
1352 if (Length(fldrecname) > 0) then raise TDynParseException.CreateFmt(pr, 'field ''%s'' of type ''%s'' cannot have another type', [fldname, 'trigdata']);
1353 lebs := TDynField.TEBS.TRec;
1354 end
1355 else if (Length(fldrecname) > 0) then
1356 begin
1357 // record
1358 if not (mType in [TType.TByte, TType.TUByte, TType.TShort, TType.TUShort, TType.TInt, TType.TUInt]) then
1359 begin
1360 raise TDynParseException.CreateFmt(pr, 'field ''%s'' of record type ''%s'' cannot have type ''%s''', [fldname, fldrecname, fldtype]);
1361 end;
1362 end;
1364 // setup default value
1365 if hasdefStr then self.mDefUnparsed := quoteStr(defstr)
1366 else if hasdefId then self.mDefUnparsed := defstr
1367 else if hasdefInt then
1368 begin
1369 if (mType = TType.TPoint) then self.mDefUnparsed := Format('(%d %d)', [defint, defint2])
1370 else if (mType = TType.TSize) then self.mDefUnparsed := Format('[%d %d]', [defint, defint2])
1371 else if (mType = TType.TColor) then self.mDefUnparsed := Format('(%d %d %d %d)', [defint, defint2, defint3, defint4])
1372 else self.mDefUnparsed := Format('%d', [defint]);
1373 end;
1375 self.mHasDefault := (hasdefStr or hasdefId or hasdefInt);
1376 self.mEBS := lebs;
1377 self.mEBSTypeName := fldrecname;
1378 self.mBitSetUnique := unique;
1379 self.mAsMonsterId := asmonid;
1380 self.mMaxDim := lmaxdim;
1381 self.mBinOfs := fldofs;
1382 self.mSepPosSize := (asxy or aswh);
1383 self.mAsT := ast;
1384 self.mWriteDef := writedef;
1385 self.mInternal := ainternal;
1386 self.mAlias := xalias;
1387 self.mTip := atip;
1388 self.mHelp := ahelp;
1389 end;
1392 function TDynField.getRecRefIndex (): Integer;
1393 begin
1394 if (mRecRef = nil) then begin result := -1; exit; end;
1395 result := mOwner.findRecordNumByType(mEBSTypeName, mRecRef);
1396 end;
1399 procedure TDynField.writeBinTo (var hasLostData: Boolean; st: TStream);
1400 var
1401 s: AnsiString;
1402 f: Integer;
1403 maxv: Integer;
1404 buf: PByte;
1405 ws: TStream = nil;
1406 begin
1407 case mEBS of
1408 TEBS.TNone: begin end;
1409 TEBS.TRec:
1410 begin
1411 if (mMaxDim >= 0) then
1412 begin
1413 // this must be triggerdata
1414 if (mType <> TType.TTrigData) then
1415 begin
1416 raise TDynRecException.CreateFmt('record reference type ''%s'' in field ''%s'' cannot be written', [mEBSTypeName, mName]);
1417 end;
1418 // write triggerdata
1419 GetMem(buf, mMaxDim);
1420 if (buf = nil) then raise TDynRecException.CreateFmt('record reference type ''%s'' in field ''%s'' cannot be written', [mEBSTypeName, mName]);
1421 try
1422 FillChar(buf^, mMaxDim, 0);
1423 if (mRecRef <> nil) then
1424 begin
1425 ws := TSFSMemoryChunkStream.Create(buf, mMaxDim);
1426 mRecRef.writeBinTo(hasLostData, ws, mMaxDim); // as trigdata
1427 end;
1428 st.WriteBuffer(buf^, mMaxDim);
1429 finally
1430 ws.Free();
1431 if (buf <> nil) then FreeMem(buf);
1432 end;
1433 exit;
1434 end;
1435 // record reference
1436 case mType of
1437 TType.TByte: maxv := 127;
1438 TType.TUByte: maxv := 254;
1439 TType.TShort: maxv := 32767;
1440 TType.TUShort: maxv := 65534;
1441 TType.TInt: maxv := $7fffffff;
1442 TType.TUInt: maxv := $7fffffff;
1443 else raise TDynRecException.CreateFmt('record reference type ''%s'' in field ''%s'' cannot be written', [mEBSTypeName, mName]);
1444 end;
1445 // find record number
1446 if (mRecRef <> nil) then
1447 begin
1448 f := mOwner.findRecordNumByType(mEBSTypeName, mRecRef);
1449 if (f < 0) then raise TDynRecException.CreateFmt('record reference type ''%s'' in field ''%s'' not found in record list', [mEBSTypeName, mName]);
1450 if mAsMonsterId then Inc(f);
1451 if (f > maxv) then raise TDynRecException.CreateFmt('record reference type ''%s'' in field ''%s'' has too big index', [mEBSTypeName, mName]);
1452 end
1453 else
1454 begin
1455 if mAsMonsterId then f := 0 else f := -1;
1456 end;
1457 case mType of
1458 TType.TByte, TType.TUByte: writeInt(st, Byte(f));
1459 TType.TShort, TType.TUShort: writeInt(st, SmallInt(f));
1460 TType.TInt, TType.TUInt: writeInt(st, LongWord(f));
1461 else raise TDynRecException.CreateFmt('record reference type ''%s'' in field ''%s'' cannot be written', [mEBSTypeName, mName]);
1462 end;
1463 exit;
1464 end;
1465 TEBS.TEnum: begin end;
1466 TEBS.TBitSet: begin end;
1467 else raise TDynRecException.Create('ketmar forgot to handle some EBS type');
1468 end;
1470 case mType of
1471 TType.TBool:
1472 begin
1473 if not mNegBool then
1474 begin
1475 if (mIVal <> 0) then writeInt(st, Byte(1)) else writeInt(st, Byte(0));
1476 end
1477 else
1478 begin
1479 if (mIVal = 0) then writeInt(st, Byte(1)) else writeInt(st, Byte(0));
1480 end;
1481 exit;
1482 end;
1483 TType.TChar:
1484 begin
1485 if (mMaxDim = 0) then raise TDynRecException.CreateFmt('invalid string size definition for field ''%s''', [mName]);
1486 if (mMaxDim < 0) then
1487 begin
1488 if (Length(mSVal) <> 1) then raise TDynRecException.CreateFmt('invalid string size definition for field ''%s''', [mName]);
1489 writeInt(st, Byte(mSVal[1]));
1490 end
1491 else
1492 begin
1493 if (Length(mSVal) > mMaxDim) then raise TDynRecException.CreateFmt('invalid string size definition for field ''%s''', [mName]);
1494 s := utf2win(mSVal);
1495 if (Length(s) > 0) then st.WriteBuffer(PChar(s)^, Length(s));
1496 for f := Length(s) to mMaxDim do writeInt(st, Byte(0));
1497 end;
1498 exit;
1499 end;
1500 TType.TByte,
1501 TType.TUByte:
1502 begin
1503 // triggerdata array was processed earlier
1504 if (mMaxDim >= 0) then TDynRecException.CreateFmt('byte array in field ''%s'' cannot be written', [mName]);
1505 writeInt(st, Byte(mIVal));
1506 exit;
1507 end;
1508 TType.TShort,
1509 TType.TUShort:
1510 begin
1511 if (mMaxDim >= 0) then raise TDynRecException.CreateFmt('short array in field ''%s'' cannot be written', [mName]);
1512 writeInt(st, Word(mIVal));
1513 exit;
1514 end;
1515 TType.TInt,
1516 TType.TUInt:
1517 begin
1518 if (mMaxDim >= 0) then raise TDynRecException.CreateFmt('int array in field ''%s'' cannot be written', [mName]);
1519 writeInt(st, LongWord(mIVal));
1520 exit;
1521 end;
1522 TType.TString:
1523 begin
1524 raise TDynRecException.CreateFmt('cannot write string field ''%s''', [mName]);
1525 end;
1526 TType.TPoint:
1527 begin
1528 if (mMaxDim >= 0) then raise TDynRecException.CreateFmt('pos/size array in field ''%s'' cannot be written', [mName]);
1529 writeInt(st, LongInt(mIVal));
1530 writeInt(st, LongInt(mIVal2));
1531 exit;
1532 end;
1533 TType.TSize:
1534 begin
1535 if (mMaxDim >= 0) then raise TDynRecException.CreateFmt('pos/size array in field ''%s'' cannot be written', [mName]);
1536 writeInt(st, Word(mIVal));
1537 writeInt(st, Word(mIVal2));
1538 exit;
1539 end;
1540 TType.TColor:
1541 begin
1542 if (mMaxDim >= 0) then raise TDynRecException.CreateFmt('color array in field ''%s'' cannot be written', [mName]);
1543 writeInt(st, Byte(mIVal));
1544 writeInt(st, Byte(mIVal2));
1545 writeInt(st, Byte(mIVal3));
1546 //writeInt(st, Byte(mIVal4)); // the only place we have RGB in binary map is effect trigger, and it has no alpha
1547 if (mIVal4 <> 255) then hasLostData := true;
1548 exit;
1549 end;
1550 TType.TList:
1551 raise TDynRecException.Create('cannot write lists to binary format');
1552 TType.TTrigData:
1553 raise TDynRecException.Create('cannot write triggers to binary format (internal error)');
1554 else raise TDynRecException.Create('ketmar forgot to handle some field type');
1555 end;
1556 end;
1559 procedure TDynField.writeTo (wr: TTextWriter);
1560 var
1561 es: TDynEBS = nil;
1562 f, mask: Integer;
1563 first, found: Boolean;
1564 begin
1565 wr.put(mName);
1566 wr.put(' ');
1567 case mEBS of
1568 TEBS.TNone: begin end;
1569 TEBS.TRec:
1570 begin
1571 if (mRecRef = nil) then
1572 begin
1573 if (mType = TType.TTrigData) then wr.put('{}'#10) else wr.put('null;'#10);
1574 end
1575 else if (Length(mRecRef.mId) = 0) then
1576 begin
1577 mRecRef.writeTo(wr, false); // only data, no header
1578 end
1579 else
1580 begin
1581 wr.put(mRecRef.mId);
1582 wr.put(';'#10);
1583 end;
1584 exit;
1585 end;
1586 TEBS.TEnum:
1587 begin
1588 //def := mOwner.mOwner;
1589 //es := def.ebsType[mEBSTypeName];
1590 es := nil;
1591 if (mEBSType <> nil) and (mEBSType is TDynEBS) then es := (mEBSType as TDynEBS);
1592 if (es = nil) or (not es.mIsEnum) then raise TDynRecException.CreateFmt('record enum type ''%s'' for field ''%s'' not found', [mEBSTypeName, mName]);
1593 for f := 0 to High(es.mVals) do
1594 begin
1595 if (es.mVals[f] = mIVal) then
1596 begin
1597 wr.put(es.mIds[f]);
1598 wr.put(';'#10);
1599 exit;
1600 end;
1601 end;
1602 raise TDynRecException.CreateFmt('value %d in record enum type ''%s'' for field ''%s'' not found', [mIVal, mEBSTypeName, mName]);
1603 end;
1604 TEBS.TBitSet:
1605 begin
1606 //def := mOwner.mOwner;
1607 //es := def.ebsType[mEBSTypeName];
1608 es := nil;
1609 if (mEBSType <> nil) and (mEBSType is TDynEBS) then es := (mEBSType as TDynEBS);
1610 if (es = nil) or es.mIsEnum then raise TDynRecException.CreateFmt('record bitset type ''%s'' for field ''%s'' not found', [mEBSTypeName, mName]);
1611 // none?
1612 if (mIVal = 0) then
1613 begin
1614 for f := 0 to High(es.mVals) do
1615 begin
1616 if (es.mVals[f] = 0) then
1617 begin
1618 wr.put(es.mIds[f]);
1619 wr.put(';'#10);
1620 exit;
1621 end;
1622 end;
1623 raise TDynRecException.CreateFmt('value %d in record bitset type ''%s'' for field ''%s'' not found', [0, mEBSTypeName, mName]);
1624 end;
1625 // not none
1626 mask := 1;
1627 first := true;
1628 while (mask <> 0) do
1629 begin
1630 if ((mIVal and mask) <> 0) then
1631 begin
1632 found := false;
1633 for f := 0 to High(es.mVals) do
1634 begin
1635 if (es.mVals[f] = mask) then
1636 begin
1637 if not first then wr.put(' | ') else first := false;
1638 wr.put(es.mIds[f]);
1639 found := true;
1640 break;
1641 end;
1642 end;
1643 if not found then raise TDynRecException.CreateFmt('value %d in record bitset type ''%s'' for field ''%s'' not found', [mask, mEBSTypeName, mName]);
1644 end;
1645 mask := mask shl 1;
1646 end;
1647 wr.put(';'#10);
1648 exit;
1649 end;
1650 else raise TDynRecException.Create('ketmar forgot to handle some EBS type');
1651 end;
1653 case mType of
1654 TType.TBool:
1655 begin
1656 if (mIVal = 0) then wr.put('false;'#10) else wr.put('true;'#10);
1657 exit;
1658 end;
1659 TType.TChar:
1660 begin
1661 if (mMaxDim = 0) then raise TDynRecException.CreateFmt('invalid string size definition for field ''%s''', [mName]);
1662 wr.put(quoteStr(mSVal));
1663 wr.put(';'#10);
1664 exit;
1665 end;
1666 TType.TByte,
1667 TType.TUByte,
1668 TType.TShort,
1669 TType.TUShort,
1670 TType.TInt,
1671 TType.TUInt:
1672 begin
1673 wr.put('%d;'#10, [mIVal]);
1674 exit;
1675 end;
1676 TType.TString:
1677 begin
1678 wr.put(quoteStr(mSVal));
1679 wr.put(';'#10);
1680 exit;
1681 end;
1682 TType.TPoint,
1683 TType.TSize:
1684 begin
1685 wr.put('(%d %d);'#10, [mIVal, mIVal2]);
1686 exit;
1687 end;
1688 TType.TColor:
1689 begin
1690 if (mIVal3 = 255) then wr.put('(%d %d %d);'#10, [mIVal, mIVal2, mIVal3])
1691 else wr.put('(%d %d %d %d);'#10, [mIVal, mIVal2, mIVal3, mIVal4]);
1692 exit;
1693 end;
1694 TType.TList:
1695 begin
1696 assert(false);
1697 exit;
1698 end;
1699 TType.TTrigData:
1700 begin
1701 assert(false);
1702 exit;
1703 end;
1704 else raise TDynRecException.Create('ketmar forgot to handle some field type');
1705 end;
1706 raise TDynRecException.CreateFmt('cannot parse field ''%s'' yet', [mName]);
1707 end;
1710 procedure TDynField.parseBinValue (st: TStream);
1711 var
1712 rec, rc: TDynRecord;
1713 tfld: TDynField;
1714 es: TDynEBS = nil;
1715 tdata: PByte = nil;
1716 f, mask: Integer;
1717 s: AnsiString;
1718 begin
1719 case mEBS of
1720 TEBS.TNone: begin end;
1721 TEBS.TRec:
1722 begin
1723 // this must be triggerdata
1724 if (mType = TType.TTrigData) then
1725 begin
1726 assert(mMaxDim > 0);
1727 rec := mOwner;
1728 // find trigger definition
1729 tfld := rec.trigTypeField();
1730 if (tfld = nil) then raise TDynRecException.CreateFmt('triggerdata value for field ''%s'' in record ''%s'' without TriggerType field', [mName, rec.mTypeName]);
1731 rc := mOwner.mOwner.trigTypeFor[tfld.mSVal]; // find in mapdef
1732 if (rc = nil) then raise TDynRecException.CreateFmt('triggerdata definition for field ''%s'' in record ''%s'' with type ''%s'' not found', [mName, rec.mTypeName, tfld.mSVal]);
1733 rc := rc.clone(mOwner.mHeaderRec);
1734 rc.mHeaderRec := mOwner.mHeaderRec;
1735 // on error, it will be freed by memowner
1736 rc.parseBinValue(st, true);
1737 mRecRef := rc;
1738 mDefined := true;
1739 exit;
1740 end
1741 else
1742 begin
1743 // not a trigger data
1744 case mType of
1745 TType.TByte: f := readShortInt(st);
1746 TType.TUByte: f := readByte(st);
1747 TType.TShort: f := readSmallInt(st);
1748 TType.TUShort: f := readWord(st);
1749 TType.TInt: f := readLongInt(st);
1750 TType.TUInt: f := readLongWord(st);
1751 else raise TDynRecException.CreateFmt('invalid non-numeric type ''%s'' for field ''%s'' of record ''%s''', [getTypeName(mType), mName, mEBSTypeName]);
1752 end;
1753 if mAsMonsterId then Dec(f);
1754 if (f < 0) then mRecRefId := '' else mRecRefId := Format('%s%d', [mEBSTypeName, f]);
1755 end;
1756 mDefined := true;
1757 exit;
1758 end;
1759 TEBS.TEnum,
1760 TEBS.TBitSet:
1761 begin
1762 assert(mMaxDim < 0);
1763 case mType of
1764 TType.TByte: f := readShortInt(st);
1765 TType.TUByte: f := readByte(st);
1766 TType.TShort: f := readSmallInt(st);
1767 TType.TUShort: f := readWord(st);
1768 TType.TInt: f := readLongInt(st);
1769 TType.TUInt: f := readLongWord(st);
1770 else raise TDynRecException.CreateFmt('invalid non-numeric type ''%s'' for field ''%s'' of record ''%s''', [getTypeName(mType), mName, mEBSTypeName]);
1771 end;
1772 es := nil;
1773 if (mEBSType <> nil) and (mEBSType is TDynEBS) then es := (mEBSType as TDynEBS);
1774 if (es = nil) or (es.mIsEnum <> (mEBS = TEBS.TEnum)) then raise TDynRecException.CreateFmt('record enum type ''%s'' for field ''%s'' not found', [mEBSTypeName, mName]);
1775 mIVal := f;
1776 // build enum/bitfield values
1777 if (mEBS = TEBS.TEnum) then
1778 begin
1779 mSVal := es.nameByValue(mIVal);
1780 if (Length(mSVal) = 0) then raise TDynRecException.CreateFmt('record enum type ''%s'' for field ''%s'' has invalid value %d', [mEBSTypeName, mName, mIVal]);
1781 end
1782 else
1783 begin
1784 // special for 'none'
1785 if (mIVal = 0) then
1786 begin
1787 mSVal := es.nameByValue(mIVal);
1788 if (Length(mSVal) = 0) then raise TDynRecException.CreateFmt('record bitset type ''%s'' for field ''%s'' has invalid value %d', [mEBSTypeName, mName, mIVal]);
1789 end
1790 else
1791 begin
1792 mSVal := '';
1793 mask := 1;
1794 while (mask <> 0) do
1795 begin
1796 if ((mIVal and mask) <> 0) then
1797 begin
1798 s := es.nameByValue(mask);
1799 if (Length(s) = 0) then raise TDynRecException.CreateFmt('record bitset type ''%s'' for field ''%s'' has invalid value %d', [mEBSTypeName, mName, mask]);
1800 if (Length(mSVal) <> 0) then mSVal += '+';
1801 mSVal += s;
1802 end;
1803 mask := mask shl 1;
1804 end;
1805 end;
1806 end;
1807 //writeln('ebs <', es.mName, '>: ', mSVal);
1808 mDefined := true;
1809 exit;
1810 end;
1811 else raise TDynRecException.Create('ketmar forgot to handle some EBS type');
1812 end;
1814 case mType of
1815 TType.TBool:
1816 begin
1817 f := readByte(st);
1818 if (f <> 0) then f := 1;
1819 if mNegBool then f := 1-f;
1820 mIVal := f;
1821 mDefined := true;
1822 exit;
1823 end;
1824 TType.TChar:
1825 begin
1826 if (mMaxDim < 0) then
1827 begin
1828 mIVal := readByte(st);
1829 end
1830 else
1831 begin
1832 mSVal := '';
1833 GetMem(tdata, mMaxDim);
1834 try
1835 st.ReadBuffer(tdata^, mMaxDim);
1836 f := 0;
1837 while (f < mMaxDim) and (tdata[f] <> 0) do Inc(f);
1838 if (f > 0) then
1839 begin
1840 SetLength(mSVal, f);
1841 Move(tdata^, PChar(mSVal)^, f);
1842 mSVal := win2utf(mSVal);
1843 end;
1844 finally
1845 FreeMem(tdata);
1846 end;
1847 end;
1848 mDefined := true;
1849 exit;
1850 end;
1851 TType.TByte: begin mIVal := readShortInt(st); mDefined := true; exit; end;
1852 TType.TUByte: begin mIVal := readByte(st); mDefined := true; exit; end;
1853 TType.TShort: begin mIVal := readSmallInt(st); mDefined := true; exit; end;
1854 TType.TUShort: begin mIVal := readWord(st); mDefined := true; exit; end;
1855 TType.TInt: begin mIVal := readLongInt(st); mDefined := true; exit; end;
1856 TType.TUInt: begin mIVal := readLongWord(st); mDefined := true; exit; end;
1857 TType.TString:
1858 begin
1859 raise TDynRecException.Create('cannot read strings from binaries yet');
1860 exit;
1861 end;
1862 TType.TPoint:
1863 begin
1864 mIVal := readLongInt(st);
1865 mIVal2 := readLongInt(st);
1866 mDefined := true;
1867 exit;
1868 end;
1869 TType.TSize:
1870 begin
1871 mIVal := readWord(st);
1872 mIVal2 := readWord(st);
1873 mDefined := true;
1874 exit;
1875 end;
1876 TType.TColor:
1877 begin
1878 mIVal := readByte(st);
1879 mIVal2 := readByte(st);
1880 mIVal3 := readByte(st);
1881 //mIVal4 := readByte(st); // the only place we have RGB in binary map is effect trigger, and it has no alpha
1882 mIVal4 := 255;
1883 mDefined := true;
1884 exit;
1885 end;
1886 TType.TList:
1887 begin
1888 assert(false);
1889 exit;
1890 end;
1891 TType.TTrigData:
1892 begin
1893 assert(false);
1894 exit;
1895 end;
1896 else raise TDynRecException.Create('ketmar forgot to handle some field type');
1897 end;
1898 raise TDynRecException.CreateFmt('cannot parse field ''%s'' yet', [mName]);
1899 end;
1902 procedure TDynField.parseValue (pr: TTextParser);
1904 procedure parseInt (min, max: Integer);
1905 begin
1906 mIVal := pr.expectInt();
1907 if (mIVal < min) or (mIVal > max) then raise TDynParseException.CreateFmt(pr, 'invalid %s value for field ''%s''', [getTypeName(mType), mName]);
1908 mDefined := true;
1909 end;
1911 var
1912 rec, rc: TDynRecord;
1913 es: TDynEBS = nil;
1914 tfld: TDynField;
1915 tk: AnsiString;
1916 edim: AnsiChar;
1917 begin
1918 if (pr.tokType = pr.TTEOF) then raise TDynParseException.Create(pr, 'field value expected');
1919 if (pr.tokType = pr.TTSemi) then raise TDynParseException.Create(pr, 'extra semicolon');
1920 // if this field should contain struct, convert type and parse struct
1921 case mEBS of
1922 TEBS.TNone: begin end;
1923 TEBS.TRec:
1924 begin
1925 // ugly hack. sorry.
1926 if (mType = TType.TTrigData) then
1927 begin
1928 pr.expectTT(pr.TTBegin);
1929 if (pr.tokType = pr.TTEnd) then
1930 begin
1931 // '{}'
1932 mRecRef := nil;
1933 pr.expectTT(pr.TTEnd);
1934 end
1935 else
1936 begin
1937 rec := mOwner;
1938 // find trigger definition
1939 tfld := rec.trigTypeField();
1940 if (tfld = nil) then raise TDynParseException.CreateFmt(pr, 'triggerdata value for field ''%s'' in record ''%s'' without ''type'' field', [mName, rec.mTypeName]);
1941 rc := mOwner.mOwner.trigTypeFor[tfld.mSVal]; // find in mapdef
1942 if (rc = nil) then raise TDynParseException.CreateFmt(pr, 'triggerdata definition for field ''%s'' in record ''%s'' with type ''%s'' not found', [mName, rec.mTypeName, tfld.mSVal]);
1943 rc := rc.clone(mOwner.mHeaderRec);
1944 rc.mHeaderRec := mOwner.mHeaderRec;
1945 //writeln(rc.definition);
1946 // on error, it will be freed by memowner
1947 rc.parseValue(pr, true);
1948 mRecRef := rc;
1949 end;
1950 mDefined := true;
1951 pr.eatTT(pr.TTSemi); // hack: allow (but don't require) semicolon after inline records
1952 exit;
1953 end;
1954 // other record types
1955 if (pr.tokType = pr.TTId) then
1956 begin
1957 if pr.eatId('null') then
1958 begin
1959 mRecRef := nil;
1960 end
1961 else
1962 begin
1963 rec := mOwner.findRecordByTypeId(mEBSTypeName, pr.tokStr);
1964 if (rec = nil) then
1965 begin
1966 mRecRefId := pr.tokStr;
1967 end
1968 else
1969 begin
1970 mRecRef := rec;
1971 mRecRefId := '';
1972 end;
1973 pr.expectId();
1974 end;
1975 mDefined := true;
1976 pr.expectTT(pr.TTSemi);
1977 exit;
1978 end
1979 else if (pr.tokType = pr.TTBegin) then
1980 begin
1981 //rec := mOwner.mOwner.recType[mEBSTypeName]; // find in mapdef
1982 rec := nil;
1983 if (mEBSType <> nil) and (mEBSType is TDynRecord) then rec := (mEBSType as TDynRecord);
1984 if (rec = nil) then raise TDynParseException.CreateFmt(pr, 'record type ''%s'' for field ''%s'' not found', [mEBSTypeName, mName]);
1985 rc := rec.clone(mOwner.mHeaderRec);
1986 rc.mHeaderRec := mOwner.mHeaderRec;
1987 rc.parseValue(pr);
1988 mRecRef := rc;
1989 mDefined := true;
1990 if mOwner.addRecordByType(mEBSTypeName, rc) then
1991 begin
1992 raise TDynParseException.CreateFmt(pr, 'duplicate record with id ''%s'' for field ''%s'' in record ''%s''', [rc.mId, mName, mOwner.mTypeName]);
1993 end;
1994 pr.eatTT(pr.TTSemi); // hack: allow (but don't require) semicolon after inline records
1995 exit;
1996 end;
1997 pr.expectTT(pr.TTBegin);
1998 end;
1999 TEBS.TEnum:
2000 begin
2001 //es := mOwner.mOwner.ebsType[mEBSTypeName]; // find in mapdef
2002 es := nil;
2003 if (mEBSType <> nil) and (mEBSType is TDynEBS) then es := (mEBSType as TDynEBS);
2004 if (es = nil) or (not es.mIsEnum) then raise TDynParseException.CreateFmt(pr, 'record enum type ''%s'' for field ''%s'' not found', [mEBSTypeName, mName]);
2005 tk := pr.expectId();
2006 if not es.has[tk] then raise TDynParseException.CreateFmt(pr, 'record enum value ''%s'' of type ''%s'' for field ''%s'' not found', [tk, mEBSTypeName, mName]);
2007 mIVal := es.field[tk];
2008 mSVal := tk;
2009 //writeln('ENUM ', mEBSName, '; element <', mSVal, '> with value ', mIVal);
2010 mDefined := true;
2011 pr.expectTT(pr.TTSemi);
2012 exit;
2013 end;
2014 TEBS.TBitSet:
2015 begin
2016 //es := mOwner.mOwner.ebsType[mEBSTypeName]; // find in mapdef
2017 es := nil;
2018 if (mEBSType <> nil) and (mEBSType is TDynEBS) then es := (mEBSType as TDynEBS);
2019 if (es = nil) or es.mIsEnum then raise TDynParseException.CreateFmt(pr, 'record bitset type ''%s'' for field ''%s'' not found', [mEBSTypeName, mName]);
2020 mIVal := 0;
2021 while true do
2022 begin
2023 tk := pr.expectId();
2024 if not es.has[tk] then raise TDynParseException.CreateFmt(pr, 'record bitset value ''%s'' of type ''%s'' for field ''%s'' not found', [tk, mEBSTypeName, mName]);
2025 mIVal := mIVal or es.field[tk];
2026 mSVal := tk;
2027 if (pr.tokType <> pr.TTDelim) or ((pr.tokChar <> '|') and (pr.tokChar <> '+')) then break;
2028 if mBitSetUnique then raise TDynParseException.CreateFmt(pr, 'record bitset of type ''%s'' for field ''%s'' expects only one value', [tk, mEBSTypeName, mName]);
2029 pr.skipToken(); // plus or pipe
2030 end;
2031 mDefined := true;
2032 pr.expectTT(pr.TTSemi);
2033 exit;
2034 end;
2035 else raise TDynParseException.Create(pr, 'ketmar forgot to handle some EBS type');
2036 end;
2038 case mType of
2039 TType.TBool:
2040 begin
2041 if pr.eatId('true') or pr.eatId('tan') or pr.eatId('yes') then mIVal := 1
2042 else if pr.eatId('false') or pr.eatId('ona') or pr.eatId('no') then mIVal := 0
2043 else raise TDynParseException.CreateFmt(pr, 'invalid bool value for field ''%s''', [mName]);
2044 mDefined := true;
2045 pr.expectTT(pr.TTSemi);
2046 exit;
2047 end;
2048 TType.TChar:
2049 begin
2050 if (mMaxDim = 0) then raise TDynParseException.CreateFmt(pr, 'invalid string size definition for field ''%s''', [mName]);
2051 mSVal := pr.expectStr(true);
2052 if (mMaxDim < 0) then
2053 begin
2054 // single char
2055 if (Length(mSVal) <> 1) then raise TDynParseException.CreateFmt(pr, 'invalid string size for field ''%s''', [mName]);
2056 mIVal := Integer(mSVal[1]);
2057 mSVal := '';
2058 end
2059 else
2060 begin
2061 // string
2062 if (Length(mSVal) > mMaxDim) then raise TDynParseException.CreateFmt(pr, 'invalid string size for field ''%s''', [mName]);
2063 end;
2064 mDefined := true;
2065 pr.expectTT(pr.TTSemi);
2066 exit;
2067 end;
2068 TType.TByte:
2069 begin
2070 parseInt(-128, 127);
2071 pr.expectTT(pr.TTSemi);
2072 exit;
2073 end;
2074 TType.TUByte:
2075 begin
2076 parseInt(0, 255);
2077 pr.expectTT(pr.TTSemi);
2078 exit;
2079 end;
2080 TType.TShort:
2081 begin
2082 parseInt(-32768, 32768);
2083 pr.expectTT(pr.TTSemi);
2084 exit;
2085 end;
2086 TType.TUShort:
2087 begin
2088 parseInt(0, 65535);
2089 pr.expectTT(pr.TTSemi);
2090 exit;
2091 end;
2092 TType.TInt:
2093 begin
2094 parseInt(Integer($80000000), $7fffffff);
2095 pr.expectTT(pr.TTSemi);
2096 exit;
2097 end;
2098 TType.TUInt:
2099 begin
2100 parseInt(0, $7fffffff); //FIXME
2101 pr.expectTT(pr.TTSemi);
2102 exit;
2103 end;
2104 TType.TString:
2105 begin
2106 mSVal := pr.expectStr(true);
2107 mDefined := true;
2108 pr.expectTT(pr.TTSemi);
2109 exit;
2110 end;
2111 TType.TPoint,
2112 TType.TSize:
2113 begin
2114 if pr.eatDelim('[') then edim := ']' else begin pr.expectDelim('('); edim := ')'; end;
2115 mIVal := pr.expectInt();
2116 if (mType = TType.TSize) then
2117 begin
2118 if (mIVal < 0) or (mIVal > 65535) then raise TDynParseException.CreateFmt(pr, 'invalid %s value for field ''%s''', [getTypeName(mType), mName]);
2119 end;
2120 mIVal2 := pr.expectInt();
2121 if (mType = TType.TSize) then
2122 begin
2123 if (mIVal2 < 0) or (mIVal2 > 65535) then raise TDynParseException.CreateFmt(pr, 'invalid %s value for field ''%s''', [getTypeName(mType), mName]);
2124 end;
2125 mDefined := true;
2126 pr.expectDelim(edim);
2127 pr.expectTT(pr.TTSemi);
2128 exit;
2129 end;
2130 TType.TColor:
2131 begin
2132 if pr.eatDelim('[') then edim := ']' else begin pr.expectDelim('('); edim := ')'; end;
2133 mIVal := pr.expectInt();
2134 if (mIVal < 0) or (mIVal > 255) then raise TDynParseException.CreateFmt(pr, 'invalid %s value for field ''%s''', [getTypeName(mType), mName]);
2135 mIVal2 := pr.expectInt();
2136 if (mIVal2 < 0) or (mIVal2 > 255) then raise TDynParseException.CreateFmt(pr, 'invalid %s value for field ''%s''', [getTypeName(mType), mName]);
2137 mIVal3 := pr.expectInt();
2138 if (mIVal3 < 0) or (mIVal3 > 255) then raise TDynParseException.CreateFmt(pr, 'invalid %s value for field ''%s''', [getTypeName(mType), mName]);
2139 if (pr.tokType = pr.TTInt) then
2140 begin
2141 mIVal4 := pr.expectInt();
2142 if (mIVal4 < 0) or (mIVal4 > 255) then raise TDynParseException.CreateFmt(pr, 'invalid %s value for field ''%s''', [getTypeName(mType), mName]);
2143 end
2144 else
2145 begin
2146 mIVal4 := 255;
2147 end;
2148 mDefined := true;
2149 pr.expectDelim(edim);
2150 pr.expectTT(pr.TTSemi);
2151 exit;
2152 end;
2153 TType.TList:
2154 begin
2155 assert(false);
2156 exit;
2157 end;
2158 TType.TTrigData:
2159 begin
2160 assert(false);
2161 exit;
2162 end;
2163 else raise TDynParseException.Create(pr, 'ketmar forgot to handle some field type');
2164 end;
2165 raise TDynParseException.CreateFmt(pr, 'cannot parse field ''%s'' yet', [mName]);
2166 end;
2169 // ////////////////////////////////////////////////////////////////////////// //
2170 constructor TDynRecord.Create (pr: TTextParser);
2171 begin
2172 if (pr = nil) then raise TDynParseException.Create(pr, 'cannot create record type without type definition');
2173 mId := '';
2174 mTypeName := '';
2175 mSize := 0;
2176 mFields := TDynFieldList.Create();
2177 {$IF DEFINED(XDYNREC_USE_FIELDHASH)}
2178 mFieldsHash := hashNewStrInt();
2179 {$ENDIF}
2180 mTrigTypes := nil;
2181 mHeader := false;
2182 mHeaderRec := nil;
2183 mBinBlock := -1;
2184 mTagInt := 0;
2185 mTagPtr := nil;
2186 parseDef(pr);
2187 end;
2190 constructor TDynRecord.Create ();
2191 begin
2192 mTypeName := '';
2193 mSize := 0;
2194 mFields := TDynFieldList.Create();
2195 {$IF DEFINED(XDYNREC_USE_FIELDHASH)}
2196 mFieldsHash := hashNewStrInt();
2197 {$ENDIF}
2198 mTrigTypes := nil;
2199 mHeader := false;
2200 mHeaderRec := nil;
2201 mTagInt := 0;
2202 mTagPtr := nil;
2203 mRec2Free := nil;
2204 end;
2207 destructor TDynRecord.Destroy ();
2208 var
2209 fld: TDynField;
2210 rec: TDynRecord;
2211 begin
2212 if (mRec2Free <> nil) then
2213 begin
2214 for rec in mRec2Free do
2215 begin
2216 if (rec <> self) then
2217 begin
2218 //writeln(formatstrf('freeing: 0x%08x; name=%s; id=%s', [Pointer(rec), rec.mName, rec.mId]));
2219 rec.Free();
2220 end;
2221 end;
2222 mRec2Free.Free();
2223 mRec2Free := nil;
2224 end;
2225 mTypeName := '';
2226 for fld in mFields do fld.Free();
2227 mFields.Free();
2228 mFields := nil;
2229 {$IF DEFINED(XDYNREC_USE_FIELDHASH)}
2230 mFieldsHash.Free();
2231 mFieldsHash := nil;
2232 {$ENDIF}
2233 mTrigTypes := nil;
2234 mHeaderRec := nil;
2235 mTagInt := 0;
2236 mTagPtr := nil;
2237 inherited;
2238 end;
2241 procedure TDynRecord.regrec (rec: TDynRecord);
2242 begin
2243 if (rec <> nil) and (rec <> self) then
2244 begin
2245 if (mRec2Free = nil) then mRec2Free := TDynRecList.Create();
2246 mRec2Free.append(rec);
2247 end;
2248 end;
2251 procedure TDynRecord.addField (fld: TDynField); inline;
2252 begin
2253 if (fld = nil) then raise TDynRecException.Create('cannot append nil field to record');
2254 mFields.append(fld);
2255 {$IF DEFINED(XDYNREC_USE_FIELDHASH)}
2256 if (Length(fld.mName) > 0) then mFieldsHash.put(fld.mName, mFields.count-1);
2257 {$ENDIF}
2258 end;
2261 function TDynRecord.addFieldChecked (fld: TDynField): Boolean; inline; // `true`: duplicate name
2262 begin
2263 result := false;
2264 if (fld = nil) then raise TDynRecException.Create('cannot append nil field to record');
2265 {$IF not DEFINED(XDYNREC_USE_FIELDHASH)}
2266 if (Length(fld.mName) > 0) then result := hasByName(fld.mName);
2267 {$ENDIF}
2268 mFields.append(fld);
2269 {$IF DEFINED(XDYNREC_USE_FIELDHASH)}
2270 if (Length(fld.mName) > 0) then result := mFieldsHash.put(fld.mName, mFields.count-1);
2271 {$ENDIF}
2272 end;
2275 function TDynRecord.findByName (const aname: AnsiString): Integer; inline;
2276 begin
2277 {$IF DEFINED(XDYNREC_USE_FIELDHASH)}
2278 if not mFieldsHash.get(aname, result) then result := -1;
2279 {$ELSE}
2280 result := 0;
2281 while (result < mFields.count) do
2282 begin
2283 if StrEqu(aname, mFields[result].mName) then exit;
2284 Inc(result);
2285 end;
2286 result := -1;
2287 {$ENDIF}
2288 end;
2291 function TDynRecord.hasByName (const aname: AnsiString): Boolean; inline;
2292 begin
2293 result := (findByName(aname) >= 0);
2294 end;
2297 function TDynRecord.getFieldByName (const aname: AnsiString): TDynField; inline;
2298 var
2299 f: Integer;
2300 begin
2301 f := findByName(aname);
2302 if (f >= 0) then result := mFields[f] else result := nil;
2303 end;
2306 function TDynRecord.getFieldAt (idx: Integer): TDynField; inline;
2307 begin
2308 if (idx >= 0) and (idx < mFields.count) then result := mFields[idx] else result := nil;
2309 end;
2312 function TDynRecord.getCount (): Integer; inline;
2313 begin
2314 result := mFields.count;
2315 end;
2318 function TDynRecord.getIsTrigData (): Boolean; inline;
2319 begin
2320 result := (Length(mTrigTypes) > 0);
2321 end;
2324 function TDynRecord.getIsForTrig (const aname: AnsiString): Boolean; inline;
2325 var
2326 f: Integer;
2327 begin
2328 result := true;
2329 for f := 0 to High(mTrigTypes) do if StrEqu(mTrigTypes[f], aname) then exit;
2330 result := false;
2331 end;
2334 function TDynRecord.getForTrigCount (): Integer; inline;
2335 begin
2336 result := Length(mTrigTypes);
2337 end;
2340 function TDynRecord.getForTrigAt (idx: Integer): AnsiString; inline;
2341 begin
2342 if (idx >= 0) and (idx < Length(mTrigTypes)) then result := mTrigTypes[idx] else result := '';
2343 end;
2346 function TDynRecord.clone (registerIn: TDynRecord): TDynRecord;
2347 var
2348 fld: TDynField;
2349 f: Integer;
2350 begin
2351 result := TDynRecord.Create();
2352 result.mOwner := mOwner;
2353 result.mId := mId;
2354 result.mTypeName := mTypeName;
2355 result.mTip := mTip;
2356 result.mHelp := mHelp;
2357 result.mSize := mSize;
2358 result.mHeader := mHeader;
2359 result.mBinBlock := mBinBlock;
2360 result.mHeaderRec := mHeaderRec;
2361 result.mTagInt := mTagInt;
2362 result.mTagPtr := mTagPtr;
2363 if (mFields.count > 0) then
2364 begin
2365 result.mFields.capacity := mFields.count;
2366 for fld in mFields do result.addField(fld.clone(result, registerIn));
2367 end;
2368 SetLength(result.mTrigTypes, Length(mTrigTypes));
2369 for f := 0 to High(mTrigTypes) do result.mTrigTypes[f] := mTrigTypes[f];
2370 if (registerIn <> nil) then registerIn.regrec(result);
2371 end;
2374 function TDynRecord.findRecordByTypeId (const atypename, aid: AnsiString): TDynRecord;
2375 var
2376 fld: TDynField;
2377 idx: Integer;
2378 begin
2379 result := nil;
2380 if (Length(aid) = 0) then exit;
2381 // find record data
2382 fld := mHeaderRec.field[atypename];
2383 if (fld = nil) then exit;
2384 if (fld.mType <> fld.TType.TList) then raise TDynRecException.CreateFmt('cannot get record of type ''%s'' due to name conflict with ordinary field', [atypename]);
2385 // find by id
2386 if (fld.mRVal <> nil) then
2387 begin
2388 if fld.mRHash.get(aid, idx) then begin result := fld.mRVal[idx]; exit; end;
2389 end;
2390 // alas
2391 end;
2394 function TDynRecord.findRecordNumByType (const atypename: AnsiString; rc: TDynRecord): Integer;
2395 var
2396 fld: TDynField;
2397 idx: Integer;
2398 begin
2399 result := -1;
2400 // find record data
2401 fld := mHeaderRec.field[atypename];
2402 if (fld = nil) then exit;
2403 if (fld.mType <> fld.TType.TList) then raise TDynRecException.CreateFmt('cannot get record of type ''%s'' due to name conflict with ordinary field', [atypename]);
2404 // find by ref
2405 if (fld.mRVal <> nil) then
2406 begin
2407 for idx := 0 to fld.mRVal.count-1 do
2408 begin
2409 if (fld.mRVal[idx] = rc) then begin result := idx; exit; end;
2410 end;
2411 end;
2412 // alas
2413 end;
2416 function TDynRecord.addRecordByType (const atypename: AnsiString; rc: TDynRecord): Boolean;
2417 var
2418 fld: TDynField;
2419 begin
2420 // find record data
2421 fld := mHeaderRec.field[atypename];
2422 if (fld = nil) then
2423 begin
2424 // first record
2425 fld := TDynField.Create(atypename, TDynField.TType.TList);
2426 fld.mOwner := mHeaderRec;
2427 mHeaderRec.addField(fld);
2428 end;
2429 if (fld.mType <> fld.TType.TList) then raise TDynRecException.CreateFmt('cannot append record of type ''%s'' due to name conflict with ordinary field', [atypename]);
2430 // append
2431 if (fld.mRVal = nil) then
2432 begin
2433 fld.mRVal := TDynRecList.Create();
2434 fld.mRHash := hashNewStrInt();
2435 end;
2436 result := fld.addListItem(rc);
2437 end;
2440 function TDynRecord.isSimpleEqu (rec: TDynRecord): Boolean;
2441 var
2442 f: Integer;
2443 begin
2444 if (rec = nil) then begin result := false; exit; end; // self.mRecRef can't be `nil` here
2445 if (rec = self) then begin result := true; exit; end;
2446 if (mFields.count <> rec.mFields.count) then begin result := false; exit; end;
2447 result := false;
2448 for f := 0 to mFields.count-1 do
2449 begin
2450 if not mFields[f].isSimpleEqu(rec.mFields[f]) then exit;
2451 end;
2452 result := true;
2453 end;
2456 function TDynRecord.trigTypeField (): TDynField;
2457 var
2458 fld: TDynField;
2459 es: TDynEBS = nil;
2460 begin
2461 for fld in mFields do
2462 begin
2463 if (fld.mEBS <> TDynField.TEBS.TEnum) then continue;
2464 if not (fld.mEBSType is TDynEBS) then continue;
2465 es := (fld.mEBSType as TDynEBS);
2466 assert(es <> nil);
2467 if StrEqu(es.mTypeName, 'TriggerType') then begin result := fld; exit; end;
2468 end;
2469 result := nil;
2470 end;
2473 // number of records of the given instance
2474 function TDynRecord.instanceCount (const atypename: AnsiString): Integer;
2475 var
2476 fld: TDynField;
2477 begin
2478 result := 0;
2479 fld := field[atypename];
2480 if (fld <> nil) and (fld.mType = fld.TType.TList) then result := fld.mRVal.count;
2481 end;
2484 function TDynRecord.newTypedRecord (const atypename, aid: AnsiString): TDynRecord;
2485 var
2486 trc: TDynRecord;
2487 fld: TDynField;
2488 begin
2489 if not mHeader then raise TDynRecException.Create('cannot create new records with non-header');
2490 if (Length(aid) = 0) then raise TDynRecException.CreateFmt('cannot create new record of type ''%s'' without id', [atypename]);
2491 trc := mapdef.recType[atypename];
2492 if (trc = nil) then begin result := nil; exit; end;
2493 // check if aid is unique
2494 fld := field[atypename];
2495 if (fld <> nil) and (fld.getListItem(aid) <> nil) then raise TDynRecException.CreateFmt('cannot create record of type ''%s'' with duplicate id ''%s''', [atypename, aid]);
2496 result := trc.clone(self);
2497 result.mId := aid;
2498 addRecordByType(atypename, result);
2499 end;
2502 procedure TDynRecord.clearRefRecs (rec: TDynRecord);
2503 procedure clearRefs (fld: TDynField);
2504 var
2505 rc: TDynRecord;
2506 begin
2507 if (fld = nil) then exit;
2508 if (fld.mRecRef = rec) then fld.mRecRef := nil;
2509 if (fld.mType = fld.TType.TList) then for rc in fld.mRVal do rc.clearRefRecs(rec);
2510 end;
2511 var
2512 fld: TDynField;
2513 begin
2514 if (rec = nil) or (mFields = nil) then exit;
2515 for fld in mFields do clearRefs(fld);
2516 end;
2519 // remove record with the given type and id
2520 // return `true` if record was successfully found and removed
2521 // this will do all necessary recref cleanup too
2522 function TDynRecord.removeTypedRecord (const atypename, aid: AnsiString): Boolean;
2523 var
2524 trc, rec: TDynRecord;
2525 fld: TDynField;
2526 f: Integer;
2527 doFree: Boolean = false;
2528 begin
2529 result := false;
2530 if not mHeader then raise TDynRecException.Create('cannot remove records with non-header');
2531 if (Length(aid) = 0) then exit;
2532 trc := mapdef.recType[atypename];
2533 if (trc = nil) then exit;
2534 fld := field[atypename];
2535 if (fld = nil) then exit;
2536 rec := fld.removeListItem(aid);
2537 if (rec = nil) then exit;
2538 clearRefRecs(rec);
2539 for f := 0 to mRec2Free.count-1 do
2540 begin
2541 if (mRec2Free[f] = rec) then
2542 begin
2543 mRec2Free[f] := nil;
2544 doFree := true;
2545 end;
2546 end;
2547 if doFree then rec.Free();
2548 end;
2551 function TDynRecord.getUserVar (const aname: AnsiString): Variant;
2552 var
2553 fld: TDynField;
2554 begin
2555 fld := getFieldByName(aname);
2556 if (fld = nil) then result := Unassigned else result := fld.value;
2557 end;
2560 procedure TDynRecord.setUserVar (const aname: AnsiString; val: Variant);
2561 var
2562 fld: TDynField;
2563 begin
2564 fld := getFieldByName(aname);
2565 if (fld = nil) then
2566 begin
2567 if (Length(aname) = 0) then raise TDynRecException.Create('cannot create nameless user field');
2568 fld := TDynField.Create(aname, val);
2569 fld.mOwner := self;
2570 fld.mInternal := true;
2571 addField(fld);
2572 end
2573 else
2574 begin
2575 fld.value := val;
2576 end;
2577 end;
2580 procedure TDynRecord.parseDef (pr: TTextParser);
2581 var
2582 fld: TDynField;
2583 tdn: AnsiString;
2584 begin
2585 if pr.eatId('TriggerData') then
2586 begin
2587 pr.expectId('for');
2588 if pr.eatDelim('(') then
2589 begin
2590 while true do
2591 begin
2592 while pr.eatTT(pr.TTComma) do begin end;
2593 if pr.eatDelim(')') then break;
2594 tdn := pr.expectId();
2595 if isForTrig[tdn] then raise TDynParseException.CreateFmt(pr, 'duplicate trigdata ''%s'' trigtype ''%s''', [mTypeName, tdn]);
2596 SetLength(mTrigTypes, Length(mTrigTypes)+1);
2597 mTrigTypes[High(mTrigTypes)] := tdn;
2598 end;
2599 end
2600 else
2601 begin
2602 tdn := pr.expectId();
2603 SetLength(mTrigTypes, 1);
2604 mTrigTypes[0] := tdn;
2605 end;
2606 mTypeName := 'TriggerData';
2607 end
2608 else
2609 begin
2610 mTypeName := pr.expectStrOrId();
2611 while (pr.tokType <> pr.TTBegin) do
2612 begin
2613 if pr.eatId('header') then begin mHeader := true; continue; end;
2614 if pr.eatId('size') then
2615 begin
2616 if (mSize > 0) then raise TDynParseException.CreateFmt(pr, 'duplicate `size` in record ''%s''', [mTypeName]);
2617 mSize := pr.expectInt();
2618 if (mSize < 1) then raise TDynParseException.CreateFmt(pr, 'invalid record ''%s'' size: %d', [mTypeName, mSize]);
2619 pr.expectId('bytes');
2620 continue;
2621 end;
2622 if pr.eatId('binblock') then
2623 begin
2624 if (mBinBlock >= 0) then raise TDynParseException.CreateFmt(pr, 'duplicate `binblock` in record ''%s''', [mTypeName]);
2625 mBinBlock := pr.expectInt();
2626 if (mBinBlock < 1) then raise TDynParseException.CreateFmt(pr, 'invalid record ''%s'' binblock: %d', [mTypeName, mBinBlock]);
2627 continue;
2628 end;
2629 if pr.eatId('tip') then
2630 begin
2631 if (Length(mTip) > 0) then raise TDynParseException.CreateFmt(pr, 'duplicate tip definition for record ''%s''', [mTypeName]);
2632 mTip := pr.expectStr(false);
2633 continue;
2634 end;
2635 if pr.eatId('help') then
2636 begin
2637 if (Length(mHelp) > 0) then raise TDynParseException.CreateFmt(pr, 'duplicate help definition for record ''%s''', [mTypeName]);
2638 mHelp := pr.expectStr(false);
2639 continue;
2640 end;
2641 end;
2642 end;
2644 pr.expectTT(pr.TTBegin);
2645 // load fields
2646 while (pr.tokType <> pr.TTEnd) do
2647 begin
2648 fld := TDynField.Create(pr);
2649 // append
2650 fld.mOwner := self;
2651 if addFieldChecked(fld) then
2652 begin
2653 fld.Free();
2654 raise TDynParseException.CreateFmt(pr, 'duplicate field ''%s''', [fld.name]);
2655 end;
2656 // done with field
2657 end;
2658 pr.expectTT(pr.TTEnd);
2659 end;
2662 function TDynRecord.definition (): AnsiString;
2663 var
2664 f: Integer;
2665 begin
2666 if isTrigData then
2667 begin
2668 // trigger data
2669 result := 'TriggerData for ';
2670 if (Length(mTrigTypes) > 1) then
2671 begin
2672 result += '(';
2673 for f := 0 to High(mTrigTypes) do
2674 begin
2675 if (f <> 0) then result += ', ';
2676 result += mTrigTypes[f];
2677 end;
2678 result += ')';
2679 end
2680 else
2681 begin
2682 result += mTrigTypes[0];
2683 end;
2684 end
2685 else
2686 begin
2687 // record
2688 result := quoteStr(mTypeName);
2689 if (mSize >= 0) then result += Format(' size %d bytes', [mSize]);
2690 if mHeader then result += ' header';
2691 end;
2692 result += ' {'#10;
2693 for f := 0 to mFields.count-1 do
2694 begin
2695 result += ' ';
2696 result += mFields[f].definition;
2697 result += ';'#10;
2698 end;
2699 result += '}';
2700 end;
2703 procedure TDynRecord.parseBinValue (st: TStream; forceData: Boolean=false);
2704 var
2705 sign: string[4];
2706 btype: Integer;
2707 bsize: Integer;
2708 buf: PByte = nil;
2709 loaded: array[0..255] of Boolean;
2710 rec, rect: TDynRecord;
2711 fld: TDynField;
2712 f: Integer;
2713 mst: TSFSMemoryChunkStream = nil;
2715 procedure linkNames (rec: TDynRecord);
2716 var
2717 fld: TDynField;
2718 rt: TDynRecord;
2719 begin
2720 //writeln('*** rec: ', rec.mName, '.', rec.mId, ' (', rec.mFields.count, ')');
2721 for fld in rec.mFields do
2722 begin
2723 if (fld.mType = TDynField.TType.TTrigData) then
2724 begin
2725 if (fld.mRecRef <> nil) then linkNames(fld.mRecRef);
2726 continue;
2727 end;
2728 if (Length(fld.mRecRefId) = 0) then continue;
2729 assert(fld.mEBSType <> nil);
2730 rt := findRecordByTypeId(fld.mEBSTypeName, fld.mRecRefId);
2731 if (rt = nil) then
2732 begin
2733 e_LogWritefln('record of type ''%s'' with id ''%s'' links to inexistant record of type ''%s'' with id ''%s''', [rec.mTypeName, rec.mId, fld.mEBSTypeName, fld.mRecRefId], MSG_WARNING);
2734 //raise TDynRecException.CreateFmt('record of type ''%s'' with id ''%s'' links to inexistant record of type ''%s'' with id ''%s''', [rec.mName, rec.mId, fld.mEBSTypeName, fld.mRecRefId]);
2735 end;
2736 //writeln(' ', rec.mName, '.', rec.mId, ':', fld.mName, ' -> ', rt.mName, '.', rt.mId, ' (', fld.mEBSTypeName, '.', fld.mRecRefId, ')');
2737 fld.mRecRefId := '';
2738 fld.mRecRef := rt;
2739 fld.mDefined := true;
2740 end;
2741 for fld in rec.mFields do
2742 begin
2743 //writeln(' ', fld.mName);
2744 fld.fixDefaultValue(); // just in case
2745 end;
2746 end;
2748 begin
2749 for f := 0 to High(loaded) do loaded[f] := false;
2750 mst := TSFSMemoryChunkStream.Create(nil, 0);
2751 try
2752 if mHeader and not forceData then
2753 begin
2754 // parse map file as sequence of blocks
2755 sign[0] := #4;
2756 st.ReadBuffer(sign[1], 4);
2757 if (sign <> 'MAP'#1) then raise TDynRecException.Create('invalid binary map signature');
2758 // parse blocks
2759 while (st.position < st.size) do
2760 begin
2761 btype := readByte(st);
2762 if (btype = 0) then break; // no more blocks
2763 readLongWord(st); // reserved
2764 bsize := readLongInt(st);
2765 {$IF DEFINED(D2D_XDYN_DEBUG)}writeln('btype=', btype, '; bsize=', bsize);{$ENDIF}
2766 if (bsize < 0) or (bsize > $1fffffff) then raise TDynRecException.CreateFmt('block of type %d has invalid size %d', [btype, bsize]);
2767 if loaded[btype] then raise TDynRecException.CreateFmt('block of type %d already loaded', [btype]);
2768 loaded[btype] := true;
2769 // find record type for this block
2770 rect := nil;
2771 for rec in mOwner.recTypes do if (rec.mBinBlock = btype) then begin rect := rec; break; end;
2772 if (rect = nil) then raise TDynRecException.CreateFmt('block of type %d has no corresponding record', [btype]);
2773 //writeln('found type ''', rec.mName, ''' for block type ', btype);
2774 if (rec.mSize = 0) or ((bsize mod rec.mSize) <> 0) then raise TDynRecException.CreateFmt('block of type %d has invalid number of records', [btype]);
2775 // header?
2776 if (rect.mHeader) then
2777 begin
2778 if (bsize <> mSize) then raise TDynRecException.CreateFmt('header block of type %d has invalid number of records', [btype]);
2779 GetMem(buf, bsize);
2780 st.ReadBuffer(buf^, bsize);
2781 mst.setup(buf, mSize);
2782 parseBinValue(mst, true); // force parsing data
2783 end
2784 else
2785 begin
2786 // create list for this type
2787 fld := TDynField.Create(rec.mTypeName, TDynField.TType.TList);
2788 fld.mOwner := self;
2789 addField(fld);
2790 if (bsize > 0) then
2791 begin
2792 GetMem(buf, bsize);
2793 st.ReadBuffer(buf^, bsize);
2794 for f := 0 to (bsize div rec.mSize)-1 do
2795 begin
2796 mst.setup(buf+f*rec.mSize, rec.mSize);
2797 rec := rect.clone(self);
2798 rec.mHeaderRec := self;
2799 rec.parseBinValue(mst);
2800 rec.mId := Format('%s%d', [rec.mTypeName, f]);
2801 fld.addListItem(rec);
2802 //writeln('parsed ''', rec.mId, '''...');
2803 end;
2804 end;
2805 end;
2806 FreeMem(buf);
2807 buf := nil;
2808 //st.position := st.position+bsize;
2809 end;
2810 // link fields
2811 for fld in mFields do
2812 begin
2813 if (fld.mType <> TDynField.TType.TList) then continue;
2814 for rec in fld.mRVal do linkNames(rec);
2815 end;
2816 exit;
2817 end;
2819 // read fields
2820 if StrEqu(mTypeName, 'TriggerData') then mSize := Integer(st.size-st.position);
2821 if (mSize < 1) then raise TDynRecException.CreateFmt('cannot read record of type ''%s'' with unknown size', [mTypeName]);
2822 GetMem(buf, mSize);
2823 st.ReadBuffer(buf^, mSize);
2824 for fld in mFields do
2825 begin
2826 if fld.mInternal then continue;
2827 if (fld.mBinOfs < 0) then continue;
2828 if (fld.mBinOfs >= st.size) then raise TDynRecException.CreateFmt('record of type ''%s'' has invalid field ''%s''', [fld.mName]);
2829 mst.setup(buf+fld.mBinOfs, mSize-fld.mBinOfs);
2830 //writeln('parsing ''', mName, '.', fld.mName, '''...');
2831 fld.parseBinValue(mst);
2832 end;
2833 finally
2834 mst.Free();
2835 if (buf <> nil) then FreeMem(buf);
2836 end;
2837 end;
2840 procedure TDynRecord.writeBinTo (var hasLostData: Boolean; st: TStream; trigbufsz: Integer=-1; onlyFields: Boolean=false);
2841 var
2842 fld: TDynField;
2843 rec, rv: TDynRecord;
2844 buf: PByte = nil;
2845 ws: TStream = nil;
2846 blk, blkmax: Integer;
2847 bufsz: Integer = 0;
2848 blksz: Integer;
2849 begin
2850 if (trigbufsz < 0) then
2851 begin
2852 if (mBinBlock < 1) then raise TDynRecException.Create('cannot write binary record without block number');
2853 if (mSize < 1) then raise TDynRecException.Create('cannot write binary record without size');
2854 bufsz := mSize;
2855 end
2856 else
2857 begin
2858 bufsz := trigbufsz;
2859 end;
2860 try
2861 GetMem(buf, bufsz);
2862 FillChar(buf^, bufsz, 0);
2863 ws := TSFSMemoryChunkStream.Create(buf, bufsz);
2865 // write normal fields
2866 for fld in mFields do
2867 begin
2868 // record list?
2869 if (fld.mType = fld.TType.TList) then continue; // later
2870 if fld.mInternal then continue;
2871 if (fld.mBinOfs < 0) then
2872 begin
2873 if not fld.equToDefault then hasLostData := true;
2874 continue;
2875 end;
2876 if (fld.mBinOfs >= bufsz) then raise TDynRecException.Create('binary value offset is outside of the buffer');
2877 TSFSMemoryChunkStream(ws).setup(buf+fld.mBinOfs, bufsz-fld.mBinOfs);
2878 //writeln('writing field <', fld.mName, '>');
2879 fld.writeBinTo(hasLostData, ws);
2880 end;
2882 // write block with normal fields
2883 if mHeader and not onlyFields then
2884 begin
2885 //writeln('writing header...');
2886 // signature and version
2887 writeIntBE(st, LongWord($4D415001));
2888 writeInt(st, Byte(mBinBlock)); // type
2889 writeInt(st, LongWord(0)); // reserved
2890 writeInt(st, LongWord(bufsz)); // size
2891 end;
2892 st.WriteBuffer(buf^, bufsz);
2894 ws.Free(); ws := nil;
2895 FreeMem(buf); buf := nil;
2897 // write other blocks, if any
2898 if mHeader and not onlyFields then
2899 begin
2900 // calculate blkmax
2901 blkmax := 0;
2902 for fld in mFields do
2903 begin
2904 // record list?
2905 if (fld.mType = fld.TType.TList) then
2906 begin
2907 if (fld.mRVal = nil) or (fld.mRVal.count = 0) then continue;
2908 rec := mOwner.recType[fld.mName];
2909 if (rec = nil) then continue;
2910 if (rec.mBinBlock <= 0) then continue;
2911 if (blkmax < rec.mBinBlock) then blkmax := rec.mBinBlock;
2912 end;
2913 end;
2914 // write blocks
2915 for blk := 1 to blkmax do
2916 begin
2917 if (blk = mBinBlock) then continue;
2918 ws := nil;
2919 for fld in mFields do
2920 begin
2921 // record list?
2922 if (fld.mType = fld.TType.TList) then
2923 begin
2924 if (fld.mRVal = nil) or (fld.mRVal.count = 0) then continue;
2925 rec := mOwner.recType[fld.mName];
2926 if (rec = nil) then continue;
2927 if (rec.mBinBlock <> blk) then continue;
2928 if (ws = nil) then ws := TMemoryStream.Create();
2929 for rv in fld.mRVal do rv.writeBinTo(hasLostData, ws);
2930 end;
2931 end;
2932 // flush block
2933 if (ws <> nil) then
2934 begin
2935 blksz := Integer(ws.position);
2936 ws.position := 0;
2937 writeInt(st, Byte(blk)); // type
2938 writeInt(st, LongWord(0)); // reserved
2939 writeInt(st, LongWord(blksz)); // size
2940 st.CopyFrom(ws, blksz);
2941 ws.Free();
2942 ws := nil;
2943 end;
2944 end;
2945 // write end marker
2946 writeInt(st, Byte(0));
2947 writeInt(st, LongWord(0));
2948 writeInt(st, LongWord(0));
2949 end;
2950 finally
2951 ws.Free();
2952 if (buf <> nil) then FreeMem(buf);
2953 end;
2954 end;
2957 procedure TDynRecord.writeTo (wr: TTextWriter; putHeader: Boolean=true);
2958 var
2959 fld: TDynField;
2960 rec: TDynRecord;
2961 putTypeComment: Boolean;
2962 f: Integer;
2963 begin
2964 if putHeader then
2965 begin
2966 wr.put(mTypeName);
2967 if (Length(mId) > 0) then begin wr.put(' '); wr.put(mId); end;
2968 wr.put(' ');
2969 end;
2970 wr.put('{'#10);
2971 wr.indent();
2972 try
2973 for fld in mFields do
2974 begin
2975 // record list?
2976 if (fld.mType = fld.TType.TList) then
2977 begin
2978 if not mHeader then raise TDynRecException.Create('record list in non-header record');
2979 if (fld.mRVal <> nil) and (fld.mRVal.count > 0) then
2980 begin
2981 putTypeComment := true;
2982 for rec in fld.mRVal do
2983 begin
2984 if (rec = nil) or (Length(rec.mId) = 0) then continue;
2985 if putTypeComment then
2986 begin
2987 wr.put(#10);
2988 if (80-wr.curIndent*2 >= 2) then
2989 begin
2990 wr.putIndent();
2991 for f := wr.curIndent to 80-wr.curIndent do wr.put('/');
2992 wr.put(#10);
2993 end;
2994 putTypeComment := false;
2995 wr.putIndent();
2996 wr.put('// ');
2997 wr.put(fld.name);
2998 wr.put(#10);
2999 end
3000 else
3001 begin
3002 wr.put(#10);
3003 end;
3004 wr.putIndent();
3005 rec.writeTo(wr, true);
3006 end;
3007 end;
3008 continue;
3009 end;
3010 if fld.mInternal then continue;
3011 if (not fld.mWriteDef) and fld.isDefaultValue then continue;
3012 wr.putIndent();
3013 fld.writeTo(wr);
3014 end;
3015 finally
3016 wr.unindent();
3017 end;
3018 wr.putIndent();
3019 wr.put('}'#10);
3020 end;
3023 {$IF DEFINED(D2D_DYNREC_PROFILER)}
3024 var
3025 profCloneRec: UInt64 = 0;
3026 profFindRecType: UInt64 = 0;
3027 profFieldSearching: UInt64 = 0;
3028 profListDupChecking: UInt64 = 0;
3029 profAddRecByType: UInt64 = 0;
3030 profFieldValParsing: UInt64 = 0;
3031 profFixDefaults: UInt64 = 0;
3032 profRecValParse: UInt64 = 0;
3034 procedure xdynDumpProfiles ();
3035 begin
3036 writeln('=== XDYNREC PROFILES ===');
3037 writeln('record cloning: ', profCloneRec div 1000, '.', profCloneRec mod 1000, ' milliseconds');
3038 writeln('findRecType : ', profFindRecType div 1000, '.', profFindRecType mod 1000, ' milliseconds');
3039 writeln('field[] : ', profFieldSearching div 1000, '.', profFieldSearching mod 1000, ' milliseconds');
3040 writeln('list dup check: ', profListDupChecking div 1000, '.', profListDupChecking mod 1000, ' milliseconds');
3041 writeln('addRecByType : ', profAddRecByType div 1000, '.', profAddRecByType mod 1000, ' milliseconds');
3042 writeln('field valparse: ', profFieldValParsing div 1000, '.', profFieldValParsing mod 1000, ' milliseconds');
3043 writeln('fix defaults : ', profFixDefaults div 1000, '.', profFixDefaults mod 1000, ' milliseconds');
3044 writeln('recvalparse : ', profRecValParse div 1000, '.', profRecValParse mod 1000, ' milliseconds');
3045 end;
3046 {$ENDIF}
3049 procedure TDynRecord.parseValue (pr: TTextParser; beginEaten: Boolean=false);
3050 var
3051 fld: TDynField;
3052 rec: TDynRecord = nil;
3053 trc{, rv}: TDynRecord;
3054 {$IF DEFINED(D2D_DYNREC_PROFILER)}
3055 stt, stall: UInt64;
3056 {$ENDIF}
3058 procedure linkNames (rec: TDynRecord);
3059 var
3060 fld: TDynField;
3061 rt: TDynRecord;
3062 begin
3063 //writeln('*** rec: ', rec.mName, '.', rec.mId, ' (', rec.mFields.count, ')');
3064 for fld in rec.mFields do
3065 begin
3066 if (fld.mType = TDynField.TType.TTrigData) then
3067 begin
3068 if (fld.mRecRef <> nil) then linkNames(fld.mRecRef);
3069 continue;
3070 end;
3071 if (Length(fld.mRecRefId) = 0) then continue;
3072 assert(fld.mEBSType <> nil);
3073 rt := findRecordByTypeId(fld.mEBSTypeName, fld.mRecRefId);
3074 if (rt = nil) then
3075 begin
3076 //e_LogWritefln('record of type ''%s'' with id ''%s'' links to inexistant record of type ''%s'' with id ''%s''', [rec.mName, rec.mId, fld.mEBSTypeName, fld.mRecRefId], MSG_WARNING);
3077 raise TDynParseException.CreateFmt(pr, 'record of type ''%s'' with id ''%s'' links to inexistant record of type ''%s'' with id ''%s''', [rec.mTypeName, rec.mId, fld.mEBSTypeName, fld.mRecRefId]);
3078 end;
3079 //writeln(' ', rec.mName, '.', rec.mId, ':', fld.mName, ' -> ', rt.mName, '.', rt.mId, ' (', fld.mEBSTypeName, '.', fld.mRecRefId, ')');
3080 fld.mRecRefId := '';
3081 fld.mRecRef := rt;
3082 fld.mDefined := true;
3083 end;
3084 for fld in rec.mFields do
3085 begin
3086 //writeln(' ', fld.mName);
3087 fld.fixDefaultValue(); // just in case
3088 end;
3089 end;
3091 begin
3092 if (mOwner = nil) then raise TDynParseException.CreateFmt(pr, 'can''t parse record ''%s'' value without owner', [mTypeName]);
3094 {$IF DEFINED(D2D_DYNREC_PROFILER)}stall := curTimeMicro();{$ENDIF}
3096 // not a header?
3097 if not mHeader then
3098 begin
3099 // id?
3100 if (not beginEaten) and (pr.tokType = pr.TTId) then mId := pr.expectId();
3101 end
3102 else
3103 begin
3104 assert(mHeaderRec = self);
3105 end;
3107 //writeln('parsing record <', mName, '>');
3108 if not beginEaten then pr.expectTT(pr.TTBegin);
3109 while (pr.tokType <> pr.TTEnd) do
3110 begin
3111 if (pr.tokType <> pr.TTId) then raise TDynParseException.Create(pr, 'identifier expected');
3112 //writeln('<', mName, '.', pr.tokStr, '>');
3114 // records
3115 if mHeader then
3116 begin
3117 // add records with this type (if any)
3118 {$IF DEFINED(D2D_DYNREC_PROFILER)}stt := curTimeMicro();{$ENDIF}
3119 trc := mOwner.recType[pr.tokStr];
3120 {$IF DEFINED(D2D_DYNREC_PROFILER)}profFindRecType := curTimeMicro()-stt;{$ENDIF}
3121 if (trc <> nil) then
3122 begin
3123 {$IF DEFINED(D2D_DYNREC_PROFILER)}stt := curTimeMicro();{$ENDIF}
3124 rec := trc.clone(mHeaderRec);
3125 {$IF DEFINED(D2D_DYNREC_PROFILER)}profCloneRec := curTimeMicro()-stt;{$ENDIF}
3126 rec.mHeaderRec := mHeaderRec;
3127 // on error, it will be freed by memowner
3128 pr.skipToken();
3129 rec.parseValue(pr);
3130 {$IF DEFINED(D2D_DYNREC_PROFILER)}stt := curTimeMicro();{$ENDIF}
3131 addRecordByType(rec.mTypeName, rec);
3132 {$IF DEFINED(D2D_DYNREC_PROFILER)}profAddRecByType := curTimeMicro()-stt;{$ENDIF}
3133 continue;
3134 end;
3135 end;
3137 // fields
3138 {$IF DEFINED(D2D_DYNREC_PROFILER)}stt := curTimeMicro();{$ENDIF}
3139 //writeln('0: <', mName, '.', pr.tokStr, '>');
3140 fld := field[pr.tokStr];
3141 //writeln('1: <', mName, '.', pr.tokStr, '>');
3142 {$IF DEFINED(D2D_DYNREC_PROFILER)}profFieldSearching := curTimeMicro()-stt;{$ENDIF}
3143 if (fld <> nil) then
3144 begin
3145 //writeln('2: <', mName, '.', pr.tokStr, '>');
3146 if fld.defined then raise TDynParseException.CreateFmt(pr, 'duplicate field ''%s'' in record ''%s''', [fld.mName, mTypeName]);
3147 if fld.internal then raise TDynParseException.CreateFmt(pr, 'internal field ''%s'' in record ''%s''', [fld.mName, mTypeName]);
3148 pr.skipToken(); // skip field name
3149 //writeln('3: <', mName, '.', pr.tokStr, '>:', pr.tokType);
3150 {$IF DEFINED(D2D_DYNREC_PROFILER)}stt := curTimeMicro();{$ENDIF}
3151 fld.parseValue(pr);
3152 {$IF DEFINED(D2D_DYNREC_PROFILER)}profFieldValParsing := curTimeMicro()-stt;{$ENDIF}
3153 continue;
3154 end;
3156 // something is wrong
3157 raise TDynParseException.CreateFmt(pr, 'unknown field ''%s'' in record ''%s''', [pr.tokStr, mTypeName]);
3158 end;
3159 pr.expectTT(pr.TTEnd);
3161 if mHeader then
3162 begin
3163 // link fields
3164 for fld in mFields do
3165 begin
3166 if (fld.mType <> TDynField.TType.TList) then continue;
3167 for rec in fld.mRVal do linkNames(rec);
3168 end;
3169 end;
3171 // fix field defaults
3172 {$IF DEFINED(D2D_DYNREC_PROFILER)}stt := curTimeMicro();{$ENDIF}
3173 for fld in mFields do fld.fixDefaultValue();
3174 {$IF DEFINED(D2D_DYNREC_PROFILER)}profFixDefaults := curTimeMicro()-stt;{$ENDIF}
3175 //writeln('done parsing record <', mName, '>');
3176 //{$IF DEFINED(D2D_DYNREC_PROFILER)}writeln('stall: ', curTimeMicro()-stall);{$ENDIF}
3177 {$IF DEFINED(D2D_DYNREC_PROFILER)}profRecValParse := curTimeMicro()-stall;{$ENDIF}
3178 end;
3181 // ////////////////////////////////////////////////////////////////////////// //
3182 constructor TDynEBS.Create (pr: TTextParser);
3183 begin
3184 cleanup();
3185 parseDef(pr);
3186 end;
3189 destructor TDynEBS.Destroy ();
3190 begin
3191 cleanup();
3192 inherited;
3193 end;
3196 procedure TDynEBS.cleanup ();
3197 begin
3198 mIsEnum := false;
3199 mTypeName := '';
3200 mTip := '';
3201 mHelp := '';
3202 mIds := nil;
3203 mVals := nil;
3204 mMaxName := '';
3205 mMaxVal := 0;
3206 end;
3209 function TDynEBS.findByName (const aname: AnsiString): Integer;
3210 begin
3211 result := 0;
3212 while (result < Length(mIds)) do
3213 begin
3214 if StrEqu(aname, mIds[result]) then exit;
3215 Inc(result);
3216 end;
3217 result := -1;
3218 end;
3221 function TDynEBS.hasByName (const aname: AnsiString): Boolean; inline;
3222 begin
3223 result := (findByName(aname) >= 0);
3224 end;
3227 function TDynEBS.getFieldByName (const aname: AnsiString): Integer; inline;
3228 var
3229 f: Integer;
3230 begin
3231 f := findByName(aname);
3232 if (f >= 0) then result := mVals[f] else result := 0;
3233 end;
3236 function TDynEBS.definition (): AnsiString;
3237 var
3238 f, cv: Integer;
3239 begin
3240 if mIsEnum then result :='enum ' else result := 'bitset ';
3241 result += mTypeName;
3242 result += ' {'#10;
3243 // fields
3244 if mIsEnum then cv := 0 else cv := 1;
3245 for f := 0 to High(mIds) do
3246 begin
3247 if (mIds[f] = mMaxName) then continue;
3248 result += ' '+mIds[f];
3249 if (mVals[f] <> cv) then
3250 begin
3251 result += Format(' = %d', [mVals[f]]);
3252 if mIsEnum then cv := mVals[f];
3253 result += ','#10;
3254 end
3255 else
3256 begin
3257 result += Format(', // %d'#10, [mVals[f]]);
3258 end;
3259 if mIsEnum then Inc(cv) else if (mVals[f] = cv) then cv := cv shl 1;
3260 end;
3261 // max field
3262 if (Length(mMaxName) > 0) then result += ' '+mMaxName+' = MAX,'#10;
3263 result += '}';
3264 end;
3267 function TDynEBS.pasdef (): AnsiString;
3268 var
3269 f: Integer;
3270 begin
3271 result := '// '+mTypeName+#10'const'#10;
3272 // fields
3273 for f := 0 to High(mIds) do
3274 begin
3275 result += formatstrf(' %s = %d;'#10, [mIds[f], mVals[f]]);
3276 end;
3277 end;
3280 function TDynEBS.nameByValue (v: Integer): AnsiString;
3281 var
3282 f: Integer;
3283 begin
3284 for f := 0 to High(mVals) do
3285 begin
3286 if (mVals[f] = v) then begin result := mIds[f]; exit; end;
3287 end;
3288 result := '';
3289 end;
3292 procedure TDynEBS.parseDef (pr: TTextParser);
3293 var
3294 idname: AnsiString;
3295 cv, v: Integer;
3296 f: Integer;
3297 skipAdd: Boolean;
3298 hasV: Boolean;
3299 begin
3300 if pr.eatId('enum') then mIsEnum := true
3301 else if pr.eatId('bitset') then mIsEnum := false
3302 else pr.expectId('enum');
3303 mTypeName := pr.expectId();
3304 mMaxVal := Integer($80000000);
3305 if mIsEnum then cv := 0 else cv := 1;
3306 while (pr.tokType <> pr.TTBegin) do
3307 begin
3308 if pr.eatId('tip') then
3309 begin
3310 if (Length(mTip) > 0) then raise TDynParseException.CreateFmt(pr, 'duplicate tip definition for enum/bitset ''%s''', [mTypeName]);
3311 mTip := pr.expectStr(false);
3312 continue;
3313 end;
3314 if pr.eatId('help') then
3315 begin
3316 if (Length(mHelp) > 0) then raise TDynParseException.CreateFmt(pr, 'duplicate help definition for enum/bitset ''%s''', [mTypeName]);
3317 mHelp := pr.expectStr(false);
3318 continue;
3319 end;
3320 break;
3321 end;
3322 pr.expectTT(pr.TTBegin);
3323 while (pr.tokType <> pr.TTEnd) do
3324 begin
3325 idname := pr.expectId();
3326 for f := 0 to High(mIds) do
3327 begin
3328 if StrEqu(mIds[f], idname) then raise TDynParseException.CreateFmt(pr, 'duplicate field ''%s'' in enum/bitset ''%s''', [idname, mTypeName]);
3329 end;
3330 if StrEqu(mMaxName, idname) then raise TDynParseException.CreateFmt(pr, 'duplicate field ''%s'' in enum/bitset ''%s''', [idname, mTypeName]);
3331 skipAdd := false;
3332 hasV := false;
3333 v := cv;
3334 // has value?
3335 if pr.eatDelim('=') then
3336 begin
3337 if pr.eatId('MAX') then
3338 begin
3339 if (Length(mMaxName) > 0) then raise TDynParseException.CreateFmt(pr, 'duplicate max field ''%s'' in enum/bitset ''%s''', [idname, mTypeName]);
3340 mMaxName := idname;
3341 skipAdd := true;
3342 end
3343 else
3344 begin
3345 v := pr.expectInt();
3346 if mIsEnum then cv := v;
3347 hasV := true;
3348 end;
3349 end;
3350 // append it?
3351 if not skipAdd then
3352 begin
3353 // fix maxvalue
3354 if mIsEnum or (not hasV) then
3355 begin
3356 if (mMaxVal < v) then mMaxVal := v;
3357 end;
3358 SetLength(mIds, Length(mIds)+1);
3359 mIds[High(mIds)] := idname;
3360 SetLength(mVals, Length(mIds));
3361 mVals[High(mVals)] := v;
3362 // next cv
3363 if mIsEnum or (not hasV) then
3364 begin
3365 if mIsEnum then Inc(cv) else cv := cv shl 1;
3366 end;
3367 end;
3368 if (pr.tokType = pr.TTEnd) then break;
3369 pr.expectTT(pr.TTComma);
3370 while pr.eatTT(pr.TTComma) do begin end;
3371 end;
3372 pr.expectTT(pr.TTEnd);
3373 // add max field
3374 if (Length(mMaxName) > 0) then
3375 begin
3376 SetLength(mIds, Length(mIds)+1);
3377 mIds[High(mIds)] := mMaxName;
3378 SetLength(mVals, Length(mIds));
3379 mVals[High(mVals)] := mMaxVal;
3380 end;
3381 end;
3384 // ////////////////////////////////////////////////////////////////////////// //
3385 constructor TDynMapDef.Create (pr: TTextParser);
3386 begin
3387 recTypes := TDynRecList.Create();
3388 trigTypes := TDynRecList.Create();
3389 ebsTypes := TDynEBSList.Create();
3390 parseDef(pr);
3391 end;
3394 destructor TDynMapDef.Destroy ();
3395 var
3396 rec: TDynRecord;
3397 ebs: TDynEBS;
3398 begin
3399 //!!!FIXME!!! check who owns trigs and recs!
3400 for rec in recTypes do rec.Free();
3401 for rec in trigTypes do rec.Free();
3402 for ebs in ebsTypes do ebs.Free();
3403 recTypes.Free();
3404 trigTypes.Free();
3405 ebsTypes.Free();
3406 recTypes := nil;
3407 trigTypes := nil;
3408 ebsTypes := nil;
3409 inherited;
3410 end;
3413 function TDynMapDef.getHeaderRecType (): TDynRecord; inline;
3414 begin
3415 if (recTypes.count = 0) then raise TDynRecException.Create('no header in empty mapdef');
3416 result := recTypes[0];
3417 end;
3420 function TDynMapDef.findRecType (const aname: AnsiString): TDynRecord;
3421 var
3422 rec: TDynRecord;
3423 begin
3424 for rec in recTypes do
3425 begin
3426 if StrEqu(rec.typeName, aname) then begin result := rec; exit; end;
3427 end;
3428 result := nil;
3429 end;
3432 function TDynMapDef.findTrigFor (const aname: AnsiString): TDynRecord;
3433 var
3434 rec: TDynRecord;
3435 begin
3436 for rec in trigTypes do
3437 begin
3438 if (rec.isForTrig[aname]) then begin result := rec; exit; end;
3439 end;
3440 result := nil;
3441 end;
3444 function TDynMapDef.findEBSType (const aname: AnsiString): TDynEBS;
3445 var
3446 ebs: TDynEBS;
3447 begin
3448 for ebs in ebsTypes do
3449 begin
3450 if StrEqu(ebs.typeName, aname) then begin result := ebs; exit; end;
3451 end;
3452 result := nil;
3453 end;
3456 procedure TDynMapDef.parseDef (pr: TTextParser);
3457 var
3458 rec, hdr: TDynRecord;
3459 eb: TDynEBS;
3460 f: Integer;
3462 // setup header links and type links
3463 procedure linkRecord (rec: TDynRecord);
3464 var
3465 fld: TDynField;
3466 begin
3467 rec.mHeaderRec := recTypes[0];
3468 for fld in rec.mFields do
3469 begin
3470 if (fld.mType = fld.TType.TTrigData) then continue;
3471 case fld.mEBS of
3472 TDynField.TEBS.TNone: begin end;
3473 TDynField.TEBS.TRec:
3474 begin
3475 fld.mEBSType := findRecType(fld.mEBSTypeName);
3476 if (fld.mEBSType = nil) then raise TDynParseException.CreateFmt(pr, 'field ''%s'' of type ''%s'' has no correcponding record definition', [fld.mName, fld.mEBSTypeName]);
3477 end;
3478 TDynField.TEBS.TEnum,
3479 TDynField.TEBS.TBitSet:
3480 begin
3481 fld.mEBSType := findEBSType(fld.mEBSTypeName);
3482 if (fld.mEBSType = nil) then raise TDynParseException.CreateFmt(pr, 'field ''%s'' of type ''%s'' has no correcponding enum/bitset', [fld.mName, fld.mEBSTypeName]);
3483 if ((fld.mEBS = TDynField.TEBS.TEnum) <> (fld.mEBSType as TDynEBS).mIsEnum) then raise TDynParseException.CreateFmt(pr, 'field ''%s'' of type ''%s'' enum/bitset type conflict', [fld.mName, fld.mEBSTypeName]);
3484 end;
3485 end;
3486 end;
3487 end;
3489 // setup default values
3490 procedure fixRecordDefaults (rec: TDynRecord);
3491 var
3492 fld: TDynField;
3493 begin
3494 for fld in rec.mFields do if fld.mHasDefault then fld.parseDefaultValue();
3495 end;
3497 begin
3498 hdr := nil;
3499 while true do
3500 begin
3501 if not pr.skipBlanks() then break;
3503 if (pr.tokType = pr.TTId) then
3504 begin
3505 // enum or bitset
3506 if (pr.tokStr = 'enum') or (pr.tokStr = 'bitset') then
3507 begin
3508 eb := TDynEBS.Create(pr);
3509 if (findEBSType(eb.typeName) <> nil) then
3510 begin
3511 eb.Free();
3512 raise TDynParseException.CreateFmt(pr, 'duplicate enum/bitset ''%s''', [eb.typeName]);
3513 end;
3514 eb.mOwner := self;
3515 ebsTypes.append(eb);
3516 //writeln(eb.definition); writeln;
3517 continue;
3518 end;
3520 // triggerdata
3521 if (pr.tokStr = 'TriggerData') then
3522 begin
3523 rec := TDynRecord.Create(pr);
3524 for f := 0 to High(rec.mTrigTypes) do
3525 begin
3526 if (findTrigFor(rec.mTrigTypes[f]) <> nil) then
3527 begin
3528 rec.Free();
3529 raise TDynParseException.CreateFmt(pr, 'duplicate trigdata ''%s''', [rec.mTrigTypes[f]]);
3530 end;
3531 end;
3532 rec.mOwner := self;
3533 trigTypes.append(rec);
3534 //writeln(dr.definition); writeln;
3535 continue;
3536 end;
3537 end;
3539 rec := TDynRecord.Create(pr);
3540 //writeln(dr.definition); writeln;
3541 if (findRecType(rec.typeName) <> nil) then begin rec.Free(); raise TDynParseException.CreateFmt(pr, 'duplicate record ''%s''', [rec.typeName]); end;
3542 if (hdr <> nil) and StrEqu(rec.typeName, hdr.typeName) then begin rec.Free(); raise TDynParseException.CreateFmt(pr, 'duplicate record ''%s''', [rec.typeName]); end;
3543 rec.mOwner := self;
3544 if rec.mHeader then
3545 begin
3546 if (hdr <> nil) then begin rec.Free(); raise TDynParseException.CreateFmt(pr, 'duplicate header record ''%s'' (previous is ''%s'')', [rec.typeName, hdr.typeName]); end;
3547 hdr := rec;
3548 end
3549 else
3550 begin
3551 recTypes.append(rec);
3552 end;
3553 end;
3555 // put header record to top
3556 if (hdr = nil) then raise TDynParseException.Create(pr, 'header definition not found in mapdef');
3557 recTypes.append(nil);
3558 for f := recTypes.count-1 downto 1 do recTypes[f] := recTypes[f-1];
3559 recTypes[0] := hdr;
3561 // setup header links and type links
3562 for rec in recTypes do linkRecord(rec);
3563 for rec in trigTypes do linkRecord(rec);
3565 // setup default values
3566 for rec in recTypes do fixRecordDefaults(rec);
3567 for rec in trigTypes do fixRecordDefaults(rec);
3568 end;
3571 // ////////////////////////////////////////////////////////////////////////// //
3572 function TDynMapDef.parseTextMap (pr: TTextParser): TDynRecord;
3573 var
3574 res: TDynRecord = nil;
3575 begin
3576 result := nil;
3577 try
3578 pr.expectId(headerType.typeName);
3579 res := headerType.clone(nil);
3580 res.mHeaderRec := res;
3581 res.parseValue(pr);
3582 result := res;
3583 res := nil;
3584 finally
3585 res.Free();
3586 end;
3587 end;
3590 function TDynMapDef.parseBinMap (st: TStream): TDynRecord;
3591 var
3592 res: TDynRecord = nil;
3593 begin
3594 result := nil;
3595 try
3596 res := headerType.clone(nil);
3597 res.mHeaderRec := res;
3598 res.parseBinValue(st);
3599 result := res;
3600 res := nil;
3601 finally
3602 res.Free();
3603 end;
3604 end;
3607 // WARNING! stream must be seekable
3608 function TDynMapDef.parseMap (st: TStream; wasBinary: PBoolean=nil): TDynRecord;
3609 var
3610 sign: packed array[0..3] of AnsiChar;
3611 pr: TTextParser;
3612 begin
3613 if (wasBinary <> nil) then wasBinary^ := false;
3614 st.position := 0;
3615 st.ReadBuffer(sign[0], 4);
3616 st.position := 0;
3617 if (sign[0] = 'M') and (sign[1] = 'A') and (sign[2] = 'P') then
3618 begin
3619 if (sign[3] = #1) then
3620 begin
3621 if (wasBinary <> nil) then wasBinary^ := true;
3622 result := parseBinMap(st);
3623 exit;
3624 end;
3625 raise TDynRecException.Create('invalid binary map version');
3626 end
3627 else
3628 begin
3629 pr := TFileTextParser.Create(st, false); // `st` is not owned
3630 try
3631 try
3632 result := parseTextMap(pr);
3633 except on e: Exception do
3634 raise TDynParseException.Create(pr, e.message);
3635 end;
3636 finally
3637 pr.Free();
3638 end;
3639 end;
3640 end;
3643 // returns `true` if the given stream can be a map file
3644 // stream position is 0 on return
3645 // WARNING! stream must be seekable
3646 class function TDynMapDef.canBeMap (st: TStream): Boolean;
3647 var
3648 sign: packed array[0..3] of AnsiChar;
3649 pr: TTextParser;
3650 begin
3651 result := false;
3652 st.position := 0;
3653 st.ReadBuffer(sign[0], 4);
3654 if (sign[0] = 'M') and (sign[1] = 'A') and (sign[2] = 'P') then
3655 begin
3656 result := (sign[3] = #1);
3657 end
3658 else
3659 begin
3660 st.position := 0;
3661 pr := TFileTextParser.Create(st, false); // `st` is not owned
3662 result := (pr.tokType = pr.TTId) and (pr.tokStr = 'map');
3663 pr.Free();
3664 end;
3665 st.position := 0;
3666 end;
3669 function TDynMapDef.pasdefconst (): AnsiString;
3670 var
3671 ebs: TDynEBS;
3672 begin
3673 result := '';
3674 result += '// ////////////////////////////////////////////////////////////////////////// //'#10;
3675 result += '// enums and bitsets'#10;
3676 for ebs in ebsTypes do result += #10+ebs.pasdef();
3677 end;
3680 function TDynMapDef.getRecTypeCount (): Integer; inline; begin result := recTypes.count; end;
3681 function TDynMapDef.getRecTypeAt (idx: Integer): TDynRecord; inline; begin if (idx >= 0) and (idx < recTypes.count) then result := recTypes[idx] else result := nil; end;
3683 function TDynMapDef.getEBSTypeCount (): Integer; inline; begin result := ebsTypes.count; end;
3684 function TDynMapDef.getEBSTypeAt (idx: Integer): TDynEBS; inline; begin if (idx >= 0) and (idx < ebsTypes.count) then result := ebsTypes[idx] else result := nil; end;
3686 function TDynMapDef.getTrigTypeCount (): Integer; inline; begin result := trigTypes.count; end;
3687 function TDynMapDef.getTrigTypeAt (idx: Integer): TDynRecord; inline; begin if (idx >= 0) and (idx < trigTypes.count) then result := trigTypes[idx] else result := nil; end;
3690 end.