| 1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666166716681669167016711672167316741675167616771678167916801681168216831684168516861687168816891690169116921693169416951696169716981699170017011702170317041705170617071708170917101711171217131714171517161717171817191720172117221723172417251726172717281729173017311732173317341735173617371738173917401741174217431744174517461747174817491750175117521753175417551756175717581759176017611762176317641765176617671768176917701771177217731774177517761777177817791780178117821783178417851786178717881789179017911792179317941795179617971798179918001801180218031804180518061807180818091810181118121813181418151816181718181819182018211822182318241825182618271828182918301831183218331834183518361837183818391840184118421843184418451846184718481849185018511852185318541855185618571858185918601861186218631864186518661867186818691870187118721873187418751876187718781879188018811882188318841885188618871888188918901891189218931894189518961897189818991900190119021903190419051906190719081909191019111912191319141915191619171918191919201921192219231924192519261927192819291930193119321933193419351936193719381939194019411942194319441945194619471948194919501951195219531954195519561957195819591960196119621963196419651966196719681969197019711972197319741975197619771978197919801981198219831984198519861987198819891990199119921993199419951996199719981999200020012002200320042005200620072008200920102011201220132014201520162017201820192020202120222023202420252026202720282029203020312032203320342035203620372038203920402041204220432044204520462047204820492050205120522053205420552056205720582059206020612062206320642065206620672068206920702071207220732074207520762077207820792080208120822083208420852086208720882089209020912092209320942095209620972098209921002101210221032104210521062107210821092110211121122113211421152116211721182119212021212122212321242125212621272128212921302131213221332134213521362137213821392140214121422143214421452146214721482149215021512152215321542155215621572158215921602161216221632164216521662167216821692170217121722173217421752176217721782179218021812182218321842185218621872188218921902191219221932194219521962197219821992200220122022203220422052206220722082209221022112212221322142215221622172218221922202221222222232224222522262227222822292230223122322233223422352236223722382239224022412242224322442245224622472248224922502251225222532254225522562257225822592260226122622263226422652266226722682269227022712272227322742275227622772278227922802281228222832284228522862287228822892290229122922293229422952296229722982299230023012302230323042305230623072308230923102311231223132314231523162317231823192320232123222323232423252326232723282329233023312332233323342335233623372338233923402341234223432344234523462347234823492350235123522353235423552356235723582359236023612362236323642365236623672368236923702371237223732374237523762377237823792380238123822383238423852386238723882389239023912392239323942395239623972398239924002401240224032404240524062407240824092410241124122413241424152416241724182419242024212422242324242425242624272428242924302431243224332434243524362437243824392440244124422443244424452446244724482449245024512452245324542455245624572458245924602461246224632464246524662467246824692470247124722473247424752476247724782479248024812482248324842485248624872488248924902491249224932494249524962497249824992500250125022503250425052506250725082509251025112512251325142515251625172518251925202521252225232524252525262527252825292530253125322533253425352536253725382539254025412542254325442545254625472548254925502551255225532554255525562557255825592560256125622563256425652566256725682569257025712572257325742575257625772578257925802581258225832584258525862587258825892590259125922593259425952596259725982599260026012602260326042605260626072608260926102611261226132614261526162617261826192620262126222623262426252626262726282629263026312632263326342635263626372638263926402641264226432644264526462647264826492650265126522653265426552656265726582659266026612662266326642665266626672668266926702671267226732674267526762677267826792680268126822683268426852686268726882689269026912692269326942695269626972698269927002701270227032704270527062707270827092710271127122713271427152716271727182719272027212722272327242725272627272728272927302731273227332734273527362737273827392740274127422743274427452746274727482749275027512752275327542755275627572758275927602761276227632764276527662767276827692770277127722773277427752776277727782779278027812782278327842785278627872788278927902791279227932794279527962797279827992800280128022803280428052806280728082809281028112812281328142815281628172818281928202821282228232824282528262827282828292830283128322833283428352836283728382839284028412842284328442845284628472848284928502851285228532854285528562857285828592860286128622863286428652866286728682869287028712872287328742875287628772878287928802881288228832884288528862887288828892890289128922893289428952896289728982899290029012902290329042905290629072908290929102911291229132914291529162917291829192920292129222923292429252926292729282929293029312932293329342935293629372938293929402941294229432944294529462947294829492950295129522953295429552956295729582959296029612962296329642965296629672968296929702971297229732974297529762977297829792980298129822983298429852986298729882989299029912992299329942995299629972998299930003001300230033004300530063007300830093010301130123013301430153016301730183019302030213022302330243025302630273028302930303031303230333034303530363037303830393040304130423043304430453046304730483049305030513052305330543055305630573058305930603061306230633064306530663067306830693070307130723073307430753076307730783079308030813082308330843085308630873088308930903091309230933094309530963097309830993100310131023103310431053106310731083109 | {    pas2jni - JNI bridge generator for Pascal.    Copyright (c) 2013 by Yury Sidorov.    This program is free software; you can redistribute it and/or modify    it under the terms of the GNU General Public License as published by    the Free Software Foundation; either version 2 of the License, or    (at your option) any later version.    This program is distributed in the hope that it will be useful,    but WITHOUT ANY WARRANTY; without even the implied warranty of    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the    GNU General Public License for more details.    You should have received a copy of the GNU General Public License    along with this program; if not, write to the Free Software    Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. ****************************************************************************}unit writer;{$mode objfpc}{$H+}interface//{$define DEBUG}{$ifdef DEBUG}{$ASSERTIONS ON}{$endif}uses  Classes, SysUtils, def, contnrs, PPUParser;const  MaxMethodPointers = 10000;type  { TTextOutStream }  TTextOutStream = class(TFileStream)  private    FIndent: integer;    FIndStr: string;    procedure SetIndednt(const AValue: integer);  public    procedure Write(const s: ansistring); overload;    procedure WriteLn(const s: ansistring = ''; ExtraIndent: integer = 0);    procedure IncI;    procedure DecI;    property Indent: integer read FIndent write SetIndednt;    property SIndent: string read FIndStr;  end;  { TClassInfo }  TClassInfo = class  public    Def: TDef;    Funcs: TObjectList;    IsCommonClass: boolean;    constructor Create;    destructor Destroy; override;  end;  { TProcInfo }  TProcInfo = class  public    Name: string;    JniName: string;    JniSignature: string;  end;  { TClassList }  TClassList = class(TStringList)  private    function GetFullName(const AName: string; Def: TDef): string;  public    constructor Create;    function Add(const AName: string; Def: TDef; Info: TClassInfo): integer;    function IndexOf(const AName: string; Def: TDef): integer; reintroduce;    function GetClassName(Index: integer): string;    function GetClassInfo(Index: integer): TClassInfo;  end;  { TWriter }  TWriter = class  private    Fjs, Fps: TTextOutStream;    FClasses: TClassList;    FPkgDir: string;    FUniqueCnt: integer;    FThisUnit: TUnitDef;    FIntegerType: TDef;    FRecords: TObjectList;    FRealClasses: TObjectList;    function DoCheckItem(const ItemName: string): TCheckItemResult;    procedure WriteClassTable;    procedure WriteFileComment(st: TTextOutStream);    function FindInStringList(list: TStringList; const s: string): integer;    procedure ProcessRules(d: TDef; const Prefix: string = '');    function GetUniqueNum: integer;    function DefToJniType(d: TDef; var err: boolean): string;    function DefToJniSig(d: TDef): string;    function DefToJavaType(d: TDef): string;    function GetJavaClassPath(d: TDef; const AClassName: string = ''): string;    function JniToPasType(d: TDef; const v: string; CheckNil: boolean): string;    function PasToJniType(d: TDef; const v: string): string;    function GetTypeInfoVar(ClassDef: TDef): string;    function GetClassPrefix(ClassDef: TDef; const AClassName: string = ''): string;    function IsJavaSimpleType(d: TDef): boolean;    function IsJavaVarParam(ParamDef: TVarDef): boolean;    function GetProcDeclaration(d: TProcDef; const ProcName: string = ''; FullTypeNames: boolean = False; InternalParaNames: boolean = False): string;    function GetJavaProcDeclaration(d: TProcDef; const ProcName: string = ''): string;    function GetJniFuncType(d: TDef): string;    function GetJavaClassName(cls: TDef; it: TDef): string;    procedure RegisterPseudoClass(d: TDef);    function GetPasIntType(Size: integer): string;    function GetPasType(d: TDef; FullName: boolean): string;//    procedure AddCustomProc(ParentDef: TDef; const JniName, Name: string; RetType: TBasicType; const Params: array of TBasicType);    function AddCustomProc(ParentDef: TDef; const JniName, Name: string; RetType: TBasicType; const Params: array of TBasicType): TProcDef;    procedure AddNativeMethod(ParentDef: TDef; const JniName, Name, Signature: string);    function GetProcSignature(d: TProcDef): string;    procedure EHandlerStart;    procedure EHandlerEnd(const EnvVarName: string; const ExtraCode: string = '');    procedure UpdateUsedUnits(u: TUnitDef);    procedure WriteClassInfoVar(d: TDef);    procedure WriteComment(d: TDef; const AType: string);    procedure WriteClass(d: TClassDef; PreInfo: boolean);    procedure WriteProc(d: TProcDef; Variable: TVarDef = nil; AParent: TDef = nil);    procedure WriteVar(d: TVarDef; AParent: TDef = nil);    procedure WriteConst(d: TConstDef);    procedure WriteEnum(d: TDef);    procedure WriteProcType(d: TProcDef; PreInfo: boolean);    procedure WriteSet(d: TSetDef);    procedure WritePointer(d: TPointerDef; PreInfo: boolean);    procedure WriteClassRef(d: TClassRefDef; PreInfo: boolean);    procedure WriteUnit(u: TUnitDef);    procedure WriteOnLoad;    procedure WriteRecordSizes;  public    SearchPath: string;    LibName: string;    JavaPackage: string;    Units: TStringList;    OutPath: string;    JavaOutPath: string;    IncludeList: TStringList;    ExcludeList: TStringList;    LibAutoLoad: boolean;    constructor Create;    destructor Destroy; override;    procedure ProcessUnits;  end;implementationconst  JNIType: array[TBasicType] of string =    ('', 'jshort', 'jbyte', 'jint', 'jshort', 'jlong', 'jint', 'jlong', 'jfloat', 'jdouble', 'jstring',     'jstring', 'jboolean', 'jchar', 'jchar', 'jint', 'jstring');  JNITypeSig: array[TBasicType] of string =    ('V', 'S', 'B', 'I', 'S', 'J', 'I', 'J', 'F', 'D', 'Ljava/lang/String;', 'Ljava/lang/String;',     'Z', 'C', 'C', 'I', 'Ljava/lang/String;');  JavaType: array[TBasicType] of string =    ('void', 'short', 'byte', 'int', 'short', 'long', 'int', 'long', 'float', 'double', 'String',     'String', 'boolean', 'char', 'char', 'int', 'String');  TextIndent = 2;  ExcludeStd: array[1..44] of string = (    'classes.TStream.ReadComponent', 'classes.TStream.ReadComponentRes', 'classes.TStream.WriteComponent', 'classes.TStream.WriteComponentRes',    'classes.TStream.WriteDescendent', 'classes.TStream.WriteDescendentRes', 'classes.TStream.WriteResourceHeader', 'classes.TStream.FixupResourceHeader',    'classes.TStream.ReadResHeader', 'classes.TComponent.WriteState', 'classes.TComponent.ExecuteAction', 'classes.TComponent.UpdateAction',    'classes.TComponent.GetEnumerator', 'classes.TComponent.VCLComObject', 'classes.TComponent.DesignInfo', 'classes.TComponent.Destroying',    'classes.TComponent.FreeNotification', 'classes.TComponent.RemoveFreeNotification', 'classes.TComponent.FreeOnRelease', 'classes.TComponent.SetSubComponent',    'system.TObject.newinstance', 'system.TObject.FreeInstance', 'system.TObject.SafeCallException', 'system.TObject.InitInstance',    'system.TObject.CleanupInstance', 'system.TObject.ClassInfo', 'system.TObject.AfterConstruction', 'system.TObject.BeforeDestruction',    'system.TObject.GetInterfaceEntry', 'system.TObject.GetInterfaceTable', 'system.TObject.MethodAddress', 'system.TObject.MethodName',    'system.TObject.FieldAddress', 'classes.TComponent.ComponentState', 'classes.TComponent.ComponentStyle', 'classes.TList.GetEnumerator',    'classes.TList.List', 'classes.TList.FPOAttachObserver', 'classes.TList.FPODetachObserver', 'classes.TList.FPONotifyObservers',    'classes.TPersistent.FPOAttachObserver', 'classes.TPersistent.FPODetachObserver', 'classes.TPersistent.FPONotifyObservers',    'system.fma'  );  ExcludeDelphi7: array[1..26] of string = (    'system.TObject.StringMessageTable', 'system.TObject.GetInterfaceEntryByStr', 'system.TObject.UnitName', 'system.TObject.Equals',    'system.TObject.GetHashCode', 'system.TObject.ToString','system.TObject.QualifiedClassName','classes.TStream.ReadByte', 'classes.TStream.ReadWord',    'classes.TStream.ReadDWord', 'classes.TStream.ReadQWord', 'classes.TStream.ReadAnsiString', 'classes.TStream.WriteByte',    'classes.TStream.WriteWord', 'classes.TStream.WriteDWord', 'classes.TStream.WriteQWord', 'classes.TStream.WriteAnsiString',    'classes.TCollection.Exchange', 'classes.TStrings.Equals', 'classes.TStrings.GetNameValue', 'classes.TStrings.ExtractName',    'classes.TStrings.TextLineBreakStyle', 'classes.TStrings.StrictDelimiter', 'classes.TStrings.GetEnumerator', 'classes.TStringList.OwnsObjects',    'classes.TList.AddList'  );  SUnsupportedType = '<unsupported type>';function JniCaliing: string;begin  Result:='{$ifdef mswindows} stdcall {$else} cdecl {$endif};';end;{ TClassList }function TClassList.IndexOf(const AName: string; Def: TDef): integer;begin  Result:=inherited IndexOf(GetFullName(AName, Def));end;function TClassList.GetClassName(Index: integer): string;var  i: integer;begin  Result:=Strings[Index];  i:=Pos('.', Result);  if i > 0 then    System.Delete(Result, 1, i);end;function TClassList.GetClassInfo(Index: integer): TClassInfo;begin  Result:=TClassInfo(Objects[Index]);end;function TClassList.GetFullName(const AName: string; Def: TDef): string;begin  if (Def = nil) or (Def.DefType = dtUnit) then    Result:=AName  else begin    while (Def.Parent <> nil) and (Def.DefType <> dtUnit) do      Def:=Def.Parent;    Result:=Def.Name + '.' + AName;  end;end;constructor TClassList.Create;begin  inherited Create;  Sorted:=True;end;function TClassList.Add(const AName: string; Def: TDef; Info: TClassInfo): integer;begin  Result:=AddObject(GetFullName(AName, Def), Info);end;{ TTextOutStream }procedure TTextOutStream.SetIndednt(const AValue: integer);begin  if FIndent = AValue then exit;  FIndent:=AValue;  SetLength(FIndStr, FIndent*TextIndent);  if FIndent > 0 then    FillChar(FIndStr[1], FIndent*TextIndent, ' ');end;procedure TTextOutStream.Write(const s: ansistring);begin  WriteBuffer(PChar(s)^, Length(s));end;procedure TTextOutStream.WriteLn(const s: ansistring; ExtraIndent: integer);begin  if s = '' then    Write(LineEnding)  else begin    Indent:=Indent + ExtraIndent;    try      Write(FIndStr + s + LineEnding);    finally      Indent:=Indent - ExtraIndent;    end;  end;end;procedure TTextOutStream.IncI;begin  Indent:=Indent + 1;end;procedure TTextOutStream.DecI;begin  if Indent > 0  then    Indent:=Indent - 1;end;{ TClassInfo }constructor TClassInfo.Create;begin  Funcs:=TObjectList.Create(True);end;destructor TClassInfo.Destroy;begin  Funcs.Free;  inherited Destroy;end;{ TWriter }function TWriter.DefToJniType(d: TDef; var err: boolean): string;begin  if d = nil then begin    Result:=SUnsupportedType;    err:=True;  end  else begin    if not d.IsUsed then begin      Result:='<excluded type> ' + d.Name;      err:=True;    end    else      case d.DefType of        dtType:          Result:=JNIType[TTypeDef(d).BasicType];        dtClass, dtEnum, dtClassRef:          Result:='jobject';        dtProcType:          if poMethodPtr in TProcDef(d).ProcOpt then            Result:='jobject'          else begin            Result:=SUnsupportedType + ' ' + d.Name;            err:=True;          end;        dtSet:          if TSetDef(d).Size <= 4 then            Result:='jobject'          else begin            Result:=SUnsupportedType + ' ' + d.Name;            err:=True;          end;        dtPointer:          if TPointerDef(d).IsObjPtr then            Result:='jobject'          else            Result:='jlong';        dtJniObject:          Result:='jobject';        else begin          Result:=SUnsupportedType + ' ' + d.Name;          err:=True;          d.SetNotUsed;        end;    end;  end;end;function TWriter.DoCheckItem(const ItemName: string): TCheckItemResult;begin  if FindInStringList(ExcludeList, ItemName) >= 0 then    Result:=crExclude  else    if FindInStringList(IncludeList, ItemName) >= 0 then      Result:=crInclude    else      Result:=crDefault;end;procedure TWriter.WriteFileComment(st: TTextOutStream);begin  st.WriteLn('// This file was automatically generated by the pas2jni utility.');  st.WriteLn('// Do not edit this file.');end;function TWriter.FindInStringList(list: TStringList; const s: string): integer;var  len, cnt: integer;  ss: string;begin  if list.Find(s, Result) or (Result < 0) then    exit;  if Result < list.Count then begin    cnt:=3;    if Result > 0 then      Dec(Result)    else      Dec(cnt);    if Result + cnt > list.Count then      Dec(cnt);    while cnt > 0 do begin      ss:=list[Result];      len:=Length(ss);      if (len > 1) and (ss[len] = '*') then begin        Dec(len);        if AnsiCompareText(Copy(s, 1, len), Copy(ss, 1, len)) = 0 then          exit;      end;      Inc(Result);      Dec(cnt);    end;  end;  Result:=-1;end;procedure TWriter.ProcessRules(d: TDef; const Prefix: string);var  i: integer;  s: string;begin  if d.DefType = dtClass then    with TClassDef(d) do      if (AncestorClass = nil) and (CType in [ctClass, ctInterface]) and (CompareText(Parent.Name, 'system') <> 0) then begin        SetNotUsed;        exit;      end;  s:=Prefix + d.Name;  if FindInStringList(ExcludeList, s) >= 0 then    d.SetNotUsed  else    if FindInStringList(IncludeList, s) >= 0 then      d.IsUsed:=True;  if not (d.DefType in [dtUnit, dtClass]) then    exit;  s:=s + '.';  for i:=0 to d.Count - 1 do    ProcessRules(d[i], s);end;function TWriter.GetUniqueNum: integer;begin  Inc(FUniqueCnt);  Result:=FUniqueCnt;end;function TWriter.DefToJniSig(d: TDef): string;begin  if d = nil then    Result:=SUnsupportedType  else    case d.DefType of      dtType:        Result:=JNITypeSig[TTypeDef(d).BasicType];      dtClass, dtProcType, dtSet, dtEnum, dtClassRef:        Result:='L' + GetJavaClassPath(d) + ';';      dtPointer:        if TPointerDef(d).IsObjPtr then          Result:='L' + GetJavaClassPath(d) + ';'        else          Result:='J';      dtJniObject:        Result:='Ljava/lang/Object;';      else        Result:=SUnsupportedType;    end;end;function TWriter.DefToJavaType(d: TDef): string;begin  if d = nil then    Result:=SUnsupportedType  else    if not d.IsUsed and (d.DefType <> dtType) then      Result:='<excluded type> ' + d.Name    else      case d.DefType of        dtType:          Result:=JavaType[TTypeDef(d).BasicType];        dtClass, dtProcType, dtSet, dtEnum, dtClassRef:          Result:=d.Name;        dtPointer:          if TPointerDef(d).IsObjPtr then            Result:=d.Name          else            Result:='long';        dtJniObject:          Result:='Object';        else          Result:=SUnsupportedType;      end;end;function TWriter.GetJavaClassPath(d: TDef; const AClassName: string): string;var  n: string;begin  if AClassName = '' then    n:=d.AliasName  else    n:=AClassName;  Result:=StringReplace(JavaPackage, '.', '/', [rfReplaceAll]);  if Result <> '' then    Result:=Result + '/';  if d.DefType = dtUnit then    Result:=Result + n  else    Result:=Result + d.Parent.AliasName + '$' + n;end;procedure TWriter.WriteClass(d: TClassDef; PreInfo: boolean);var  WrittenItems: TList;  procedure _WriteConstructors(c: TClassDef; Written: TStringList);  var    i, j: integer;    p: TProcDef;    OldRet: TDef;    s: string;  begin    if c = nil then      exit;    for i:=0 to c.Count - 1 do      with c[i] do begin        if (DefType = dtProc) and not c.IsPrivate and (TProcDef(c[i]).ProcType = ptConstructor) then begin          p:=TProcDef(c[i]);          j:=Written.IndexOf(p.Name);          if (j < 0) or (Written.Objects[j] = c) then begin            s:=p.Name + ':';            for j:=0 to p.Count - 1 do              if p[j].DefType = dtParam then                s:=s + DefToJniSig(TVarDef(p[j]).VarType);            if Written.IndexOf(s) < 0 then begin              OldRet:=p.ReturnType;              p.ReturnType:=d;              p.Parent:=d;              try                WriteProc(p);              finally                p.ReturnType:=OldRet;                p.Parent:=c;              end;              Written.Add(s);              if not (poOverload in p.ProcOpt) then                Written.AddObject(p.Name, c);            end;          end;        end;      end;    _WriteConstructors(c.AncestorClass, Written);  end;  procedure WriteConstructors;  var    cc: TStringList;    i: integer;  begin    if not TClassDef(d).HasAbstractMethods then begin      // Writing all constructors including parent's      cc:=TStringList.Create;      try        cc.Sorted:=True;        _WriteConstructors(TClassDef(d), cc);      finally        cc.Free;      end;    end;    if d.CType = ctClass then begin      i:=FRealClasses.Add(d);      Fjs.WriteLn(Format('public static %s Class() { return new %0:s(system.GetClassRef(%d)); }', [d.AliasName, i]));      Fjs.WriteLn(Format('public static system.TClass TClass() { return system.GetTClass(%d); }', [i]));    end;  end;  procedure _WriteReplacedItems(c: TClassDef);  var    i: integer;    p: TReplDef;  begin    c:=c.AncestorClass;    if c = nil then      exit;    if c.HasReplacedItems then begin      for i:=0 to c.Count - 1 do        with c[i] do begin          p:=TReplDef(c[i]);          if (DefType in ReplDefs) and ((p.IsReplaced) or p.IsReplImpl) then begin            if p.ReplacedItem <> nil then              WrittenItems.Add(p.ReplacedItem);            if WrittenItems.IndexOf(p) >= 0 then              continue;            case p.DefType of              dtProc:                WriteProc(TProcDef(p), nil, d);              dtProp, dtField:                WriteVar(TVarDef(p), d);            end;          end;        end;    end;    _WriteReplacedItems(c);  end;  procedure WriteReplacedItems;  begin    _WriteReplacedItems(TClassDef(d));  end;  procedure WriteItems(Regular, Replaced, ReplImpl: boolean);  var    i: integer;    it: TReplDef;  begin    for i:=0 to d.Count - 1 do begin      it:=TReplDef(d[i]);      if not (it.DefType in ReplDefs) then        continue;      if not (it.IsReplImpl or it.IsReplaced) then begin        if not Regular then          continue;      end      else        if (not Replaced and it.IsReplaced) or (not ReplImpl and it.IsReplImpl) then          continue;      if it.ReplacedItem <> nil then        WrittenItems.Add(it.ReplacedItem);      case it.DefType of        dtProc:          if TProcDef(it).ProcType <> ptConstructor  then            WriteProc(TProcDef(it));        dtProp, dtField:          WriteVar(TVarDef(it));      end;    end;  end;  procedure WriteTypeCast(const AName: string; SecondPass: boolean);  var    s, ss: string;  begin    with TClassDef(d) do begin      if HasReplacedItems and not SecondPass then        s:='protected'      else        s:='public';      if CType = ctInterface then begin        Fjs.WriteLn('private native long __AsIntf(long objptr);');        ss:=IID;        if ss = '' then          ss:='null'        else          ss:='"' + ss + '"';        Fjs.WriteLn(Format('%s %s(PascalObject obj) { super(0, true); __TypeCast(obj, %s); }', [s, AName, ss]));        Fjs.WriteLn(Format('%s %s(long objptr) { super(objptr, true); }', [s, AName]));      end      else begin        Fjs.WriteLn(Format('%s %s(PascalObject obj) { super(obj); }', [s, AName]));        Fjs.WriteLn(Format('%s %s(long objptr) { super(objptr); }', [s, AName]));      end;    end;  end;var  s, ss, n: string;  RegularClass: boolean;begin  if PreInfo then begin    WriteClassInfoVar(d);    if d.CType in [ctObject, ctRecord] then begin      s:=d.Parent.Name + '.' + d.Name;      Fps.WriteLn;      Fps.WriteLn(Format('function _%s_CreateObj(env: PJNIEnv; const r: %s): jobject;', [GetClassPrefix(d), s]));      Fps.WriteLn(Format('var pr: ^%s;', [s]));      Fps.WriteLn('begin');      Fps.IncI;      Fps.WriteLn(Format('pr:=AllocMem(SizeOf(%s)); pr^:=r;', [s]));      Fps.WriteLn(Format('Result:=_CreateJavaObj(env, pr, %s);', [GetTypeInfoVar(d)]));      Fps.DecI;      Fps.WriteLn('end;');      Fps.WriteLn;      ss:=Format('_%s_Free', [GetClassPrefix(d)]);      Fps.WriteLn(Format('procedure %s(env: PJNIEnv; _self: JObject; r: jlong);', [ss]) + JniCaliing);      Fps.WriteLn(Format('var pr: ^%s;', [s]));      Fps.WriteLn('begin');      Fps.WriteLn('pr:=pointer(ptruint(r));', 1);      Fps.WriteLn('system.Dispose(pr);', 1);      Fps.WriteLn('end;');      AddNativeMethod(d, ss, '__Destroy', '(J)V');    end;    exit;  end;  // Java  case d.CType of    ctInterface:      s:='interface';    ctObject:      s:='interface';    ctRecord:      s:='record';    else      s:='class';  end;  WriteComment(d, s);  n:=GetJavaClassName(d, nil);  s:='public static class ' + n + ' extends ';  with d do begin    if AncestorClass <> nil then begin      ss:=AncestorClass.Name;      if ImplementsReplacedItems then        ss:='__' + ss;      s:=s + ss;    end    else      if d.CType in [ctObject, ctRecord] then        s:=s + Format('%s.system.Record', [JavaPackage])      else        if d.CType = ctInterface then          s:=s + 'PascalInterface'        else          s:=s + 'PascalObject';  end;  Fjs.WriteLn(s + ' {');  Fjs.IncI;  case d.CType of    ctObject, ctRecord:      begin        Fjs.WriteLn('private native void __Destroy(long pasobj);');        if d.AncestorClass = nil then          s:='__Init'        else          s:='super';        Fjs.WriteLn(Format('protected %s(long objptr, boolean cleanup) { %s(objptr, cleanup); }', [d.Name, s]));        Fjs.WriteLn(Format('public %s() { %s(0, true); }', [d.Name, s]));        Fjs.WriteLn(Format('@Override public void __Release() { __Destroy(_pasobj); _pasobj=0; }', [d.Name]));        Fjs.WriteLn(Format('@Override public int __Size() { return __Size(%d); }', [FRecords.Add(d)]));      end;    ctInterface:      begin        if d.AncestorClass = nil then begin          Fjs.WriteLn('@Override public void __Release() { if (_pasobj != 0) _Release(); _pasobj = 0; }');          Fjs.WriteLn('@Override protected void __Init() { _cleanup=true; if (_pasobj != 0) _AddRef(); }');        end;        Fjs.WriteLn(Format('protected %s(long objptr, boolean cleanup) { super(objptr, cleanup); }', [d.Name]));      end;  end;  WriteTypeCast(n, False);  WrittenItems:=TList.Create;  try    RegularClass:=(d.DefType = dtClass) and not TClassDef(d).HasReplacedItems;    if RegularClass then      WriteConstructors;    // Write regular items    WriteItems(True, False, RegularClass);    if RegularClass and TClassDef(d).ImplementsReplacedItems then      // Write implementation wrappers for replaced mehods      WriteReplacedItems;    Fjs.DecI;    Fjs.WriteLn('}');    Fjs.WriteLn;    if (d.DefType = dtClass) and (TClassDef(d).HasReplacedItems) then begin      // Write replaced items      Fjs.WriteLn(Format('public static class %s extends __%0:s {', [d.AliasName]));      Fjs.IncI;      WriteTypeCast(d.AliasName, True);      WriteConstructors;      WriteItems(False, True, True);      if TClassDef(d).ImplementsReplacedItems then        // Write implementation wrappers for replaced mehods        WriteReplacedItems;      Fjs.DecI;      Fjs.WriteLn('}');      Fjs.WriteLn;    end;  finally    WrittenItems.Free;  end;end;procedure TWriter.WriteProc(d: TProcDef; Variable: TVarDef; AParent: TDef);var  i, j, ClassIdx: integer;  s, ss, ps, TempRes, VarFin: string;  err, tf: boolean;  pi: TProcInfo;  ci: TClassInfo;  IsTObject: boolean;  tempvars: TStringList;  vd: TVarDef;  UseTempObjVar, IsObj, IsProcVar: boolean;  ItemDef: TDef;begin  ASSERT(d.DefType = dtProc);  if d.IsPrivate or not d.IsUsed then    exit;  IsTObject:=(d.Parent.DefType = dtClass) and (TClassDef(d.Parent).AncestorClass = nil);  if (d.ProcType = ptDestructor) and not IsTObject then    exit;  if Variable <> nil then    ItemDef:=Variable  else    ItemDef:=d;  tempvars:=nil;  pi:=TProcInfo.Create;  with d do  try    IsObj:=(d.Parent.DefType = dtClass) and (TClassDef(d.Parent).CType = ctObject);    if not IsObj and (poClassMethod in ProcOpt) and (Name = 'ClassType') then      ProcOpt:=ProcOpt - [poClassMethod];    pi.Name:=Name;    s:=GetClassPrefix(d.Parent) + pi.Name;    pi.JniName:=s;    pi.JniSignature:=GetProcSignature(d);    if AParent = nil then begin      // Checking duplicate proc name and duplicate param types      ClassIdx:=FClasses.IndexOf(GetJavaClassName(d.Parent, ItemDef), d.Parent);      if ClassIdx >= 0 then begin        ci:=FClasses.GetClassInfo(ClassIdx);        j:=1;        ss:=Copy(pi.JniSignature, 1, Pos(')', pi.JniSignature));        repeat          err:=False;          for i:=0 to ci.Funcs.Count - 1 do            with TProcInfo(ci.Funcs[i]) do              if CompareText(JniName, pi.JniName) = 0 then begin                Inc(j);                pi.JniName:=Format('%s_%d', [s, j]);                err:=True;                break;              end              else                if (CompareText(Name, pi.Name) = 0) and (ss = Copy(JniSignature, 1, Pos(')', JniSignature))) then                  // Duplicate params                  exit;        until not err;      end;      err:=False;      if ProcType in [ptFunction, ptConstructor] then        s:='function'      else        s:='procedure';      s:=s + ' ' + pi.JniName + '(_env: PJNIEnv; _jobj: jobject';      if IsObj and (ProcType in [ptConstructor, ptDestructor]) then        TempRes:='__tempres';      IsProcVar:=(Variable <> nil) and (Variable.VarType <> nil) and (Variable.VarType.DefType = dtProcType);      UseTempObjVar:=IsProcVar and (ProcType = ptProcedure) and (Variable.Parent.DefType <> dtUnit);      for j:=0 to Count - 1 do begin        vd:=TVarDef(Items[j]);        if vd.DefType <> dtParam then          continue;        with vd do begin          if (VarType <> nil) and (VarType.DefType = dtJniEnv) then            continue;          s:=s + '; ' + Name + ': ';          if not IsJavaVarParam(vd) then            s:=s + DefToJniType(VarType, err)          else begin            s:=s + 'jarray';            if tempvars = nil then              tempvars:=TStringList.Create;            if VarType = nil then              err:=True            else              Tag:=tempvars.AddObject('__tmp_' + Name, d.Items[j]) + 1;          end;        end;      end;      s:=s + ')';      if ProcType in [ptFunction, ptConstructor] then        s:=s + ': ' + DefToJniType(ReturnType, err);      s:=s + '; ' + JniCaliing;      if err then begin        s:='// ' + s;        Fjs.WriteLn('// NOT PROCESSED: ' + GetJavaProcDeclaration(d));        d.SetNotUsed;      end;      Fps.WriteLn;      Fps.WriteLn(s);      if err then        exit;      if (poClassMethod in ProcOpt) and not IsObj then begin        Fps.WriteLn(Format('type _classt = %s;', [Parent.Parent.Name + '.' + Parent.Name]));        Fps.WriteLn('type _class = class of _classt;');      end;      if (tempvars <> nil) or UseTempObjVar or (TempRes <> '') then begin        s:='';        Fps.WriteLn('var');        Fps.IncI;        if tempvars <> nil then begin          for i:=0 to tempvars.Count - 1 do begin            vd:=TVarDef(tempvars.Objects[i]);            Fps.WriteLn(Format('%s: %s;', [tempvars[i], GetPasType(vd.VarType, True)]));            if IsJavaSimpleType(vd.VarType) then begin              Fps.WriteLn(Format('%s_arr: P%s;', [tempvars[i], DefToJniType(vd.VarType, err)]));              if s = '' then                s:='__iscopy: JBoolean;';            end;          end;          if s <> '' then            Fps.WriteLn(s);        end;        if UseTempObjVar then          Fps.WriteLn('__objvar: ' + d.Parent.Name + ';');        if TempRes <> '' then begin          s:=TempRes + ': ';          if IsObj and (ProcType in [ptConstructor, ptDestructor]) then            s:=s + '^' + GetPasType(d.Parent, True)          else            s:=s + GetPasType(d.ReturnType, True);          Fps.WriteLn(s + ';');        end;        Fps.DecI;      end;      if IsProcVar and (ProcType = ptProcedure) then        Fps.WriteLn('var __mvar: TMethod;');      Fps.WriteLn('begin');      Fps.IncI;      EHandlerStart;      tf:=False;      // Assign var parameter values to local vars      if tempvars <> nil then begin        for i:=0 to tempvars.Count - 1 do begin          vd:=TVarDef(tempvars.Objects[i]);          Fps.WriteLn(Format('if _env^^.GetArrayLength(_env, %s) <> 1 then _RaiseVarParamException(''%s'');', [vd.Name, vd.Name]));          if IsJavaSimpleType(vd.VarType) then begin            Fps.WriteLn(Format('%s_arr:=_env^^.Get%sArrayElements(_env, %s, __iscopy);', [tempvars[i], GetJniFuncType(vd.VarType), vd.Name]));            Fps.WriteLn(Format('if %s_arr = nil then _RaiseVarParamException(''%s'');', [tempvars[i], vd.Name]));            s:=tempvars[i] + '_arr^';            tf:=True;          end          else            s:=Format('_env^^.GetObjectArrayElement(_env, %s, 0)', [vd.Name]);          if voVar in vd.VarOpt then            Fps.WriteLn(tempvars[i] + ':=' + JniToPasType(vd.VarType, s, False) + ';');        end;      end;      if tf then begin        Fps.WriteLn('try');        Fps.IncI;      end;      s:='';      if not (IsObj and (ProcType in [ptConstructor, ptDestructor])) then        if Parent.DefType = dtUnit then          s:=Parent.Name + '.'        else          if ProcType = ptConstructor then            s:=Parent.Parent.Name + '.' + Parent.Name + '.'          else            if (poClassMethod in ProcOpt) and not IsObj then              s:='_class(_GetClass(_env, _jobj, ' + GetTypeInfoVar(d.Parent) + '))' + '.'            else              s:=JniToPasType(d.Parent, '_jobj', True) + '.';      if Variable = nil then begin        // Regular proc        s:=s + pi.Name;        if Count > 0 then begin          s:=s + '(';          ps:='';          for j:=0 to Count - 1 do begin            vd:=TVarDef(Items[j]);            if vd.DefType <> dtParam then              continue;            if vd.VarType.DefType = dtJniEnv then              ss:='_env'            else              if vd.Tag <> 0 then                ss:=tempvars[vd.Tag - 1]              else begin                ss:=Items[j].Name;                ss:=JniToPasType(vd.VarType, ss, False);              end;            if ps <> '' then              ps:=ps + ', ';            ps:=ps + ss;          end;          s:=s + ps + ')';        end;      end      else begin        // Var access        if UseTempObjVar then begin          System.Delete(s, Length(s), 1);          Fps.WriteLn('__objvar:=' + s + ';');          s:='__objvar.';        end;        s:=s + Variable.Name;        j:=Count;        if ProcType = ptProcedure then          Dec(j);        if j > 0 then begin          i:=j;          ss:='';          for j:=0 to j - 1 do begin            if ss <> '' then              ss:=ss + ', ';            ss:=ss + JniToPasType(TVarDef(Items[j]).VarType, Items[j].Name, False);          end;          s:=Format('%s[%s]', [s, ss]);        end        else          i:=0;        if ProcType = ptProcedure then begin          ASSERT(Count = i + 1);          if Variable.VarType.DefType = dtProcType then begin            Fps.WriteLn(Format('__mvar:=TMethod(%s);', [s]));            VarFin:=Format('_RefMethodPtr(_env, TMethod(%s), True); _RefMethodPtr(_env, __mvar, False);', [s]);          end;          s:=s + ':=' + JniToPasType(TVarDef(Items[i]).VarType, Items[i].Name, False);        end;      end;      if IsObj and (ProcType = ptConstructor) then begin        s:=Format('system.New(%s, %s);', [TempRes, s]);        Fps.WriteLn(s);        s:=Format('Result:=_CreateJavaObj(_env, %s, %s, False);', [TempRes, GetTypeInfoVar(ReturnType)]);        Fps.WriteLn(s);      end      else      if IsObj and (ProcType = ptDestructor) then begin        Fps.WriteLn(TempRes + ':=@' + JniToPasType(d.Parent, '_jobj', True) + ';');        Fps.WriteLn(Format('%s^.%s;', [TempRes, s]));        Fps.WriteLn(Format('_env^^.SetLongField(_env, _jobj, %s.ObjFieldId, -jlong(ptruint(%s)));', [GetTypeInfoVar(d.Parent), TempRes]));      end      else begin        if ProcType in [ptFunction, ptConstructor] then          s:='Result:=' + PasToJniType(ReturnType, s);        s:=s + ';';        Fps.WriteLn(s);      end;      if VarFin <> '' then        Fps.WriteLn(VarFin);      // Return var/out parameters      if tempvars <> nil then begin        for i:=0 to tempvars.Count - 1 do begin          vd:=TVarDef(tempvars.Objects[i]);          if IsJavaSimpleType(vd.VarType) then            Fps.WriteLn(Format('%s_arr^:=%s;', [tempvars[i], PasToJniType(vd.VarType, tempvars[i])]))          else            Fps.WriteLn(Format('_env^^.SetObjectArrayElement(_env, %s, 0, %s);', [vd.Name, PasToJniType(vd.VarType, tempvars[i])]));        end;      end;      if not IsObj then        if IsTObject and ( (ProcType = ptDestructor) or (CompareText(Name, 'Free') = 0) ) then          Fps.WriteLn(Format('_env^^.SetLongField(_env, _jobj, %s.ObjFieldId, 0);', [GetTypeInfoVar(d.Parent)]));      if tf then begin        Fps.WriteLn('finally', -1);        if tempvars <> nil then begin          for i:=0 to tempvars.Count - 1 do begin            vd:=TVarDef(tempvars.Objects[i]);            if IsJavaSimpleType(vd.VarType) then              Fps.WriteLn(Format('_env^^.Release%sArrayElements(_env, %s, %s_arr, 0);', [GetJniFuncType(vd.VarType), vd.Name, tempvars[i]]));          end;        end;        Fps.DecI;        Fps.WriteLn('end;');      end;      s:='';      if ProcType in [ptFunction, ptConstructor] then begin        s:='0';        if (ReturnType.DefType = dtType) and (TTypeDef(ReturnType).BasicType <= btDouble) then          s:='0'        else          s:=Format('%s(0)', [DefToJniType(ReturnType, err)]);        s:='Result:=' + s + ';';      end;      EHandlerEnd('_env', s);      Fps.DecI;      Fps.WriteLn('end;');      AParent:=d.Parent;    end    else      ClassIdx:=FClasses.IndexOf(GetJavaClassName(AParent, ItemDef), AParent);    if ClassIdx < 0 then begin      ci:=TClassInfo.Create;      ci.Def:=AParent;      s:=GetJavaClassName(AParent, ItemDef);      ci.IsCommonClass:=s <> AParent.Name;      ClassIdx:=FClasses.Add(s, AParent, ci);    end;    FClasses.GetClassInfo(ClassIdx).Funcs.Add(pi);    pi:=nil;    // Java part    s:=GetJavaProcDeclaration(d) + ';';    if (Parent.DefType = dtUnit) or (ProcType = ptConstructor) then      s:='static ' + s;    if Variable = nil then      Fjs.WriteLn('// ' + GetProcDeclaration(d));    if poPrivate in ProcOpt then      ss:='private'    else      if poProtected in ProcOpt then        ss:='protected'      else        ss:='public';    Fjs.WriteLn(ss + ' native ' + s);  finally    pi.Free;    tempvars.Free;  end;end;procedure TWriter.WriteVar(d: TVarDef; AParent: TDef);  function _WriteArrayIndex(pd: TProcDef): TDef;  var    ad: TArrayDef;    i: integer;  begin    ad:=TArrayDef(d.VarType);    i:=1;    repeat      with TVarDef.Create(pd, dtParam) do begin        Name:='Index';        if i > 1 then          Name:=Name + IntToStr(i);        VarType:=ad.RangeType;        if (VarType.DefType = dtType) and (TTypeDef(VarType).BasicType in [btByte, btShortInt, btSmallInt]) then          VarType:=FIntegerType;        VarOpt:=[voRead];        IsUsed:=True;      end;      Result:=ad.ElType;      ad:=TArrayDef(Result);      Inc(i);    until Result.DefType <> dtArray;  end;var  pd: TProcDef;  vd: TVarDef;  t: TTypeDef;  vt: TDef;  s, ss: string;  i: integer;  isarray, isdynarray: boolean;begin  if not d.IsUsed then    exit;  isarray:=(d.VarType <> nil) and (d.VarType.DefType = dtArray);  isdynarray:=isarray and (TArrayDef(d.VarType).RangeHigh < TArrayDef(d.VarType).RangeLow);  if isdynarray then    if not (voRead in d.VarOpt) then      exit    else      d.VarOpt:=d.VarOpt + [voWrite];  if d.VarType <> nil then begin    case d.DefType of      dtVar:        s:='var';      dtProp:        s:='property';      else        s:='';    end;    s:=Trim(s + ' ' + d.Name);    if d.Count > 0 then      s:=s + '[]';    ss:=d.VarType.Name;    if ss = '' then      if d.VarType.DefType = dtArray then        ss:='array';    Fjs.WriteLn(Format('// %s: %s', [s, ss]));  end;  if voRead in d.VarOpt then begin    pd:=TProcDef.Create(nil, dtProc);    try      pd.IsUsed:=True;      pd.Parent:=d.Parent;      pd.ProcType:=ptFunction;      pd.Name:='get' + d.Name;      if isarray then        // Array var        pd.ReturnType:=_WriteArrayIndex(pd)      else begin        pd.ReturnType:=d.VarType;        if d.DefType = dtProp then begin          for i:=0 to d.Count - 1 do begin            vd:=TVarDef(d.Items[i]);            with TVarDef.Create(pd, dtParam) do begin              Name:=vd.Name;              VarType:=vd.VarType;              VarOpt:=[voRead];            end;          end;        end;      end;      WriteProc(pd, d, AParent);    finally      pd.Free;    end;  end;  if voWrite in d.VarOpt then begin    pd:=TProcDef.Create(nil, dtProc);    try      pd.IsUsed:=True;      pd.Parent:=d.Parent;      pd.ProcType:=ptProcedure;      pd.Name:='set' + d.Name;      vt:=d.VarType;;      if isarray then begin        // Array var        if (d.DefType = dtProp) and not isdynarray then          exit;        vt:=_WriteArrayIndex(pd);      end      else        if d.DefType = dtProp then begin          for i:=0 to d.Count - 1 do begin            vd:=TVarDef(d.Items[i]);            with TVarDef.Create(pd, dtParam) do begin              Name:=vd.Name;              VarType:=vd.VarType;              VarOpt:=[voRead];            end;          end;        end;      s:='Value';      // Check if the name of value parameter is unique      i:=0;      while i < d.Count do begin        if AnsiCompareText(s, d.Items[i].AliasName) = 0 then begin          i:=0;          s:='_' + s;          continue;        end;        Inc(i);      end;      with TVarDef.Create(pd, dtParam) do begin        Name:=s;        VarType:=vt;        VarOpt:=[voRead];      end;      t:=TTypeDef.Create(nil, dtType);      try        t.BasicType:=btVoid;        pd.ReturnType:=t;        WriteProc(pd, d, AParent);      finally        t.Free;      end;    finally      pd.Free;    end;  end;end;procedure TWriter.WriteConst(d: TConstDef);var  s, v: string;begin  if not d.IsUsed then    exit;  v:=d.Value;  if d.VarType = nil then begin    if Copy(d.Value, 1, 1) = '"' then      s:='String'    else      s:='double';  end  else begin    s:='';    case d.VarType.DefType of      dtType:        case TTypeDef(d.VarType).BasicType of          btLongWord, btInt64:            v:=v + 'L';          btBoolean:            if v = '1' then              v:='true'            else              v:='false';        end;      dtArray:        with TArrayDef(d.VarType) do          if (ElType.DefType = dtType) and (TTypeDef(ElType).BasicType in [btChar, btWideChar]) then            s:='String';    end;    if s = '' then      s:=DefToJavaType(d.VarType);  end;  v:=Format('public static final %s %s = %s;', [s, d.Name, v]);  if s = SUnsupportedType then    v:='// ' + v;  Fjs.WriteLn(v);end;procedure TWriter.WriteEnum(d: TDef);var  i: integer;  s: string;begin  if not d.IsUsed then    exit;  RegisterPseudoClass(d);  WriteComment(d, 'enum');  Fjs.WriteLn(Format('public static class %s extends %s.system.Enum {', [d.Name, JavaPackage]));  Fjs.IncI;  for i:=0 to d.Count - 1 do begin    s:=Format('public final static int %s = %s;', [d[i].Name, TConstDef(d[i]).Value]);    Fjs.WriteLn(s);  end;  Fjs.WriteLn;  for i:=0 to d.Count - 1 do begin    s:=Format('public static %s %s() { return new %0:s(%1:s); }', [d.Name, d[i].Name]);    Fjs.WriteLn(s);  end;  Fjs.WriteLn;  Fjs.WriteLn(Format('public %s(int v) { Value = v; }', [d.Name]));  Fjs.WriteLn(Format('@Override public boolean equals(Object o) { return ((o instanceof %0:s) && Value == ((%0:s)o).Value) || super.equals(o); }', [d.Name]));  Fjs.DecI;  Fjs.WriteLn('}');  Fjs.WriteLn;end;procedure TWriter.WriteProcType(d: TProcDef; PreInfo: boolean);  procedure _AccessSimpleArray(vd: TVarDef; VarIndex: integer; DoSet: boolean);  begin    with vd do begin      Fps.WriteLn(Format('_tmp_%s:=_env^^.Get%sArrayElements(_env, _args[%d].L, PJBoolean(nil)^);', [Name, GetJniFuncType(VarType), VarIndex]));      Fps.WriteLn(Format('if _tmp_%s <> nil then', [Name]));      if DoSet then        Fps.WriteLn(Format('_tmp_%s^:=%s;', [Name, PasToJniType(VarType, Name)]), 1)      else        Fps.WriteLn(Format('%s:=%s;', [Name, JniToPasType(VarType, '_tmp_' + Name + '^', False)]), 1);      Fps.WriteLn(Format('_env^^.Release%sArrayElements(_env, _args[%d].L, _tmp_%s, 0);', [GetJniFuncType(VarType), VarIndex, Name]));    end;  end;var  vd: TVarDef;  i: integer;  s, ss, hclass: string;  err: boolean;begin  if not d.IsUsed or not (poMethodPtr in d.ProcOpt) then    exit;  if PreInfo then begin    WriteClassInfoVar(d);    // Handler proc    hclass:=GetClassPrefix(d) + 'Class';    Fps.WriteLn;    Fps.WriteLn(Format('type %s = class', [hclass]));    Fps.WriteLn(Format('private %s;', [ GetProcDeclaration(d, 'Handler', True, True)]), 1);    Fps.WriteLn('end;');    Fps.WriteLn;    Fps.WriteLn(GetProcDeclaration(d, Format('%s.Handler', [hclass]), True, True) + ';');    Fps.WriteLn('var');    Fps.IncI;    Fps.WriteLn('_env: PJNIEnv;');    Fps.WriteLn('_new_env: boolean;');    Fps.WriteLn('_mpi: _TMethodPtrInfo;');    if d.Count > 0 then begin      Fps.WriteLn(Format('_args: array[0..%d] of jvalue;', [d.Count - 1]));      for i:=0 to d.Count - 1 do begin        vd:=TVarDef(d[i]);        if vd.DefType <> dtParam then          continue;        with vd do          if IsJavaVarParam(vd) and IsJavaSimpleType(VarType) then            Fps.WriteLn(Format('_tmp_%s: P%s;', [Name, DefToJniType(VarType, err)]));      end;    end;    Fps.DecI;    Fps.WriteLn('begin');    Fps.IncI;    Fps.WriteLn('CurJavaVM^^.GetEnv(CurJavaVM, @_env, JNI_VERSION_1_6);');    Fps.WriteLn('_new_env:=_env = nil;');    Fps.WriteLn('if _new_env then CurJavaVM^^.AttachCurrentThread(CurJavaVM, @_env, nil);');    Fps.WriteLn('_env^^.PushLocalFrame(_env, 100);');    Fps.WriteLn('try');    Fps.IncI;    Fps.WriteLn('_MethodPointersCS.Enter;');    Fps.WriteLn('try');    Fps.WriteLn('_mpi:=_TMethodPtrInfo(_MethodPointers[-integer(ptruint(Self)) - 1]);', 1);    Fps.WriteLn('finally');    Fps.WriteLn('_MethodPointersCS.Leave;', 1);    Fps.WriteLn('end;');    for i:=0 to d.Count - 1 do begin      vd:=TVarDef(d[i]);      if vd.DefType <> dtParam then        continue;      with vd do begin        if not IsJavaVarParam(vd) then begin          s:='L';          if VarType.DefType = dtType then            s:=Copy(JNITypeSig[TTypeDef(VarType).BasicType], 1, 1);          ss:=PasToJniType(VarType, Name);        end        else begin          s:='L';          if IsJavaSimpleType(VarType) then            ss:=Format('_env^^.New%sArray(_env, 1)', [GetJniFuncType(VarType)])          else begin            if voVar in VarOpt then              ss:=PasToJniType(VarType, Name)            else              ss:='nil';            ss:=Format('_env^^.NewObjectArray(_env, 1, %s.ClassRef, %s)', [GetTypeInfoVar(VarType), ss]);          end;        end;        Fps.WriteLn(Format('_args[%d].%s:=%s;', [i, s, ss]));        if IsJavaVarParam(vd) and (voVar in VarOpt) and IsJavaSimpleType(VarType) then          _AccessSimpleArray(TVarDef(d[i]), i, True);      end;    end;    if d.Count > 0 then      s:='@_args'    else      s:='nil';    // Calling Java handler    s:=Format('_env^^.Call%sMethodA(_env, _mpi.Obj, _mpi.MethodId, %s)', [GetJniFuncType(d.ReturnType), s]);    if d.ProcType = ptFunction then      s:=Format('Result:=%s', [JniToPasType(d.ReturnType, s, False)]);    Fps.WriteLn(s + ';');    // Java exception check    Fps.WriteLn('_HandleJavaException(_env);');    // Processing var/out parameters    for i:=0 to d.Count - 1 do begin      vd:=TVarDef(d[i]);      if vd.DefType <> dtParam then        continue;      with vd do        if IsJavaVarParam(vd) then          if IsJavaSimpleType(VarType) then            _AccessSimpleArray(TVarDef(d[i]), i, False)          else begin            s:=Format('_env^^.GetObjectArrayElement(_env, _args[%d].L, 0)', [i]);            Fps.WriteLn(Format('%s:=%s;', [Name, JniToPasType(VarType, s, False)]));          end;    end;    Fps.DecI;    Fps.WriteLn('finally');    Fps.WriteLn('_env^^.PopLocalFrame(_env, nil);', 1);    Fps.WriteLn('if _new_env then CurJavaVM^^.DetachCurrentThread(CurJavaVM);', 1);    Fps.WriteLn('end;');    Fps.DecI;    Fps.WriteLn('end;');    // Get handler proc    Fps.WriteLn;    Fps.WriteLn(Format('function %sGetHandler(env: PJNIEnv; jobj: jobject; const ci: _TJavaClassInfo): %s.%s;',                       [GetClassPrefix(d), d.Parent.Name, d.Name]));    Fps.WriteLn('begin');    Fps.WriteLn(Format('TMethod(Result):=_GetMethodPtrHandler(env, jobj, @%s.Handler, %s);', [hclass, GetTypeInfoVar(d)]), 1);    Fps.WriteLn('end;');    exit;  end;  err:=False;  WriteComment(d, 'procedural type');  RegisterPseudoClass(d);  Fjs.WriteLn(Format('/* Pascal prototype: %s */', [GetProcDeclaration(d, 'Execute')]));  Fjs.WriteLn(Format('/* Java prototype: %s */', [GetJavaProcDeclaration(d, 'Execute')]));  Fjs.WriteLn(Format('public static class %s extends %s.system.MethodPtr {', [d.Name, JavaPackage]));  Fjs.IncI;  Fjs.WriteLn(Format('{ mSignature = "%s"; }', [GetProcSignature(d)]));  Fjs.WriteLn(Format('protected %s(long objptr, boolean cleanup) { _pasobj=objptr; }', [d.Name]));  Fjs.WriteLn(Format('@Deprecated public %s(Object Obj, String MethodName) { mObject=Obj; mName=MethodName; }', [d.Name]));  Fjs.WriteLn(Format('public %s() { mObject=this; mName="Execute"; }', [d.Name]));  Fjs.WriteLn(Format('protected %s throws NoSuchMethodException { throw new NoSuchMethodException(); }', [GetJavaProcDeclaration(d, 'Execute')]));  Fjs.DecI;  Fjs.WriteLn('}');  Fjs.WriteLn;end;procedure TWriter.WriteSet(d: TSetDef);begin  if not d.IsUsed then    exit;  if d.ElType = nil then    raise Exception.Create('No element type.');  WriteComment(d, '');  Fjs.WriteLn(Format('/* set of %s */', [d.ElType.Name]));  if d.Size > 4 then begin    Fjs.WriteLn('/* Set size more than 32 bits is not supported */');    exit;  end;  RegisterPseudoClass(d);  Fjs.WriteLn(Format('public static class %s extends %s.system.Set<%s,%s> {', [d.Name, JavaPackage, d.Name, d.ElType.Name]));  Fjs.IncI;  Fjs.WriteLn(Format('@Override protected byte Size() { return %d; }', [d.Size]));  Fjs.WriteLn(Format('@Override protected int Base() { return %d; }', [d.Base]));  Fjs.WriteLn(Format('@Override protected int ElMax() { return %d; }', [d.ElMax]));  Fjs.WriteLn(Format('public %s() { }', [d.Name]));  Fjs.WriteLn(Format('public %s(%s... Elements) { super(Elements); }', [d.Name, d.ElType.Name]));  Fjs.WriteLn(Format('public %0:s(%0:s... Elements) { super(Elements); }', [d.Name]));  Fjs.WriteLn(Format('public static %0:s Exclude(%0:s s1, %0:s s2) { %0:s r = new %0:s(s1); r.Exclude(s2); return r; }', [d.Name]));  Fjs.WriteLn(Format('public static %0:s Intersect(%0:s s1, %0:s s2) { %0:s r = new %0:s(s1); r.Intersect(s2); return r; }', [d.Name]));  Fjs.DecI;  Fjs.WriteLn('}');  Fjs.WriteLn;end;procedure TWriter.WritePointer(d: TPointerDef; PreInfo: boolean);begin  if not d.IsUsed or not d.IsObjPtr then    exit;  if PreInfo then begin    RegisterPseudoClass(d);    WriteClassInfoVar(d);    exit;  end;  WriteComment(d, 'pointer');  Fjs.WriteLn(Format('public static class %s extends %s {', [d.Name, d.PtrType.Name]));  Fjs.IncI;  if TClassDef(d.PtrType).CType in [ctObject, ctRecord] then    Fjs.WriteLn(Format('protected %s(long objptr, boolean cleanup) { super(objptr, false); }', [d.Name]));  Fjs.WriteLn(Format('public %s(PascalObject obj) { super(obj); }', [d.Name]));  Fjs.WriteLn(Format('public %s(long objptr) { super(objptr); }', [d.Name]));  Fjs.DecI;  Fjs.WriteLn('}');  Fjs.WriteLn;end;procedure TWriter.WriteClassRef(d: TClassRefDef; PreInfo: boolean);begin  if not d.IsUsed then    exit;  if PreInfo then begin    RegisterPseudoClass(d);    WriteClassInfoVar(d);    exit;  end;  WriteComment(d, 'class ref');  Fjs.WriteLn(Format('public static class %s extends %s {', [d.Name, d.ClassRef.Name]));  Fjs.IncI;  Fjs.WriteLn(Format('public %s(PascalObject obj) { super(obj); }', [d.Name]));  Fjs.DecI;  Fjs.WriteLn('}');  Fjs.WriteLn;end;procedure TWriter.WriteUnit(u: TUnitDef);  procedure _ProcessExcludedProcParams(d: TDef);  var    i: integer;  begin    if not d.IsUsed then      exit;    if d.DefType in [dtProc, dtProcType] then begin      for i:=0 to d.Count - 1 do        if d[i].DefType = dtParam then          with TVarDef(d[i]) do            if (VarType <> nil) and not VarType.IsUsed then begin              d.SetNotUsed;              break;            end;    end    else      for i:=0 to d.Count - 1 do        _ProcessExcludedProcParams(d[i]);  end;var  d: TDef;  i: integer;  f: boolean;begin  if u.Processed then    exit;  u.Processed:=True;  if not u.IsUsed then    exit;  _ProcessExcludedProcParams(u);  for i:=0 to High(u.UsedUnits) do    WriteUnit(u.UsedUnits[i]);  Fps.WriteLn;  Fps.WriteLn(Format('{ Unit %s }', [u.Name]));  u.Name:=LowerCase(u.Name);  Fjs:=TTextOutStream.Create(IncludeTrailingPathDelimiter(FPkgDir) + u.Name + '.java', fmCreate);  try    WriteFileComment(Fjs);    Fjs.WriteLn(Format('package %s;', [JavaPackage]));    if Length(u.UsedUnits) > 0 then begin      UpdateUsedUnits(u);      f:=False;      for i:=0 to High(u.UsedUnits) do        if u.UsedUnits[i].IsUnitUsed then begin          if not f then begin            Fjs.WriteLn;            f:=True;          end;          Fjs.WriteLn(Format('import %s.%s.*;', [JavaPackage, LowerCase(u.UsedUnits[i].Name)]));        end;    end;    if u.Name = 'system' then begin      Fjs.WriteLn;      Fjs.WriteLn('import java.util.Date;');      Fjs.WriteLn('import java.util.TimeZone;');    end;    Fjs.WriteLn;    Fjs.WriteLn('public class ' + u.Name + ' {');    Fjs.IncI;    if u.Name = 'system' then begin      for i:=0 to u.Count - 1 do begin        d:=u[i];        if (d.DefType = dtType) and (TTypeDef(d).BasicType = btLongInt) and (Copy(d.Name, 1, 1) <> '$') then begin          FIntegerType:=d;          break;        end;      end;      if FIntegerType = nil then        raise Exception.Create('LongInt type has not been found in the System unit.');      if LibAutoLoad then begin        Fjs.WriteLn('static private boolean _JniLibLoaded = false;');        Fjs.WriteLn('public static void InitJni() {');        Fjs.WriteLn('if (!_JniLibLoaded) {', 1);        Fjs.WriteLn('_JniLibLoaded=true;', 2);        Fjs.WriteLn(Format('System.loadLibrary("%s");', [LibName]), 2);        Fjs.WriteLn('}', 1);        Fjs.WriteLn('}');      end;      // Public support functions      Fjs.WriteLn('public native static long AllocMemory(int Size);');      AddNativeMethod(u, '_AllocMemory', 'AllocMemory', '(I)J');      Fjs.WriteLn('public native static byte[] GetMemoryAsArray(long SrcBuf, int BufSize);');      AddNativeMethod(u, '_GetMemoryAsArray', 'GetMemoryAsArray', '(JI)[B');      Fjs.WriteLn('public native static void SetMemoryFromArray(long DstBuf, byte[] SrcArray);');      AddNativeMethod(u, '_SetMemoryFromArray', 'SetMemoryFromArray', '(J[B)V');      // Base object      Fjs.WriteLn;      Fjs.WriteLn('public static class PascalObject {');      Fjs.IncI;      if LibAutoLoad then        Fjs.WriteLn(Format('static { %s.system.InitJni(); }', [JavaPackage]));      Fjs.WriteLn('protected long _pasobj = 0;');      Fjs.WriteLn('protected PascalObject() { }');      Fjs.WriteLn('protected PascalObject(PascalObject obj) { if (obj != null) _pasobj=obj._pasobj; }');      Fjs.WriteLn('protected PascalObject(long objptr) { _pasobj=objptr; }');      Fjs.WriteLn('@Override protected void finalize() { }');      Fjs.WriteLn('@Override public boolean equals(Object o) { return ((o instanceof PascalObject) && _pasobj == ((PascalObject)o)._pasobj); }');      Fjs.WriteLn('@Override public int hashCode() { return (int)_pasobj; }');      Fjs.DecI;      Fjs.WriteLn('}');      Fjs.WriteLn;      Fjs.WriteLn('public static long Pointer(PascalObject obj) { return (obj == null) ? 0 : obj._pasobj; }');      // Object with finalization      Fjs.WriteLn;      Fjs.WriteLn('public static class PascalObjectEx extends PascalObject {');      Fjs.IncI;      Fjs.WriteLn('protected boolean _cleanup = false;');      Fjs.WriteLn('@Override protected void finalize() { ');{$ifdef DEBUG}      Fjs.WriteLn('String s = "finalize(): " + getClass().getName(); if (_cleanup) s=s+". Need __Release(). ptr="+_pasobj; System.out.println(s);', 1);{$endif DEBUG}      Fjs.WriteLn('if (_cleanup) __Release();', 1);      Fjs.WriteLn('super.finalize();', 1);      Fjs.WriteLn('}');      Fjs.WriteLn('protected PascalObjectEx() { }');      Fjs.WriteLn('protected PascalObjectEx(PascalObject obj) { super(obj); }');      Fjs.WriteLn('protected PascalObjectEx(long objptr) { super(objptr); }');      Fjs.WriteLn('public void __Release() { _pasobj = 0; }');      Fjs.DecI;      Fjs.WriteLn('}');      // Record      Fjs.WriteLn;      Fjs.WriteLn('public static class Record extends PascalObjectEx {');      Fjs.IncI;      Fjs.WriteLn('protected PascalObject _objref;');      Fjs.WriteLn('@Override protected void finalize() { if (_pasobj < 0) { _pasobj=-_pasobj; _cleanup=true; } super.finalize(); }');      Fjs.WriteLn('protected void __Init(long objptr, boolean cleanup) { _pasobj=objptr; _cleanup=cleanup; if (_pasobj==0 && __Size() != 0) _pasobj=AllocMemory(__Size()); }');      Fjs.WriteLn('protected Record(PascalObject obj) { super(obj); _objref=obj; }');      Fjs.WriteLn('protected Record(long objptr) { super(objptr); }');      Fjs.WriteLn('protected final int __Size(int index) { return GetRecordSize(index); }');      Fjs.WriteLn('public Record() { }');      Fjs.WriteLn('public int __Size() { return 0; }');      Fjs.DecI;      Fjs.WriteLn('}');      Fjs.WriteLn;      Fjs.WriteLn('private native static int GetRecordSize(int index);');      AddNativeMethod(u, '_GetRecordSize', 'GetRecordSize', '(I)I');      // Method pointer base class      d:=TClassDef.Create(FThisUnit, dtClass);      d.Name:='_TMethodPtrInfo';      d.AliasName:='MethodPtr';      WriteClassInfoVar(d);      // Method pointer support      Fps.WriteLn;      Fps.WriteLn('type');      Fps.IncI;      Fps.WriteLn('_TMethodPtrInfo = class');      Fps.IncI;      Fps.WriteLn('Obj: JObject;');      Fps.WriteLn('MethodId: JMethodID;');      Fps.WriteLn('Index, RefCnt: integer;');      Fps.WriteLn('RealMethod: TMethod;');      Fps.WriteLn('InlineHandler: boolean;');      Fps.WriteLn('constructor Create(env: PJNIEnv; JavaObj: JObject; const AMethodName, AMethodSig: ansistring);');      Fps.WriteLn('procedure Release(env: PJNIEnv);');      Fps.DecI;      Fps.WriteLn('end;');      Fps.DecI;      Fps.WriteLn;      Fps.WriteLn('var _MethodPointers: array of _TMethodPtrInfo;');      Fps.WriteLn('var _MethodPointersCS: TCriticalSection;');      Fps.WriteLn;      Fps.WriteLn('constructor _TMethodPtrInfo.Create(env: PJNIEnv; JavaObj: JObject; const AMethodName, AMethodSig: ansistring);');      Fps.WriteLn('var c: JClass;');      Fps.WriteLn('begin');      Fps.IncI;      Fps.WriteLn('if (JavaObj = nil) or (AMethodName = '''') then exit;');      Fps.WriteLn('c:=env^^.GetObjectClass(env, JavaObj);');      Fps.WriteLn('if c = nil then exit;');      Fps.WriteLn('MethodId:=env^^.GetMethodID(env, c, PAnsiChar(AMethodName), PAnsiChar(AMethodSig));');      Fps.WriteLn('if MethodId = nil then raise Exception.CreateFmt(''Method "%s" does not exist or has wrong parameters.'', [AMethodName]);');      Fps.WriteLn('Obj:=env^^.NewGlobalRef(env, JavaObj);');      Fps.WriteLn('_MethodPointersCS.Enter;');      Fps.WriteLn('try');      Fps.IncI;      Fps.WriteLn('Index:=Length(_MethodPointers) + 1;');      Fps.WriteLn(Format('if Index > %d then raise Exception.Create(''Too many method pointers.'');', [MaxMethodPointers]));      Fps.WriteLn('SetLength(_MethodPointers, Index);');      Fps.WriteLn('_MethodPointers[Index - 1]:=Self;');      Fps.WriteLn('finally', -1);      Fps.WriteLn('_MethodPointersCS.Leave;');      Fps.DecI;      Fps.WriteLn('end;');      Fps.DecI;      Fps.WriteLn('end;');      Fps.WriteLn;      Fps.WriteLn('procedure _TMethodPtrInfo.Release(env: PJNIEnv);');      Fps.WriteLn('var i: integer;');      Fps.WriteLn('begin');      Fps.IncI;      Fps.WriteLn('i:=InterlockedDecrement(RefCnt);');      {$ifdef DEBUG}      Fps.WriteLn('writeln(''_TMethodPtrInfo.Release(). RefCnt='',i,'' ptr='',ptruint(Self));');      {$endif DEBUG}      Fps.WriteLn('if i <> 0 then exit;');      Fps.WriteLn('if Index > 0 then begin');      Fps.IncI;      Fps.WriteLn('_MethodPointersCS.Enter;');      Fps.WriteLn('try');      Fps.IncI;      Fps.WriteLn('if InlineHandler then begin');      Fps.IncI;      {$ifdef DEBUG}      Fps.WriteLn('writeln(''Finalizing Java inline handler.'');');      {$endif DEBUG}      Fps.WriteLn(Format('env^^.SetLongField(env, Obj, %s.ObjFieldId, -1);', [GetTypeInfoVar(d)]));      Fps.DecI;      Fps.WriteLn('end;');      Fps.WriteLn('env^^.DeleteGlobalRef(env, Obj);');      Fps.WriteLn('_MethodPointers[Index-1]:=nil;');      Fps.WriteLn('Index:=High(_MethodPointers);');      Fps.WriteLn('while (Index >= 0) and (_MethodPointers[Index] = nil) do Dec(Index);');      Fps.WriteLn('SetLength(_MethodPointers, Index + 1);');      Fps.WriteLn('finally', -1);      Fps.WriteLn('_MethodPointersCS.Leave;');      Fps.DecI;      Fps.WriteLn('end;');      Fps.DecI;      Fps.WriteLn('end;');      Fps.WriteLn('Self.Destroy;');  {$ifdef DEBUG}      Fps.WriteLn('writeln(''_TMethodPtrInfo destroyed.'');');  {$endif DEBUG}      Fps.DecI;      Fps.WriteLn('end;');      Fps.WriteLn;      Fps.WriteLn('procedure _RefMethodPtr(env: PJNIEnv; const m: TMethod; AddRef: boolean);');      Fps.WriteLn('var i: integer;');      Fps.WriteLn('begin');      Fps.IncI;      Fps.WriteLn('i:=-integer(ptruint(m.Data));');  {$ifdef DEBUG}      Fps.WriteLn('writeln(''_RefMethodPtr. i='',i,''  AddRef='',AddRef);');  {$endif DEBUG}      Fps.WriteLn(Format('if (i < 1) or (i > %d) then exit;', [MaxMethodPointers]));      Fps.WriteLn('_MethodPointersCS.Enter;');      Fps.WriteLn('try');      Fps.IncI;      Fps.WriteLn('with _MethodPointers[i - 1] do');      Fps.WriteLn('if AddRef then InterlockedIncrement(RefCnt) else Release(env);', 1);      Fps.WriteLn('finally', -1);      Fps.WriteLn('_MethodPointersCS.Leave;');      Fps.DecI;      Fps.WriteLn('end;');      Fps.DecI;      Fps.WriteLn('end;');      Fps.WriteLn;      Fps.WriteLn('function _CreateMethodPtrObject(env: PJNIEnv; const m: TMethod; const ci: _TJavaClassInfo): jobject;');      Fps.WriteLn('var i: integer;');      Fps.WriteLn('var mpi: _TMethodPtrInfo;');      Fps.WriteLn('begin');      Fps.IncI;      Fps.WriteLn('Result:=nil;');      Fps.WriteLn('if (m.Data = nil) and (m.Code = nil) then exit;');      Fps.WriteLn('_MethodPointersCS.Enter;');      Fps.WriteLn('try');      Fps.IncI;      Fps.WriteLn('i:=-integer(ptruint(m.Data));');      Fps.WriteLn(Format('if (i > 0) and (i <= %d) then begin', [MaxMethodPointers]));      Fps.WriteLn('mpi:=_MethodPointers[i - 1];', 1);      Fps.WriteLn('end');      Fps.WriteLn('else begin');      Fps.WriteLn('mpi:=_TMethodPtrInfo.Create(env, nil, '''', '''');', 1);      Fps.WriteLn('mpi.RealMethod:=m;', 1);      Fps.WriteLn('end;');      Fps.WriteLn('InterlockedIncrement(mpi.RefCnt);');      Fps.WriteLn('finally', -1);      Fps.WriteLn('_MethodPointersCS.Leave;');      Fps.DecI;      Fps.WriteLn('end;');      Fps.WriteLn('Result:=_CreateJavaObj(env, pointer(mpi), ci);');      Fps.DecI;      Fps.WriteLn('end;');      Fps.WriteLn;      Fps.WriteLn('function _GetMethodPtrHandler(env: PJNIEnv; jobj: jobject; hptr: pointer; const ci: _TJavaClassInfo): TMethod;');      Fps.WriteLn('var mpi: _TMethodPtrInfo;');      Fps.WriteLn('begin');      Fps.IncI;      Fps.WriteLn(  'Result.Data:=nil; Result.Code:=nil;');      Fps.WriteLn(  'mpi:=_TMethodPtrInfo(_GetPasObj(env, jobj, ci, False));');      Fps.WriteLn(  'if mpi = nil then exit;');      Fps.WriteLn(  'if pointer(mpi) = pointer(ptruint(-1)) then begin');      Fps.WriteLn(    'env^^.CallVoidMethodA(env, jobj, env^^.GetMethodID(env, ci.ClassRef, ''Init'', ''()V''), nil);', 1);      Fps.WriteLn(    'Result:=_GetMethodPtrHandler(env, jobj, hptr, ci);', 1);      Fps.WriteLn(    'exit;', 1);      Fps.WriteLn(  'end;');      Fps.WriteLn(  'if mpi.Index = 0 then');      Fps.WriteLn(    'TMethod(Result):=mpi.RealMethod', 1);      Fps.WriteLn(  'else');      Fps.WriteLn(    'with TMethod(Result) do begin', 1);      Fps.WriteLn(      'Data:=pointer(ptruint(-integer(mpi.Index)));', 2);      Fps.WriteLn(      'Code:=hptr;', 2);      Fps.WriteLn(    'end;', 1);      Fps.DecI;      Fps.WriteLn('end;');      Fps.WriteLn;      Fps.WriteLn('procedure _TMethodPtrInfo_Init(env: PJNIEnv; _self, JavaObj: JObject; AMethodName, AMethodSig: jstring; IncRef: jboolean);' + JniCaliing);      Fps.WriteLn('var mpi: _TMethodPtrInfo;');      Fps.WriteLn('begin');      Fps.IncI;      EHandlerStart;      Fps.WriteLn('mpi:=_TMethodPtrInfo.Create(env, JavaObj, ansistring(_StringFromJString(env, AMethodName)), ansistring(_StringFromJString(env, AMethodSig)));');      Fps.WriteLn('if IncRef <> 0 then');      Fps.WriteLn('InterlockedIncrement(mpi.RefCnt)', 1);      Fps.WriteLn('else');      Fps.WriteLn('mpi.InlineHandler:=True;', 1);{$ifdef DEBUG}      Fps.WriteLn('writeln(''_TMethodPtrInfo_Init. RefCnt='',mpi.RefCnt,'' ptr='',ptruint(mpi));');{$endif DEBUG}      Fps.WriteLn(Format('env^^.SetLongField(env, _self, %s.ObjFieldId, Int64(ptruint(mpi)));', [GetTypeInfoVar(d)]));      EHandlerEnd('env');      Fps.DecI;      Fps.WriteLn('end;');      AddNativeMethod(d, '_TMethodPtrInfo_Init', '__Init', Format('(Ljava/lang/Object;%s%sZ)V', [JNITypeSig[btString], JNITypeSig[btString]]));      Fps.WriteLn;      Fps.WriteLn('procedure _TMethodPtrInfo_Release(env: PJNIEnv; _self: JObject);' + JniCaliing);      Fps.WriteLn('begin');      Fps.IncI;      EHandlerStart;      Fps.WriteLn(Format('_TMethodPtrInfo(_GetPasObj(env, _self, %s, True)).Release(env);', [GetTypeInfoVar(d)]));      EHandlerEnd('env');      Fps.DecI;      Fps.WriteLn('end;');      AddNativeMethod(d, '_TMethodPtrInfo_Release', '__Destroy', '()V');      Fjs.WriteLn;      Fjs.WriteLn('public static class MethodPtr extends PascalObjectEx {');      Fjs.IncI;      Fjs.WriteLn('private native void __Init(Object Obj, String MethodName, String MethodSignature, boolean IncRef);');      Fjs.WriteLn('private native void __Destroy();');      Fjs.WriteLn('protected Object mObject;');      Fjs.WriteLn('protected String mName;');      Fjs.WriteLn('protected String mSignature;');      Fjs.WriteLn('protected void Init() { __Init(mObject, mName, mSignature, this != mObject); }');      Fjs.WriteLn('protected MethodPtr() { _cleanup=true; _pasobj=-1; }');      Fjs.WriteLn('public void __Release() { if (_pasobj > 0) __Destroy(); }');      Fjs.DecI;      Fjs.WriteLn('}');      Fjs.WriteLn;      // Base class for Enum      Fjs.WriteLn('public static class Enum {');      Fjs.IncI;      Fjs.WriteLn('public int Value;');      Fjs.WriteLn('public int Ord() { return Value; }');      Fjs.WriteLn('@Override public boolean equals(Object o) { return (o instanceof Integer) && Value == (Integer)o; }');      Fjs.WriteLn('public boolean equals(int v) { return Value == v; }');      Fjs.WriteLn('@Override public int hashCode() { return Value; }');      Fjs.DecI;      Fjs.WriteLn('}');      Fjs.WriteLn;      // Base class for Set      Fjs.WriteLn('private static abstract class BaseSet {');      Fjs.IncI;      Fjs.WriteLn('protected int Value = 0;');      Fjs.WriteLn('protected abstract byte Size();');      Fjs.WriteLn('protected abstract int Base();');      Fjs.WriteLn('protected abstract int ElMax();');      Fjs.WriteLn('public BaseSet() { }');      Fjs.DecI;      Fjs.WriteLn('}');      Fjs.WriteLn('public static abstract class Set<TS extends BaseSet,TE extends Enum> extends BaseSet {');      Fjs.IncI;      Fjs.WriteLn('protected int GetMask(TE Element) { return 1 << (Element.Ord() - Base()); }');      Fjs.WriteLn('public Set() { }');      Fjs.WriteLn('@SuppressWarnings({"unchecked", "varargs"})');      Fjs.WriteLn('public Set(TE... Elements) { Include(Elements); }');      Fjs.WriteLn('@SuppressWarnings({"unchecked", "varargs"})');      Fjs.WriteLn('public Set(TS... Elements) { for (TS e : Elements) Include(e); }');      Fjs.WriteLn('@SuppressWarnings({"unchecked", "varargs"})');      Fjs.WriteLn('public void Include(TE... Elements) { for (TE e: Elements) Value = Value | GetMask(e); }');      Fjs.WriteLn('public void Include(TS s) { Value=Value | s.Value; }');      Fjs.WriteLn('@SuppressWarnings({"unchecked", "varargs"})');      Fjs.WriteLn('public void Exclude(TE... Elements) { for (TE e: Elements) Value = Value & ~GetMask(e); }');      Fjs.WriteLn('public void Exclude(TS s) { Value=Value & ~s.Value; }');      Fjs.WriteLn('public void Assign(TS s) { Value=s.Value; }');      Fjs.WriteLn('public void Intersect(TS s) { Value=Value & s.Value; }');      Fjs.WriteLn('public boolean Has(TE Element) { return (Value & GetMask(Element)) != 0; }');      Fjs.WriteLn('public boolean IsEmpty() { return Value == 0; }');      Fjs.WriteLn('public boolean equals(TS s) { return Value == s.Value; }');      Fjs.WriteLn('public boolean equals(TE Element) { return Value == Element.Ord(); }');      Fjs.WriteLn('public boolean equals(int Element) { return Value == Element; }');      Fjs.DecI;      Fjs.WriteLn('}');      Fjs.WriteLn;      // TDateTime support      Fjs.WriteLn('public static class TDateTime {');      Fjs.IncI;      Fjs.WriteLn('public static Date toDateUTC(double d) {');      Fjs.WriteLn('return new Date(Math.round((d - 25569)*86400000.0));', 1);      Fjs.WriteLn('}');      Fjs.WriteLn('public static Date toDate(double d) {');      Fjs.WriteLn('long t = Math.round((d - 25569)*86400000.0); return new Date(t - TimeZone.getDefault().getOffset(t));', 1);      Fjs.WriteLn('}');      Fjs.WriteLn('public static double getUTC(Date d) {');      Fjs.WriteLn('return d.getTime()/86400000.0 + 25569;', 1);      Fjs.WriteLn('}');      Fjs.WriteLn('public static double get(Date d) {');      Fjs.WriteLn('return (d.getTime() + TimeZone.getDefault().getOffset(d.getTime()))/86400000.0 + 25569;', 1);      Fjs.WriteLn('}');      Fjs.DecI;      Fjs.WriteLn('}');      Fjs.WriteLn;      // Interface support      Fps.WriteLn;      Fps.WriteLn('function _IntfCast(env: PJNIEnv; _self: JObject; objptr: jlong; objid: jstring): jlong;' + JniCaliing);      Fps.WriteLn('var');      Fps.WriteLn('obj: system.TObject;', 1);      Fps.WriteLn('intf: IUnknown;', 1);      Fps.WriteLn('begin');      Fps.IncI;      Fps.WriteLn('Result:=0;');      EHandlerStart;      Fps.WriteLn('if objptr = 0 then exit;');      Fps.WriteLn('if objid = nil then');      Fps.WriteLn('raise Exception.Create(''A GUID must be assigned for the interface to allow a type cast.'');', 1);      Fps.WriteLn('obj:=system.TObject(pointer(ptruint(objptr)));');      Fps.WriteLn('if not (obj is system.TInterfacedObject) then');      Fps.WriteLn('raise Exception.Create(''Object must be inherited from TInterfacedObject.'');', 1);      Fps.WriteLn('if (system.TInterfacedObject(obj) as IUnknown).QueryInterface(StringToGUID(ansistring(_StringFromJString(env, objid))), intf) <> 0 then');      Fps.WriteLn('raise Exception.Create(''Invalid type cast.'');', 1);      Fps.WriteLn('intf._AddRef;');      Fps.WriteLn('Result:=ptruint(intf);');      EHandlerEnd('env');      Fps.DecI;      Fps.WriteLn('end;');      AddNativeMethod(u, '_IntfCast', 'InterfaceCast', '(JLjava/lang/String;)J');      Fjs.WriteLn('private native static long InterfaceCast(long objptr, String objid);');      Fjs.WriteLn;      Fjs.WriteLn('public static abstract class PascalInterface extends PascalObjectEx {');      Fjs.IncI;      Fjs.WriteLn('abstract protected void __Init();');      Fjs.WriteLn('public void __TypeCast(PascalObject obj, String intfId) {');      Fjs.WriteLn('if (obj != null) {', 1);      Fjs.WriteLn('if (obj instanceof PascalInterface) {', 2);      Fjs.WriteLn('_pasobj=obj._pasobj;',3);      Fjs.WriteLn('__Init();',3);      Fjs.WriteLn('} else',2);      Fjs.WriteLn('_pasobj=InterfaceCast(obj._pasobj, intfId);', 3);      Fjs.WriteLn('}', 1);      Fjs.WriteLn('}');      Fjs.WriteLn('protected PascalInterface(long objptr, boolean cleanup) { _pasobj=objptr; __Init(); }');      Fjs.DecI;      Fjs.WriteLn('}');      Fjs.WriteLn;    end;    if LibAutoLoad then begin      Fjs.WriteLn(Format('static { %s.system.InitJni(); }', [JavaPackage]));      Fjs.WriteLn;    end;    // First pass    for i:=0 to u.Count - 1 do begin      d:=u[i];      if not d.IsUsed then        continue;      case d.DefType of        dtSet, dtEnum:          WriteClassInfoVar(d);        dtClass:          WriteClass(TClassDef(d), True);        dtProcType:          WriteProcType(TProcDef(d), True);        dtPointer:          WritePointer(TPointerDef(d), True);        dtClassRef:          WriteClassRef(TClassRefDef(d), True);      end;    end;    // Second pass    for i:=0 to u.Count - 1 do begin      d:=u[i];      if not d.IsUsed then        continue;      case d.DefType of        dtClass:          WriteClass(TClassDef(d), False);        dtProc:          WriteProc(TProcDef(d));        dtVar, dtProp:          WriteVar(TVarDef(d));        dtEnum:          WriteEnum(d);        dtProcType:          WriteProcType(TProcDef(d), False);        dtSet:          WriteSet(TSetDef(d));        dtConst:          WriteConst(TConstDef(d));        dtPointer:          WritePointer(TPointerDef(d), False);        dtClassRef:          WriteClassRef(TClassRefDef(d), False);      end;    end;    // Class ref helpers    if (u.Name = 'system') and (FClasses.IndexOf('system.TClass', nil) >= 0) then begin      Fjs.WriteLn('native static long GetClassRef(int index);');      AddNativeMethod(u, '_GetClassRef', 'GetClassRef', '(I)J');      Fjs.WriteLn('static TClass GetTClass(int index) { TClass c = new TClass(null); c._pasobj=GetClassRef(index); return c; }');      Fjs.WriteLn;    end;    Fjs.DecI;    Fjs.WriteLn('}');  finally    Fjs.Free;  end;end;procedure TWriter.WriteOnLoad;var  i, j: integer;  ci: TClassInfo;  s, ss, fn: string;  d: TTypeDef;begin  if FClasses.Count = 0 then    exit;  Fps.WriteLn;  Fps.WriteLn('function JNI_OnLoad(vm: PJavaVM; reserved: pointer): jint;' + JniCaliing);  Fps.WriteLn('const');  for i:=0 to FClasses.Count - 1 do begin    ci:=FClasses.GetClassInfo(i);    if ci.Funcs.Count = 0 then      continue;    Fps.WriteLn(Format('  _%sNativeMethods: array[0..%d] of JNINativeMethod = (', [GetClassPrefix(ci.Def, FClasses.GetClassName(i)), ci.Funcs.Count - 1]));    for j:=0 to ci.Funcs.Count - 1 do begin      with TProcInfo(ci.Funcs[j]) do        Fps.Write(Format('    (name: ''%s''; signature: ''%s''; fnPtr: @%s)', [Name, JniSignature, JniName]));      if j < ci.Funcs.Count - 1 then        Fps.Write(',');      Fps.WriteLn;    end;    Fps.WriteLn('  );');  end;  Fps.WriteLn;  Fps.WriteLn('var');  Fps.IncI;  Fps.WriteLn('env: PJNIEnv;');  Fps.WriteLn;  Fps.WriteLn('function _Reg(ClassName: PAnsiChar; Methods: PJNINativeMethod; Count: integer; ci: _PJavaClassInfo; const FieldName: ansistring = ''_pasobj''; const FieldSig: ansistring = ''J''): boolean;');  Fps.WriteLn('var');  Fps.WriteLn('c: jclass;', 1);  Fps.WriteLn('begin');  Fps.IncI;  Fps.WriteLn('Result:=False;');  Fps.WriteLn('c:=env^^.FindClass(env, ClassName);');  Fps.WriteLn('if c = nil then exit;');  Fps.WriteLn('Result:=(Count = 0) or (env^^.RegisterNatives(env, c, Methods, Count) = 0);');  Fps.WriteLn('if Result and (ci <> nil) then begin');  Fps.IncI;  Fps.WriteLn('ci^.ClassRef:=env^^.NewGlobalRef(env, c);');  Fps.WriteLn('Result:=ci^.ClassRef <> nil;');  Fps.WriteLn('if Result and (env^^.ExceptionCheck(env) = 0) then begin');  Fps.WriteLn('ci^.ConstrId:=env^^.GetMethodID(env, ci^.ClassRef, ''<init>'', ''(JZ)V'');', 1);  Fps.WriteLn('env^^.ExceptionClear(env);', 1);  Fps.WriteLn('end;');  Fps.WriteLn('if Result and (FieldName <> '''') then begin');  Fps.WriteLn('ci^.ObjFieldId:=env^^.GetFieldID(env, ci^.ClassRef, PAnsiChar(FieldName), PAnsiChar(FieldSig));', 1);  Fps.WriteLn('Result:=ci^.ObjFieldId <> nil;', 1);  Fps.WriteLn('end;');  Fps.DecI;  Fps.WriteLn('end;');  Fps.DecI;  Fps.WriteLn('end;');  Fps.WriteLn;  Fps.WriteLn('begin', -1);  Fps.WriteLn('Result:=JNI_ERR;');  Fps.WriteLn('if vm^^.GetEnv(vm, @env, JNI_VERSION_1_6) <> JNI_OK then exit;');  Fps.WriteLn('CurJavaVM:=vm;');  d:=TTypeDef.Create(nil, dtType);  try    d.BasicType:=btString;    s:=JNITypeSig[d.BasicType];    s:=Copy(s, 2, Length(s) - 2);    Fps.WriteLn(Format('if not _Reg(''%s'', nil, 0, @%s, '''', '''') then exit;',                       [s, GetTypeInfoVar(d)]));  finally    d.Free;  end;  for i:=0 to FClasses.Count - 1 do begin    ci:=FClasses.GetClassInfo(i);    s:=GetTypeInfoVar(ci.Def);    if (s = '') or (ci.IsCommonClass) then      s:='nil'    else      s:='@' + s;    if ci.Funcs.Count = 0 then      ss:='nil'    else      ss:=Format('@_%sNativeMethods', [GetClassPrefix(ci.Def, FClasses.GetClassName(i))]);    fn:='';    if ci.Def <> nil then      if ci.Def.DefType in [dtSet, dtEnum] then        fn:=', ''Value'', ''I''';    Fps.WriteLn(Format('if not _Reg(''%s'', %s, %d, %s%s) then exit;',                       [GetJavaClassPath(ci.Def, FClasses.GetClassName(i)), ss, ci.Funcs.Count, s, fn]));  end;  Fps.WriteLn('Result:=JNI_VERSION_1_6;');  Fps.DecI;  Fps.WriteLn('end;');  Fps.WriteLn;  Fps.WriteLn('exports JNI_OnLoad;');end;procedure TWriter.WriteRecordSizes;var  i, j: integer;  s: string;begin  Fps.WriteLn;  Fps.WriteLn('function _GetRecordSize(env: PJNIEnv; jobj: jobject; index: jint): jint;' + JniCaliing);  if FRecords.Count > 0 then begin    Fps.WriteLn(Format('const sizes: array[0..%d] of longint =', [FRecords.Count - 1]));    Fps.IncI;    s:='(';    j:=0;    for i:=0 to FRecords.Count - 1 do begin      if i > 0 then        s:=s + ',';      Inc(j);      if j > 20 then begin        Fps.WriteLn(s);        s:='';        j:=0;      end;      s:=s + IntToStr(TClassDef(FRecords[i]).Size);    end;    Fps.WriteLn(s + ');');    Fps.DecI;  end;  Fps.WriteLn('begin');  if FRecords.Count > 0 then    s:='sizes[index]'  else    s:='0';  Fps.WriteLn('Result:=' + s + ';', 1);  Fps.WriteLn('end;');end;procedure TWriter.WriteClassTable;var  i: integer;  s,ss: string;begin  Fps.WriteLn;  Fps.WriteLn('function _GetClassRef(env: PJNIEnv; jobj: jobject; index: jint): jlong;' + JniCaliing);  if FRealClasses.Count > 0 then begin    Fps.WriteLn(Format('const cls: array[0..%d] of TClass =', [FRealClasses.Count - 1]));    Fps.IncI;    s:='(';    for i:=0 to FRealClasses.Count - 1 do begin      if i > 0 then        s:=s + ',';      if Length(s) > 100 then begin        Fps.WriteLn(s);        s:='';      end;      with TClassDef(FRealClasses[i]) do        ss:=Parent.Name + '.' + Name;      s:=s + ss;    end;    Fps.WriteLn(s + ');');    Fps.DecI;  end;  Fps.WriteLn('begin');  if FRealClasses.Count > 0 then    s:='cls[index]'  else    s:='nil';  Fps.WriteLn('Result:=-jlong(ptruint(pointer(' + s + ')));', 1);  Fps.WriteLn('end;');end;function TWriter.JniToPasType(d: TDef; const v: string; CheckNil: boolean): string;var  n: string;begin  Result:=v;  if d = nil then    exit;  case d.DefType of    dtType:      with TTypeDef(d) do        case BasicType of          btString, btWideString:            begin              Result:=Format('_StringFromJString(_env, %s)', [Result]);              if BasicType <> btWideString then                Result:=Format('%s(%s)', [d.Name, Result]);            end;          btBoolean:            Result:=Format('LongBool(%s)', [Result]);          btChar:            Result:=Format('char(widechar(%s))', [Result]);          btWideChar:            Result:=Format('widechar(%s)', [Result]);          btGuid:            Result:=Format('StringToGUID(ansistring(_StringFromJString(_env, %s)))', [Result]);          else            Result:=Format('%s.%s(%s)', [d.Parent.Name, d.Name, Result]);        end;    dtClass:      begin        if TClassDef(d).CType = ctRecord then          n:='True'        else          if CheckNil then            n:='True'          else            n:='False';        Result:=Format('_GetPasObj(_env, %s, %s, %s)', [Result, GetTypeInfoVar(d), n]);        if TClassDef(d).CType in [ctObject, ctRecord] then          Result:=Result + '^';        Result:=Format('%s.%s(%s)', [d.Parent.Name, d.Name, Result]);      end;    dtProcType:      Result:=Format('%sGetHandler(_env, %s, %s)', [GetClassPrefix(d), Result, GetTypeInfoVar(d)]);    dtEnum:      Result:=Format('%s.%s(_GetIntObjValue(_env, %s, %s))', [d.Parent.Name, d.Name, Result, GetTypeInfoVar(d)]);    dtSet:      Result:=Format('%s.%s(%s(_GetIntObjValue(_env, %s, %s)))', [d.Parent.Name, d.Name, GetPasIntType(TSetDef(d).Size), Result, GetTypeInfoVar(d)]);    dtPointer:      begin        if TPointerDef(d).IsObjPtr then          Result:=Format('%s.%s(_GetPasObj(_env, %s, %s, False))', [d.Parent.Name, d.Name, Result, GetTypeInfoVar(d)])        else          Result:=Format('pointer(ptruint(%s))', [Result]);      end;    dtClassRef:      begin        Result:=Format('_GetClass(_env, %s, %s)', [Result, GetTypeInfoVar(d)]);        Result:=Format('%s.%s(%s)', [d.Parent.Name, d.Name, Result]);      end;  end;end;function TWriter.PasToJniType(d: TDef; const v: string): string;begin  Result:=v;  if d = nil then    exit;  case d.DefType of    dtType:      with TTypeDef(d) do        case BasicType of          btString, btWideString:            Result:=Format('_StringToJString(_env, _JNIString(%s))', [Result]);          btBoolean:            Result:=Format('(jboolean(%s) and 1)', [Result]);          btChar:            Result:=Format('jchar(widechar(%s))', [Result]);          btWideChar:            Result:=Format('jchar(%s)', [Result]);          btEnum:            Result:=Format('jint(%s)', [Result]);          btGuid:            Result:=Format('_StringToJString(_env, _JNIString(GUIDToString(%s)))', [Result]);        end;    dtClass:      case TClassDef(d).CType of        ctObject, ctRecord:          Result:=Format('_%s_CreateObj(_env, %s)', [GetClassPrefix(d), Result]);        ctInterface:          Result:=Format('_CreateJavaObj(_env, pointer(%s), %s)', [Result, GetTypeInfoVar(d)]);        else          Result:=Format('_CreateJavaObj(_env, %s, %s)', [Result, GetTypeInfoVar(d)]);      end;    dtProcType:      Result:=Format('_CreateMethodPtrObject(_env, TMethod(%s), %s)', [Result, GetTypeInfoVar(d)]);    dtEnum:      Result:=Format('_CreateIntObj(_env, longint(%s), %s)', [Result, GetTypeInfoVar(d)]);    dtSet:      Result:=Format('_CreateIntObj(_env, %s(%s), %s)', [GetPasIntType(TSetDef(d).Size), Result, GetTypeInfoVar(d)]);    dtPointer:      if TPointerDef(d).IsObjPtr then        Result:=Format('_CreateJavaObj(_env, %s, %s)', [Result, GetTypeInfoVar(d)])      else        Result:=Format('ptruint(pointer(%s))', [Result]);    dtClassRef:      Result:=Format('_CreateJavaObj(_env, -jlong(ptruint(pointer(%s))), %s)', [Result, GetTypeInfoVar(d)])  end;end;function TWriter.GetTypeInfoVar(ClassDef: TDef): string;begin  if ClassDef.DefType = dtUnit then    Result:=''  else    if ClassDef.DefType = dtType then      Result:='_Java_' + JavaType[TTypeDef(ClassDef).BasicType] + '_Info'    else      Result:='_JNI_' + ClassDef.Parent.Name + '_' + ClassDef.Name + '_Info';end;function TWriter.GetClassPrefix(ClassDef: TDef; const AClassName: string): string;begin  if AClassName = '' then    Result:=ClassDef.Name  else    Result:=AClassName;  Result:=Result + '_';  if ClassDef.DefType <> dtUnit then    Result:=ClassDef.Parent.Name + '_' + Result;  Result:='JNI_' + Result;end;function TWriter.IsJavaSimpleType(d: TDef): boolean;begin  Result:=d <> nil;  if Result then    case d.DefType of      dtType:        Result:=Length(JNITypeSig[TTypeDef(d).BasicType]) = 1;      dtPointer:        Result:=not TPointerDef(d).IsObjPtr;      else        Result:=False;    end;end;function TWriter.IsJavaVarParam(ParamDef: TVarDef): boolean;begin  with ParamDef do    Result:=VarOpt * [voVar, voOut] <> [];end;function TWriter.GetProcDeclaration(d: TProcDef; const ProcName: string; FullTypeNames: boolean; InternalParaNames: boolean): string;var  s, ss: string;  j: integer;begin  with d do begin    s:='';    for j:=0 to Count - 1 do      with TVarDef(Items[j]) do begin        if DefType <> dtParam then          continue;        if s <> '' then          s:=s + '; ';        if voVar in VarOpt then          s:=s + 'var '        else        if voOut in VarOpt then          s:=s + 'out '        else        if voConst in VarOpt then          s:=s + 'const ';        if InternalParaNames then          s:=s + Name        else          s:=s + AliasName;        s:=s + ': ' + GetPasType(VarType, FullTypeNames);      end;    if s <> '' then      s:='(' + s + ')';    case ProcType of      ptConstructor:        ss:='constructor';      ptDestructor:        ss:='destructor';      ptProcedure:        ss:='procedure';      ptFunction:        ss:='function';      else        ss:='';    end;    if ProcType in [ptConstructor, ptFunction] then      s:=s + ': ' + GetPasType(ReturnType, FullTypeNames);    ss:=ss + ' ';    if ProcName <> '' then      ss:=ss + ProcName    else      ss:=ss + Name;    Result:=ss + s;  end;end;function TWriter.GetJavaProcDeclaration(d: TProcDef; const ProcName: string): string;var  s, ss: string;  j: integer;  vd: TVarDef;begin  with d do begin    if ProcName <> '' then      ss:=ProcName    else      ss:=AliasName;    ss:=DefToJavaType(ReturnType) + ' ' + ss + '(';    s:='';    for j:=0 to Count - 1 do begin      vd:=TVarDef(Items[j]);      if vd.DefType <> dtParam then        continue;      with vd do begin        if (VarType <> nil) and (VarType.DefType = dtJniEnv) then          continue;        if s <> '' then          s:=s + ', ';        s:=s + DefToJavaType(VarType);        if IsJavaVarParam(vd) then          s:=s + '[]';        s:=s + ' ' + AliasName;      end;    end;    ss:=ss + s + ')';  end;  Result:=ss;end;function TWriter.GetJniFuncType(d: TDef): string;begin  if IsJavaSimpleType(d) then begin    if d.DefType = dtPointer then      Result:='Long'    else begin      Result:=JavaType[TTypeDef(d).BasicType];      Result[1]:=UpCase(Result[1]);    end;  end  else    Result:='Object';end;function TWriter.GetJavaClassName(cls: TDef; it: TDef): string;begin  Result:=cls.AliasName;  if (cls.DefType <> dtClass) or ((it <> nil) and not (it.DefType in ReplDefs)) then    exit;  with TClassDef(cls) do begin    if not (HasReplacedItems or ImplementsReplacedItems) then      exit;    if ImplementsReplacedItems and not HasReplacedItems then      exit;    if it <> nil then      with TReplDef(it) do begin        if (it.DefType = dtProc) and (TProcDef(it).ProcType = ptConstructor) then          exit;        if IsReplaced or IsReplImpl then          exit;      end;  end;  Result:='__' + Result;end;procedure TWriter.RegisterPseudoClass(d: TDef);var  ci: TClassInfo;begin  if FClasses.IndexOf(d.Name, d) < 0 then begin    ci:=TClassInfo.Create;    ci.Def:=d;    FClasses.Add(d.Name, d, ci);  end;end;function TWriter.GetPasIntType(Size: integer): string;begin  case Size of    1: Result:='byte';    2: Result:='word';    else       Result:='cardinal';  end;end;function TWriter.GetPasType(d: TDef; FullName: boolean): string;begin  Result:=d.Name;  if FullName and (d.DefType <> dtType) then    Result:=d.Parent.Name + '.' + Result;end;function TWriter.AddCustomProc(ParentDef: TDef; const JniName, Name: string; RetType: TBasicType; const Params: array of TBasicType): TProcDef;var  i: integer;  vd: TVarDef;begin  Result:=TProcDef.Create(ParentDef, dtProc);  Result.Name:=JniName;  Result.AliasName:=Name;  if RetType = btVoid then    Result.ProcType:=ptProcedure  else    Result.ProcType:=ptFunction;  for i:=0 to High(Params) do begin    vd:=TVarDef.Create(Result, dtParam);    vd.Name:=Format('p%d', [i + 1]);    vd.VarType:=TTypeDef.Create(vd, dtType);    TTypeDef(vd.VarType).BasicType:=Params[i];  end;  Result.ReturnType:=TTypeDef.Create(ParentDef, dtType);  TTypeDef(Result.ReturnType).BasicType:=RetType;end;procedure TWriter.AddNativeMethod(ParentDef: TDef; const JniName, Name, Signature: string);var  i: integer;  ci: TClassInfo;  pi: TProcInfo;begin  pi:=TProcInfo.Create;  pi.Name:=Name;  pi.JniName:=JniName;  pi.JniSignature:=Signature;  i:=FClasses.IndexOf(ParentDef.AliasName, ParentDef);  if i < 0 then begin    ci:=TClassInfo.Create;    ci.Def:=ParentDef;    i:=FClasses.Add(ParentDef.AliasName, ParentDef, ci);  end;  FClasses.GetClassInfo(i).Funcs.Add(pi);end;function TWriter.GetProcSignature(d: TProcDef): string;var  j: integer;  vd: TVarDef;begin  Result:='(';  for j:=0 to d.Count - 1 do begin    vd:=TVarDef(d[j]);    if vd.DefType <> dtParam then      continue;    with vd do begin      if (VarType <> nil) and (VarType.DefType = dtJniEnv) then        continue;      if IsJavaVarParam(vd) then        Result:=Result + '[';      Result:=Result + DefToJniSig(VarType);    end;  end;  Result:=Result + ')' + DefToJniSig(d.ReturnType);end;procedure TWriter.EHandlerStart;begin  Fps.WriteLn('try');  Fps.IncI;end;procedure TWriter.EHandlerEnd(const EnvVarName: string; const ExtraCode: string);begin  Fps.WriteLn('except', -1);  Fps.WriteLn(Format('_HandleJNIException(%s);', [EnvVarName]));  if ExtraCode <> '' then    Fps.WriteLn(ExtraCode);  Fps.DecI;  Fps.WriteLn('end;');end;procedure TWriter.UpdateUsedUnits(u: TUnitDef);  procedure _CheckDef(d: TDef);  begin    if (d = nil) or not d.IsUsed then      exit;    d:=d.Parent;    if (d <> nil) and (d.DefType = dtUnit) then      with TUnitDef(d) do        if not IsUnitUsed and IsUsed then          IsUnitUsed:=True;  end;  procedure _ScanDef(def: TDef);  var    i: integer;    d: TDef;  begin    for i:=0 to def.Count - 1 do begin      d:=def[i];      if not d.IsUsed then        continue;      _CheckDef(d.GetRefDef);      _CheckDef(d.GetRefDef2);      _ScanDef(d);    end;  end;var  i: integer;begin  for i:=0 to High(u.UsedUnits) do    u.UsedUnits[i].IsUnitUsed:=False;  _ScanDef(u);end;procedure TWriter.WriteClassInfoVar(d: TDef);begin  Fps.WriteLn;  Fps.WriteLn(Format('var %s: _TJavaClassInfo;', [GetTypeInfoVar(d)]));end;procedure TWriter.WriteComment(d: TDef; const AType: string);begin  Fps.WriteLn;  Fps.WriteLn(Format('{ %s }', [d.Name]));  Fjs.WriteLn(Format('/* %s */', [Trim(AType + ' ' + d.Name)]));{$ifdef DEBUG}  Fjs.WriteLn(Format('/* Ref count: %d */', [d.RefCnt]));{$endif}end;{procedure TWriter.AddCustomProc(ParentDef: TDef; const JniName, Name: string; RetType: TBasicType; const Params: array of TBasicType);var  i: integer;  ci: TClassInfo;  pi: TProcInfo;begin  pi:=TProcInfo.Create;  pi.Name:=Name;  pi.JniName:=JniName;  pi.JniSignature:='(';  for i:=0 to High(Params) do    pi.JniSignature:=pi.JniSignature + JNITypeSig[Params[i]];  pi.JniSignature:=pi.JniSignature + ')';  pi.JniSignature:=pi.JniSignature + JNITypeSig[RetType];  i:=FClasses.IndexOf(ParentDef.Name);  if i < 0 then begin    ci:=TClassInfo.Create;    ci.Def:=ParentDef;    i:=FClasses.AddObject(ParentDef.Name, ci);  end;  TClassInfo(FClasses.Objects[i]).Funcs.Add(pi);end;}constructor TWriter.Create;var  i: integer;begin  Units:=TStringList.Create;  FClasses:=TClassList.Create;  JavaPackage:='pas';  IncludeList:=TStringList.Create;  IncludeList.Duplicates:=dupIgnore;  ExcludeList:=TStringList.Create;  ExcludeList.Duplicates:=dupIgnore;  for i:=Low(ExcludeStd) to High(ExcludeStd) do    ExcludeList.Add(ExcludeStd[i]);  for i:=Low(ExcludeDelphi7) to High(ExcludeDelphi7) do    ExcludeList.Add(ExcludeDelphi7[i]);  FThisUnit:=TUnitDef.Create(nil, dtUnit);  FRecords:=TObjectList.Create(False);  FRealClasses:=TObjectList.Create(False);  LibAutoLoad:=True;end;function DoCanUseDef(def, refdef: TDef): boolean;begin  Result:=True;  if (def.DefType = dtArray) and (refdef is TVarDef) then begin    // Arrays are supported only for variables, fields, properties and constants    Result:=refdef.DefType in [dtVar, dtProp, dtField, dtConst];  end;end;destructor TWriter.Destroy;var  i: integer;begin  for i:=0 to FClasses.Count - 1 do    FClasses.Objects[i].Free;  FClasses.Free;  Units.Free;  IncludeList.Free;  ExcludeList.Free;  FThisUnit.Free;  FRecords.Free;  FRealClasses.Free;  inherited Destroy;end;procedure TWriter.ProcessUnits;  procedure _ExcludeClasses(u: TDef; AAncestorClass: TClassDef);  var    i: integer;    d: TDef;    s: string;    excl: boolean;  begin    for i:=0 to u.Count - 1 do begin      d:=u[i];      if d.DefType = dtClass then begin        s:=u.Name + '.' + d.Name;        if AAncestorClass = nil then begin          excl:=DoCheckItem(s) = crExclude;          if not excl and (TClassDef(d).AncestorClass <> nil) then            with TClassDef(d).AncestorClass do              excl:=DoCheckItem(Parent.Name + '.' + Name) = crExclude;        end        else          excl:=TClassDef(d).AncestorClass = AAncestorClass;        if excl then begin          d.SetNotUsed;          ExcludeList.Add(s);          _ExcludeClasses(u, TClassDef(d));        end;      end;    end;  end;var  p: TPPUParser;  i: integer;  s, ss: string;  d: TDef;begin  if Units.Count = 0 then    raise Exception.Create('No unit name specified.');  if (OutPath <> '') and not DirectoryExists(OutPath) then    raise Exception.CreateFmt('Output path "%s" does not exist.', [OutPath]);  if (JavaOutPath <> '') and not DirectoryExists(JavaOutPath) then    raise Exception.CreateFmt('Output path "%s" does not exist.', [JavaOutPath]);  if LibName = '' then    LibName:=AnsiLowerCase(ChangeFileExt(Units[0], '')) + 'jni';  for i:=0 to IncludeList.Count - 1 do    IncludeList[i]:=Trim(IncludeList[i]);  IncludeList.Sorted:=True;  for i:=0 to ExcludeList.Count - 1 do    ExcludeList[i]:=Trim(ExcludeList[i]);  ExcludeList.Sorted:=True;  FThisUnit.Name:=LibName;  FThisUnit.AliasName:='system';  p:=TPPUParser.Create(SearchPath);  try    p.OnCheckItem:=@DoCheckItem;    OnCanUseDef:=@DoCanUseDef;    for i:=0 to Units.Count - 1 do      IncludeList.Add(ChangeFileExt(ExtractFileName(Units[i]), ''));    for i:=0 to Units.Count - 1 do      p.Parse(ChangeFileExt(ExtractFileName(Units[i]), ''));    if OutPath <> '' then      OutPath:=IncludeTrailingPathDelimiter(OutPath);    if JavaOutPath <> '' then      JavaOutPath:=IncludeTrailingPathDelimiter(JavaOutPath);    FPkgDir:=JavaOutPath + StringReplace(JavaPackage, '.', DirectorySeparator, [rfReplaceAll]);    ForceDirectories(FPkgDir);    Fps:=TTextOutStream.Create(OutPath + LibName + '.pas', fmCreate);    WriteFileComment(Fps);    Fps.WriteLn('library '+ LibName + ';');    Fps.WriteLn('{$ifdef fpc} {$mode objfpc} {$H+} {$endif}');    Fps.WriteLn;    Fps.WriteLn('uses');    Fps.WriteLn('{$ifdef unix} cthreads, {$endif}', 1);    s:='';    for i:=0 to p.Units.Count - 1 do begin      ProcessRules(p.Units[i]);      ss:=LowerCase(p.Units[i].Name);      if (ss ='system') or (ss = 'objpas') or (ss = 'sysutils') or (ss = 'syncobjs') or (ss = 'jni')         or (ss = 'cthreads') or (ss = 'windows')      then        continue;      if s <> '' then        s:=s + ', ';      if Length(s) >= 100 then begin        Fps.WriteLn(s, 1);        s:='';      end;      s:=s + p.Units[i].Name;    end;    if s <> '' then      Fps.WriteLn(s + ',', 1);    Fps.WriteLn('{$ifndef FPC} Windows, {$endif} SysUtils, SyncObjs, jni;', 1);    // Types    Fps.WriteLn;    Fps.WriteLn('type');    Fps.IncI;    Fps.WriteLn('_JNIString = {$ifdef FPC} unicodestring {$else} widestring {$endif};');    Fps.WriteLn('{$ifndef FPC} ptruint = cardinal; {$endif}');    Fps.WriteLn;    Fps.WriteLn('_TJavaClassInfo = record');    Fps.WriteLn('ClassRef: JClass;', 1);    Fps.WriteLn('ConstrId: JMethodId;', 1);    Fps.WriteLn('ObjFieldId: JFieldId;', 1);    Fps.WriteLn('end;');    Fps.WriteLn('_PJavaClassInfo = ^_TJavaClassInfo;');    Fps.DecI;    Fps.WriteLn;    d:=TtypeDef.Create(nil, dtType);    TtypeDef(d).BasicType:=btString;    Fps.WriteLn(Format('var %s: _TJavaClassInfo;', [GetTypeInfoVar(d)]));    d.Free;    // Support functions    Fps.WriteLn;    Fps.WriteLn('function _StringFromJString(env: PJNIEnv; s: jstring): _JNIString;');    Fps.WriteLn('var');    Fps.WriteLn('p: PJChar;', 1);    Fps.WriteLn('c: JBoolean;', 1);    Fps.WriteLn('begin');    Fps.WriteLn('if s = nil then begin', 1);    Fps.WriteLn('Result:='''';', 2);    Fps.WriteLn('exit;', 2);    Fps.WriteLn('end;', 1);    Fps.WriteLn('p:=env^^.GetStringChars(env, s, c);', 1);    Fps.WriteLn('SetString(Result, PWideChar(p), env^^.GetStringLength(env, s));', 1);    Fps.WriteLn('env^^.ReleaseStringChars(env, s, p);', 1);    Fps.WriteLn('end;');    Fps.WriteLn;    Fps.WriteLn('function _StringToJString(env: PJNIEnv; const s: _JNIString): jstring;');    Fps.WriteLn('begin');    Fps.WriteLn('Result:=env^^.NewString(env, PJChar(PWideChar(s)), Length(s));', 1);    Fps.WriteLn('end;');    Fps.WriteLn;    Fps.WriteLn('function _CreateJavaObj(env: PJNIEnv; PasObj: jlong; const ci: _TJavaClassInfo; cleanup: boolean = True): jobject; overload;');    Fps.WriteLn('var v: array [0..1] of jvalue;');    Fps.WriteLn('begin');    Fps.IncI;    Fps.WriteLn('Result:=nil;');    Fps.WriteLn('if PasObj = 0 then exit;');    Fps.WriteLn('v[0].J:=PasObj;');    Fps.WriteLn('if ci.ConstrId = nil then begin');    Fps.WriteLn('Result:=env^^.AllocObject(env, ci.ClassRef);', 1);    Fps.WriteLn('if Result = nil then exit;', 1);    Fps.WriteLn('env^^.SetLongField(env, Result, ci.ObjFieldId, v[0].J);', 1);    Fps.WriteLn('end else begin');    Fps.WriteLn('v[1].Z:=byte(cleanup) and 1;', 1);    Fps.WriteLn('Result:=env^^.NewObjectA(env, ci.ClassRef, ci.ConstrId, @v);', 1);    Fps.WriteLn('end;');    Fps.DecI;    Fps.WriteLn('end;');    Fps.WriteLn;    Fps.WriteLn('function _CreateJavaObj(env: PJNIEnv; PasObj: pointer; const ci: _TJavaClassInfo; cleanup: boolean = True): jobject; overload;');    Fps.WriteLn('begin');    Fps.WriteLn('Result:=_CreateJavaObj(env, jlong(ptruint(PasObj)), ci, cleanup)', 1);    Fps.WriteLn('end;');    Fps.WriteLn;    Fps.WriteLn('function _GetPasObj(env: PJNIEnv; jobj: jobject; const ci: _TJavaClassInfo; CheckNil: boolean): pointer;');    Fps.WriteLn('var pasobj: jlong;');    Fps.WriteLn('begin');    Fps.IncI;    Fps.WriteLn('if jobj <> nil then');    Fps.WriteLn('pasobj:=env^^.GetLongField(env, jobj, ci.ObjFieldId)', 1);    Fps.WriteLn('else');    Fps.WriteLn('pasobj:=0;', 1);    Fps.WriteLn('if CheckNil and (pasobj <= 0) then');    Fps.WriteLn('raise Exception.Create(''Attempt to access a released Pascal object.'');', 1);    Fps.WriteLn('Result:=pointer(ptruint(pasobj));');    Fps.DecI;    Fps.WriteLn('end;');    Fps.WriteLn;    Fps.WriteLn('function _GetClass(env: PJNIEnv; jobj: jobject; const ci: _TJavaClassInfo): TClass;');    Fps.WriteLn('var pasobj: jlong;');    Fps.WriteLn('begin');    Fps.IncI;    Fps.WriteLn('if jobj <> nil then');    Fps.WriteLn('pasobj:=env^^.GetLongField(env, jobj, ci.ObjFieldId)', 1);    Fps.WriteLn('else');    Fps.WriteLn('pasobj:=0;', 1);    Fps.WriteLn('if pasobj > 0 then');    Fps.WriteLn('Result:=TObject(ptruint(pasobj)).ClassType', 1);    Fps.WriteLn('else');    Fps.WriteLn('Result:=TClass(ptruint(-pasobj));', 1);    Fps.DecI;    Fps.WriteLn('end;');    Fps.WriteLn;    Fps.WriteLn('procedure _HandleJNIException(env: PJNIEnv);');    Fps.WriteLn('begin');    Fps.WriteLn('if env^^.ExceptionCheck(env) <> 0 then exit;', 1);    if p.OnExceptionProc <> nil then begin      Fps.WriteLn(Format('%s.%s;', [p.OnExceptionProc.Parent.Name, p.OnExceptionProc.Name]), 1);      p.OnExceptionProc.SetNotUsed;    end;    Fps.WriteLn('env^^.ThrowNew(env, env^^.FindClass(env, ''java/lang/Exception''), PAnsiChar(Utf8Encode(Exception(ExceptObject).Message)));', 1);    Fps.WriteLn('end;');    Fps.WriteLn;    Fps.WriteLn('procedure _HandleJavaException(env: PJNIEnv);');    Fps.WriteLn('begin');    Fps.WriteLn('if env^^.ExceptionCheck(env) <> 0 then raise Exception.Create(''Java exception.'');', 1);    Fps.WriteLn('end;');    Fps.WriteLn;    Fps.WriteLn('procedure _RaiseVarParamException(const VarName: string);');    Fps.WriteLn('begin');    Fps.WriteLn('raise Exception.CreateFmt(''An array with only single element must be passed as parameter "%s".'', [VarName]);', 1);    Fps.WriteLn('end;');    // Public support functions    Fps.WriteLn;    Fps.WriteLn('function _AllocMemory(env: PJNIEnv; jobj: jobject; size: jint): jlong;' + JniCaliing);    Fps.WriteLn('var p: pointer;');    Fps.WriteLn('begin');    Fps.WriteLn('GetMem(p, size);', 1);    Fps.WriteLn('FillChar(p^, size, 0);', 1);    Fps.WriteLn('Result:=ptruint(p);', 1);    Fps.WriteLn('end;');    Fps.WriteLn;    Fps.WriteLn('function _GetMemoryAsArray(env: PJNIEnv; jobj: jobject; SrcBuf: jlong; BufSize: jint): jarray;' + JniCaliing);    Fps.WriteLn('begin');    Fps.WriteLn('Result:=env^^.NewByteArray(env, BufSize);', 1);    Fps.WriteLn('env^^.SetByteArrayRegion(env, Result, 0, BufSize, pointer(ptruint(SrcBuf)));', 1);    Fps.WriteLn('end;');    Fps.WriteLn;    Fps.WriteLn('procedure _SetMemoryFromArray(env: PJNIEnv; jobj: jobject; DstBuf: jlong; SrcArray: jarray);' + JniCaliing);    Fps.WriteLn('begin');    Fps.WriteLn('env^^.GetByteArrayRegion(env, SrcArray, 0, env^^.GetArrayLength(env, SrcArray), pointer(ptruint(DstBuf)));', 1);    Fps.WriteLn('end;');    // Set support    Fps.WriteLn;    Fps.WriteLn('function _GetIntObjValue(env: PJNIEnv; jobj: jobject; const ci: _TJavaClassInfo): longint;');    Fps.WriteLn('begin');    Fps.IncI;    Fps.WriteLn('if jobj = nil then raise Exception.Create(''Attempt to access a NULL set.'');');    Fps.WriteLn('Result:=env^^.GetIntField(env, jobj, ci.ObjFieldId);');    Fps.DecI;    Fps.WriteLn('end;');    Fps.WriteLn;    Fps.WriteLn('function _CreateIntObj(env: PJNIEnv; Value: longint; const ci: _TJavaClassInfo): jobject;');    Fps.WriteLn('begin');    Fps.IncI;    Fps.WriteLn('Result:=nil;');    Fps.WriteLn('Result:=env^^.AllocObject(env, ci.ClassRef);');    Fps.WriteLn('if Result = nil then exit;');    Fps.WriteLn('env^^.SetIntField(env, Result, ci.ObjFieldId, Value);');    Fps.DecI;    Fps.WriteLn('end;');    // Preprocess units    for i:=0 to p.Units.Count - 1 do begin      if AnsiCompareText(p.Units[i].Name, 'system') <> 0 then        _ExcludeClasses(p.Units[i], nil);    end;    // Write units    for i:=0 to p.Units.Count - 1 do      with TUnitDef(p.Units[i]) do begin        WriteUnit(TUnitDef(p.Units[i]));      end;    WriteRecordSizes;    WriteClassTable;    WriteOnLoad;    Fps.WriteLn;    Fps.WriteLn('procedure ___doexit;');    Fps.WriteLn('begin');    Fps.WriteLn('_MethodPointersCS.Free;', 1);    Fps.WriteLn('end;');    Fps.WriteLn;    Fps.WriteLn('begin');    Fps.WriteLn('ExitProc:=@___doexit;', 1);    Fps.WriteLn('IsMultiThread:=True;', 1);    Fps.WriteLn('_MethodPointersCS:=TCriticalSection.Create;', 1);    Fps.WriteLn('end.');  finally    Fps.Free;    p.Free;  end;end;end.
 |