javaembed.cpp 188 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085208620872088208920902091209220932094209520962097209820992100210121022103210421052106210721082109211021112112211321142115211621172118211921202121212221232124212521262127212821292130213121322133213421352136213721382139214021412142214321442145214621472148214921502151215221532154215521562157215821592160216121622163216421652166216721682169217021712172217321742175217621772178217921802181218221832184218521862187218821892190219121922193219421952196219721982199220022012202220322042205220622072208220922102211221222132214221522162217221822192220222122222223222422252226222722282229223022312232223322342235223622372238223922402241224222432244224522462247224822492250225122522253225422552256225722582259226022612262226322642265226622672268226922702271227222732274227522762277227822792280228122822283228422852286228722882289229022912292229322942295229622972298229923002301230223032304230523062307230823092310231123122313231423152316231723182319232023212322232323242325232623272328232923302331233223332334233523362337233823392340234123422343234423452346234723482349235023512352235323542355235623572358235923602361236223632364236523662367236823692370237123722373237423752376237723782379238023812382238323842385238623872388238923902391239223932394239523962397239823992400240124022403240424052406240724082409241024112412241324142415241624172418241924202421242224232424242524262427242824292430243124322433243424352436243724382439244024412442244324442445244624472448244924502451245224532454245524562457245824592460246124622463246424652466246724682469247024712472247324742475247624772478247924802481248224832484248524862487248824892490249124922493249424952496249724982499250025012502250325042505250625072508250925102511251225132514251525162517251825192520252125222523252425252526252725282529253025312532253325342535253625372538253925402541254225432544254525462547254825492550255125522553255425552556255725582559256025612562256325642565256625672568256925702571257225732574257525762577257825792580258125822583258425852586258725882589259025912592259325942595259625972598259926002601260226032604260526062607260826092610261126122613261426152616261726182619262026212622262326242625262626272628262926302631263226332634263526362637263826392640264126422643264426452646264726482649265026512652265326542655265626572658265926602661266226632664266526662667266826692670267126722673267426752676267726782679268026812682268326842685268626872688268926902691269226932694269526962697269826992700270127022703270427052706270727082709271027112712271327142715271627172718271927202721272227232724272527262727272827292730273127322733273427352736273727382739274027412742274327442745274627472748274927502751275227532754275527562757275827592760276127622763276427652766276727682769277027712772277327742775277627772778277927802781278227832784278527862787278827892790279127922793279427952796279727982799280028012802280328042805280628072808280928102811281228132814281528162817281828192820282128222823282428252826282728282829283028312832283328342835283628372838283928402841284228432844284528462847284828492850285128522853285428552856285728582859286028612862286328642865286628672868286928702871287228732874287528762877287828792880288128822883288428852886288728882889289028912892289328942895289628972898289929002901290229032904290529062907290829092910291129122913291429152916291729182919292029212922292329242925292629272928292929302931293229332934293529362937293829392940294129422943294429452946294729482949295029512952295329542955295629572958295929602961296229632964296529662967296829692970297129722973297429752976297729782979298029812982298329842985298629872988298929902991299229932994299529962997299829993000300130023003300430053006300730083009301030113012301330143015301630173018301930203021302230233024302530263027302830293030303130323033303430353036303730383039304030413042304330443045304630473048304930503051305230533054305530563057305830593060306130623063306430653066306730683069307030713072307330743075307630773078307930803081308230833084308530863087308830893090309130923093309430953096309730983099310031013102310331043105310631073108310931103111311231133114311531163117311831193120312131223123312431253126312731283129313031313132313331343135313631373138313931403141314231433144314531463147314831493150315131523153315431553156315731583159316031613162316331643165316631673168316931703171317231733174317531763177317831793180318131823183318431853186318731883189319031913192319331943195319631973198319932003201320232033204320532063207320832093210321132123213321432153216321732183219322032213222322332243225322632273228322932303231323232333234323532363237323832393240324132423243324432453246324732483249325032513252325332543255325632573258325932603261326232633264326532663267326832693270327132723273327432753276327732783279328032813282328332843285328632873288328932903291329232933294329532963297329832993300330133023303330433053306330733083309331033113312331333143315331633173318331933203321332233233324332533263327332833293330333133323333333433353336333733383339334033413342334333443345334633473348334933503351335233533354335533563357335833593360336133623363336433653366336733683369337033713372337333743375337633773378337933803381338233833384338533863387338833893390339133923393339433953396339733983399340034013402340334043405340634073408340934103411341234133414341534163417341834193420342134223423342434253426342734283429343034313432343334343435343634373438343934403441344234433444344534463447344834493450345134523453345434553456345734583459346034613462346334643465346634673468346934703471347234733474347534763477347834793480348134823483348434853486348734883489349034913492349334943495349634973498349935003501350235033504350535063507350835093510351135123513351435153516351735183519352035213522352335243525352635273528352935303531353235333534353535363537353835393540354135423543354435453546354735483549355035513552355335543555355635573558355935603561356235633564356535663567356835693570357135723573357435753576357735783579358035813582358335843585358635873588358935903591359235933594359535963597359835993600360136023603360436053606360736083609361036113612361336143615361636173618361936203621362236233624362536263627362836293630363136323633363436353636363736383639364036413642364336443645364636473648364936503651365236533654365536563657365836593660366136623663366436653666366736683669367036713672367336743675367636773678367936803681368236833684368536863687368836893690369136923693369436953696369736983699370037013702370337043705370637073708370937103711371237133714371537163717371837193720372137223723372437253726372737283729373037313732373337343735373637373738373937403741374237433744374537463747374837493750375137523753375437553756375737583759376037613762376337643765376637673768376937703771377237733774377537763777377837793780378137823783378437853786378737883789379037913792379337943795379637973798379938003801380238033804380538063807380838093810381138123813381438153816381738183819382038213822382338243825382638273828382938303831383238333834383538363837383838393840384138423843384438453846384738483849385038513852385338543855385638573858385938603861386238633864386538663867386838693870387138723873387438753876387738783879388038813882388338843885388638873888388938903891389238933894389538963897389838993900390139023903390439053906390739083909391039113912391339143915391639173918391939203921392239233924392539263927392839293930393139323933393439353936393739383939394039413942394339443945394639473948394939503951395239533954395539563957395839593960396139623963396439653966396739683969397039713972397339743975397639773978397939803981398239833984398539863987398839893990399139923993399439953996399739983999400040014002400340044005400640074008400940104011401240134014401540164017401840194020402140224023402440254026402740284029403040314032403340344035403640374038403940404041404240434044404540464047404840494050405140524053405440554056405740584059406040614062406340644065406640674068406940704071407240734074407540764077407840794080408140824083408440854086408740884089409040914092409340944095409640974098409941004101410241034104410541064107410841094110411141124113411441154116411741184119412041214122412341244125412641274128412941304131413241334134413541364137413841394140414141424143414441454146414741484149415041514152415341544155415641574158415941604161416241634164416541664167416841694170417141724173417441754176417741784179418041814182418341844185418641874188418941904191419241934194419541964197419841994200420142024203420442054206420742084209421042114212421342144215421642174218421942204221422242234224422542264227422842294230423142324233423442354236423742384239424042414242424342444245424642474248424942504251425242534254425542564257425842594260426142624263426442654266426742684269427042714272427342744275427642774278427942804281428242834284428542864287428842894290429142924293429442954296429742984299430043014302430343044305430643074308430943104311431243134314431543164317431843194320432143224323432443254326432743284329433043314332433343344335433643374338433943404341434243434344434543464347434843494350435143524353435443554356435743584359436043614362436343644365436643674368436943704371437243734374437543764377437843794380438143824383438443854386438743884389439043914392439343944395439643974398439944004401440244034404440544064407440844094410441144124413441444154416441744184419442044214422442344244425442644274428442944304431443244334434443544364437443844394440444144424443444444454446444744484449445044514452445344544455445644574458445944604461446244634464446544664467446844694470447144724473447444754476447744784479448044814482448344844485448644874488448944904491449244934494449544964497449844994500450145024503450445054506450745084509451045114512451345144515451645174518451945204521452245234524452545264527452845294530453145324533453445354536453745384539454045414542454345444545454645474548454945504551455245534554455545564557455845594560456145624563456445654566456745684569457045714572457345744575457645774578457945804581458245834584458545864587458845894590459145924593459445954596459745984599460046014602460346044605460646074608460946104611461246134614461546164617461846194620462146224623462446254626462746284629463046314632463346344635463646374638463946404641464246434644464546464647464846494650465146524653465446554656465746584659466046614662466346644665466646674668466946704671467246734674467546764677467846794680468146824683468446854686468746884689469046914692469346944695469646974698469947004701470247034704470547064707470847094710471147124713471447154716471747184719472047214722472347244725472647274728472947304731473247334734473547364737473847394740474147424743474447454746474747484749475047514752475347544755475647574758475947604761476247634764476547664767476847694770477147724773477447754776477747784779478047814782478347844785478647874788478947904791479247934794479547964797479847994800480148024803480448054806480748084809481048114812481348144815481648174818481948204821482248234824482548264827482848294830483148324833483448354836483748384839484048414842484348444845484648474848484948504851485248534854485548564857485848594860486148624863486448654866486748684869487048714872487348744875487648774878487948804881488248834884488548864887488848894890489148924893489448954896489748984899490049014902490349044905490649074908490949104911491249134914491549164917491849194920492149224923492449254926492749284929493049314932493349344935493649374938493949404941494249434944494549464947494849494950495149524953495449554956495749584959496049614962496349644965496649674968496949704971497249734974497549764977497849794980498149824983498449854986498749884989499049914992499349944995499649974998499950005001500250035004500550065007500850095010501150125013501450155016501750185019502050215022502350245025502650275028502950305031503250335034503550365037503850395040504150425043504450455046504750485049505050515052505350545055505650575058505950605061506250635064506550665067506850695070507150725073507450755076507750785079508050815082508350845085508650875088508950905091509250935094509550965097509850995100510151025103510451055106510751085109511051115112511351145115511651175118511951205121512251235124512551265127512851295130513151325133513451355136513751385139514051415142514351445145514651475148514951505151515251535154515551565157515851595160516151625163516451655166516751685169517051715172
  1. /*##############################################################################
  2. HPCC SYSTEMS software Copyright (C) 2012 HPCC Systems®.
  3. Licensed under the Apache License, Version 2.0 (the "License");
  4. you may not use this file except in compliance with the License.
  5. You may obtain a copy of the License at
  6. http://www.apache.org/licenses/LICENSE-2.0
  7. Unless required by applicable law or agreed to in writing, software
  8. distributed under the License is distributed on an "AS IS" BASIS,
  9. WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  10. See the License for the specific language governing permissions and
  11. limitations under the License.
  12. ############################################################################## */
  13. #include "platform.h"
  14. #include <jni.h>
  15. #include "jexcept.hpp"
  16. #include "jthread.hpp"
  17. #include "junicode.hpp"
  18. #include "hqlplugins.hpp"
  19. #include "deftype.hpp"
  20. #include "eclhelper.hpp"
  21. #include "eclrtl.hpp"
  22. #include "eclrtl_imp.hpp"
  23. #include "rtlfield.hpp"
  24. #include "rtlds_imp.hpp"
  25. #include "jprop.hpp"
  26. #include "roxiemem.hpp"
  27. #include "nbcd.hpp"
  28. #include "rtlformat.hpp"
  29. #include "esdl_def.hpp"
  30. #include "enginecontext.hpp"
  31. #ifndef _WIN32
  32. #include <sys/resource.h>
  33. #endif
  34. static const char * compatibleVersions[] = {
  35. "Java Embed Helper 1.0.0",
  36. NULL };
  37. static const char *version = "Java Embed Helper 1.0.0";
  38. #ifdef _DEBUG
  39. //#define TRACE_GLOBALREF
  40. //#define TRACE_CLASSFILE
  41. //#define CHECK_JNI
  42. //#define FORCE_GC
  43. /* Note - if you enable CHECK_JNI and see output like:
  44. * WARNING in native method: JNI call made without checking exceptions when required to from CallObjectMethodV
  45. * where for 'from' may be any of several functions, then the cause is likely to be a missing call to checkException()
  46. * after a call to the named function. One way to find the responsible call is with a breakpoint on checked_jni_CallObjectMethodV
  47. * The last time that breakpoint is hit before the warning is given should have a stack trace that tells you all you need to know.
  48. */
  49. #endif
  50. extern "C" DECL_EXPORT bool getECLPluginDefinition(ECLPluginDefinitionBlock *pb)
  51. {
  52. if (pb->size == sizeof(ECLPluginDefinitionBlockEx))
  53. {
  54. ECLPluginDefinitionBlockEx * pbx = (ECLPluginDefinitionBlockEx *) pb;
  55. pbx->compatibleVersions = compatibleVersions;
  56. }
  57. else if (pb->size != sizeof(ECLPluginDefinitionBlock))
  58. return false;
  59. pb->magicVersion = PLUGIN_VERSION;
  60. pb->version = version;
  61. pb->moduleName = "java";
  62. pb->ECL = NULL;
  63. pb->flags = PLUGIN_MULTIPLE_VERSIONS;
  64. pb->description = "Java Embed Helper";
  65. return true;
  66. }
  67. __declspec(noreturn) static void UNSUPPORTED(const char *feature) __attribute__((noreturn));
  68. static void UNSUPPORTED(const char *feature)
  69. {
  70. throw MakeStringException(-1, "UNSUPPORTED feature: %s not supported in java plugin", feature);
  71. }
  72. namespace javaembed {
  73. static jmethodID throwable_toString;
  74. static jmethodID throwable_getStackTrace;
  75. static jmethodID throwable_getCause;
  76. static jmethodID frame_toString;
  77. static void forceGC(class CheckedJNIEnv* JNIenv);
  78. /**
  79. * CheckedJNIEnv is a wrapper around JNIEnv that ensures that we check for exceptions after every call (and turn them into C++ exceptions).
  80. *
  81. * It should probably be refactored to have the JNIEnv pointer as a member rather than a base class. As it stands, it's possible to cast
  82. * safely between CheckedJNIEnv * and JNIEnv*
  83. *
  84. */
  85. class CheckedJNIEnv : private JNIEnv
  86. {
  87. template<typename T> T checkException(T a) { checkException(); return a; }
  88. public:
  89. StringBuffer &getString(StringBuffer &ret, jstring s)
  90. {
  91. if (s)
  92. {
  93. const char* str = GetStringUTFChars(s, NULL);
  94. ret.append(str);
  95. ReleaseStringUTFChars(s, str);
  96. }
  97. return ret;
  98. }
  99. StringAttr &getString(StringAttr &ret, jstring s)
  100. {
  101. if (s)
  102. {
  103. const char* str = GetStringUTFChars(s, NULL);
  104. ret.set(str);
  105. ReleaseStringUTFChars(s, str);
  106. }
  107. return ret;
  108. }
  109. void checkUnexpectedException()
  110. {
  111. if (JNIEnv::ExceptionCheck())
  112. {
  113. DBGLOG("javaembed: Uunexpected java exception while processing exception");
  114. JNIEnv::ExceptionDescribe();
  115. JNIEnv::ExceptionClear();
  116. throwUnexpected();
  117. }
  118. }
  119. void traceException(jthrowable exception)
  120. {
  121. // Don't use auto-checking in here, as we are already inside exception-checking code
  122. jstring msg = (jstring) JNIEnv::CallObjectMethod(exception, throwable_toString);
  123. checkUnexpectedException();
  124. const char *text = JNIEnv::GetStringUTFChars(msg, 0);
  125. DBGLOG("javaembed: exception: %s", text);
  126. JNIEnv::ReleaseStringUTFChars(msg, text);
  127. checkUnexpectedException();
  128. JNIEnv::DeleteLocalRef(msg);
  129. jobjectArray frames = (jobjectArray) JNIEnv::CallObjectMethod(exception, throwable_getStackTrace);
  130. checkUnexpectedException();
  131. jsize length = JNIEnv::GetArrayLength(frames);
  132. for (jsize i = 0; i < length; i++)
  133. {
  134. jobject frame = JNIEnv::GetObjectArrayElement(frames, i);
  135. checkUnexpectedException();
  136. msg = (jstring) JNIEnv::CallObjectMethod(frame, frame_toString);
  137. checkUnexpectedException();
  138. text = JNIEnv::GetStringUTFChars(msg, 0);
  139. DBGLOG("javaembed: exception: stack: %s", text);
  140. JNIEnv::ReleaseStringUTFChars(msg, text);
  141. checkUnexpectedException();
  142. JNIEnv::DeleteLocalRef(msg);
  143. JNIEnv::DeleteLocalRef(frame);
  144. }
  145. jthrowable cause = (jthrowable) JNIEnv::CallObjectMethod(exception, throwable_getCause);
  146. checkUnexpectedException();
  147. if (cause && cause != exception)
  148. {
  149. DBGLOG("javaembed: exception: Caused by:");
  150. traceException(cause);
  151. }
  152. }
  153. void checkException()
  154. {
  155. if (JNIEnv::ExceptionCheck())
  156. {
  157. jthrowable exception = JNIEnv::ExceptionOccurred();
  158. JNIEnv::ExceptionClear();
  159. traceException(exception);
  160. jstring cause = (jstring) JNIEnv::CallObjectMethod(exception, throwable_toString);
  161. JNIEnv::ExceptionClear();
  162. const char *text = JNIEnv::GetStringUTFChars(cause, 0);
  163. VStringBuffer message("javaembed: %s", text);
  164. JNIEnv::ReleaseStringUTFChars(cause, text);
  165. JNIEnv::ExceptionClear();
  166. rtlFail(0, message);
  167. }
  168. }
  169. jclass FindClass(const char *name)
  170. {
  171. return checkException(JNIEnv::FindClass(name));
  172. }
  173. jclass FindGlobalClass(const char *name)
  174. {
  175. return (jclass) NewGlobalRef(FindClass(name), "NewGlobalRef");
  176. }
  177. void GetBooleanArrayRegion(jbooleanArray array,
  178. jsize start, jsize len, jboolean *buf) {
  179. functions->GetBooleanArrayRegion(this,array,start,len,buf);
  180. checkException();
  181. }
  182. void GetByteArrayRegion(jbyteArray array,
  183. jsize start, jsize len, jbyte *buf) {
  184. functions->GetByteArrayRegion(this,array,start,len,buf);
  185. checkException();
  186. }
  187. void GetCharArrayRegion(jcharArray array,
  188. jsize start, jsize len, jchar *buf) {
  189. functions->GetCharArrayRegion(this,array,start,len,buf);
  190. checkException();
  191. }
  192. void GetShortArrayRegion(jshortArray array,
  193. jsize start, jsize len, jshort *buf) {
  194. functions->GetShortArrayRegion(this,array,start,len,buf);
  195. checkException();
  196. }
  197. void GetIntArrayRegion(jintArray array,
  198. jsize start, jsize len, jint *buf) {
  199. functions->GetIntArrayRegion(this,array,start,len,buf);
  200. checkException();
  201. }
  202. void GetLongArrayRegion(jlongArray array,
  203. jsize start, jsize len, jlong *buf) {
  204. functions->GetLongArrayRegion(this,array,start,len,buf);
  205. checkException();
  206. }
  207. void GetFloatArrayRegion(jfloatArray array,
  208. jsize start, jsize len, jfloat *buf) {
  209. functions->GetFloatArrayRegion(this,array,start,len,buf);
  210. checkException();
  211. }
  212. void GetDoubleArrayRegion(jdoubleArray array,
  213. jsize start, jsize len, jdouble *buf) {
  214. functions->GetDoubleArrayRegion(this,array,start,len,buf);
  215. checkException();
  216. }
  217. jsize GetArrayLength(jarray array) {
  218. return checkException(functions->GetArrayLength(this,array));
  219. }
  220. jsize GetStringUTFLength(jstring str) {
  221. return checkException(functions->GetStringUTFLength(this,str));
  222. }
  223. const char* GetStringUTFChars(jstring str, jboolean *isCopy) {
  224. return checkException(functions->GetStringUTFChars(this,str,isCopy));
  225. }
  226. void ReleaseStringUTFChars(jstring str, const char* chars) {
  227. functions->ReleaseStringUTFChars(this,str,chars);
  228. checkException();
  229. }
  230. jboolean GetBooleanField(jobject obj, jfieldID fieldID)
  231. {
  232. return checkException(JNIEnv::GetBooleanField(obj,fieldID));
  233. }
  234. jobject GetObjectField(jobject obj, jfieldID fieldID)
  235. {
  236. return checkException(JNIEnv::GetObjectField(obj,fieldID));
  237. }
  238. jbyte GetByteField(jobject obj, jfieldID fieldID) {
  239. return checkException(JNIEnv::GetByteField(obj,fieldID));
  240. }
  241. jchar GetCharField(jobject obj, jfieldID fieldID) {
  242. return checkException(JNIEnv::GetCharField(obj,fieldID));
  243. }
  244. jshort GetShortField(jobject obj, jfieldID fieldID) {
  245. return checkException(JNIEnv::GetShortField(obj,fieldID));
  246. }
  247. jint GetIntField(jobject obj, jfieldID fieldID) {
  248. return checkException(JNIEnv::GetIntField(obj,fieldID));
  249. }
  250. jlong GetLongField(jobject obj, jfieldID fieldID) {
  251. return checkException(JNIEnv::GetLongField(obj,fieldID));
  252. }
  253. jfloat GetFloatField(jobject obj, jfieldID fieldID)
  254. {
  255. return checkException(JNIEnv::GetFloatField(obj,fieldID));
  256. }
  257. jdouble GetDoubleField(jobject obj, jfieldID fieldID)
  258. {
  259. return checkException(JNIEnv::GetDoubleField(obj,fieldID));
  260. }
  261. jboolean IsSameObject(jobject obj1, jobject obj2)
  262. {
  263. return checkException(JNIEnv::IsSameObject(obj1, obj2));
  264. }
  265. void SetObjectField(jobject obj, jfieldID fieldID, jobject val) {
  266. functions->SetObjectField(this,obj,fieldID,val);
  267. checkException();
  268. }
  269. void SetBooleanField(jobject obj, jfieldID fieldID,
  270. jboolean val) {
  271. functions->SetBooleanField(this,obj,fieldID,val);
  272. checkException();
  273. }
  274. void SetByteField(jobject obj, jfieldID fieldID,
  275. jbyte val) {
  276. functions->SetByteField(this,obj,fieldID,val);
  277. checkException();
  278. }
  279. void SetCharField(jobject obj, jfieldID fieldID,
  280. jchar val) {
  281. functions->SetCharField(this,obj,fieldID,val);
  282. checkException();
  283. }
  284. void SetShortField(jobject obj, jfieldID fieldID,
  285. jshort val) {
  286. functions->SetShortField(this,obj,fieldID,val);
  287. checkException();
  288. }
  289. void SetIntField(jobject obj, jfieldID fieldID,
  290. jint val) {
  291. functions->SetIntField(this,obj,fieldID,val);
  292. checkException();
  293. }
  294. void SetLongField(jobject obj, jfieldID fieldID,
  295. jlong val) {
  296. functions->SetLongField(this,obj,fieldID,val);
  297. checkException();
  298. }
  299. void SetFloatField(jobject obj, jfieldID fieldID,
  300. jfloat val) {
  301. functions->SetFloatField(this,obj,fieldID,val);
  302. checkException();
  303. }
  304. void SetDoubleField(jobject obj, jfieldID fieldID,
  305. jdouble val) {
  306. functions->SetDoubleField(this,obj,fieldID,val);
  307. checkException();
  308. }
  309. jstring NewString(const jchar *unicode, jsize len) {
  310. return checkException(functions->NewString(this,unicode,len));
  311. }
  312. jbooleanArray NewBooleanArray(jsize len) {
  313. return checkException(functions->NewBooleanArray(this,len));
  314. }
  315. jbyteArray NewByteArray(jsize len) {
  316. return checkException(functions->NewByteArray(this,len));
  317. }
  318. jcharArray NewCharArray(jsize len) {
  319. return checkException(functions->NewCharArray(this,len));
  320. }
  321. jshortArray NewShortArray(jsize len) {
  322. return checkException(functions->NewShortArray(this,len));
  323. }
  324. jintArray NewIntArray(jsize len) {
  325. return checkException(functions->NewIntArray(this,len));
  326. }
  327. jlongArray NewLongArray(jsize len) {
  328. return checkException(functions->NewLongArray(this,len));
  329. }
  330. jfloatArray NewFloatArray(jsize len) {
  331. return checkException(functions->NewFloatArray(this,len));
  332. }
  333. jdoubleArray NewDoubleArray(jsize len) {
  334. return checkException(functions->NewDoubleArray(this,len));
  335. }
  336. void SetBooleanArrayRegion(jbooleanArray array, jsize start, jsize len,
  337. const jboolean *buf) {
  338. functions->SetBooleanArrayRegion(this,array,start,len,buf);
  339. checkException();
  340. }
  341. void SetByteArrayRegion(jbyteArray array, jsize start, jsize len,
  342. const jbyte *buf) {
  343. functions->SetByteArrayRegion(this,array,start,len,buf);
  344. checkException();
  345. }
  346. void SetCharArrayRegion(jcharArray array, jsize start, jsize len,
  347. const jchar *buf) {
  348. functions->SetCharArrayRegion(this,array,start,len,buf);
  349. checkException();
  350. }
  351. void SetShortArrayRegion(jshortArray array, jsize start, jsize len,
  352. const jshort *buf) {
  353. functions->SetShortArrayRegion(this,array,start,len,buf);
  354. checkException();
  355. }
  356. void SetIntArrayRegion(jintArray array, jsize start, jsize len,
  357. const jint *buf) {
  358. functions->SetIntArrayRegion(this,array,start,len,buf);
  359. checkException();
  360. }
  361. void SetLongArrayRegion(jlongArray array, jsize start, jsize len,
  362. const jlong *buf) {
  363. functions->SetLongArrayRegion(this,array,start,len,buf);
  364. checkException();
  365. }
  366. void SetFloatArrayRegion(jfloatArray array, jsize start, jsize len,
  367. const jfloat *buf) {
  368. functions->SetFloatArrayRegion(this,array,start,len,buf);
  369. checkException();
  370. }
  371. void SetDoubleArrayRegion(jdoubleArray array, jsize start, jsize len,
  372. const jdouble *buf) {
  373. functions->SetDoubleArrayRegion(this,array,start,len,buf);
  374. checkException();
  375. }
  376. jobjectArray NewObjectArray(jsize len, jclass clazz,
  377. jobject init) {
  378. return checkException(functions->NewObjectArray(this,len,clazz,init));
  379. }
  380. jobject GetObjectArrayElement(jobjectArray array, jsize index) {
  381. return checkException(functions->GetObjectArrayElement(this,array,index));
  382. }
  383. void SetObjectArrayElement(jobjectArray array, jsize index,
  384. jobject val) {
  385. functions->SetObjectArrayElement(this,array,index,val);
  386. checkException();
  387. }
  388. jobject ToReflectedField(jclass cls, jfieldID fieldID, jboolean isStatic) {
  389. return checkException(functions->ToReflectedField(this,cls,fieldID,isStatic));
  390. }
  391. jboolean IsAssignableFrom(jclass clazz1, jclass clazz2) {
  392. return checkException(functions->IsAssignableFrom(this,clazz1,clazz2));
  393. }
  394. jclass GetObjectClass(jobject obj)
  395. {
  396. return checkException(JNIEnv::GetObjectClass(obj));
  397. }
  398. jfieldID GetFieldID(jclass clazz, const char *name, const char *sig)
  399. {
  400. return checkException(JNIEnv::GetFieldID(clazz,name,sig));
  401. }
  402. jfieldID GetFieldIDUnchecked(jclass clazz, const char *name, const char *sig)
  403. {
  404. jfieldID ret = JNIEnv::GetFieldID(clazz,name,sig);
  405. ExceptionClear();
  406. return ret;
  407. }
  408. jmethodID GetMethodID(jclass clazz, const char *name, const char *sig)
  409. {
  410. return checkException(JNIEnv::GetMethodID(clazz, name, sig));
  411. }
  412. jmethodID GetMethodIDUnchecked(jclass clazz, const char *name, const char *sig)
  413. {
  414. jmethodID ret = JNIEnv::GetMethodID(clazz, name, sig);
  415. ExceptionClear();
  416. return ret;
  417. }
  418. jmethodID GetStaticMethodID(jclass clazz, const char *name, const char *sig)
  419. {
  420. return checkException(JNIEnv::GetStaticMethodID(clazz, name, sig));
  421. }
  422. void CallStaticVoidMethod(jclass cls, jmethodID methodID, ...) {
  423. va_list args;
  424. va_start(args,methodID);
  425. functions->CallStaticVoidMethodV(this,cls,methodID,args);
  426. va_end(args);
  427. checkException();
  428. }
  429. jobject CallStaticObjectMethod(jclass clazz, jmethodID methodID, ...)
  430. {
  431. va_list args;
  432. jobject result;
  433. va_start(args,methodID);
  434. result = JNIEnv::CallStaticObjectMethodV(clazz,methodID,args);
  435. va_end(args);
  436. checkException();
  437. return result;
  438. }
  439. jdouble CallDoubleMethod(jobject obj, jmethodID methodID, ...) {
  440. va_list args;
  441. jdouble result;
  442. va_start(args,methodID);
  443. result = functions->CallDoubleMethodV(this,obj,methodID,args);
  444. va_end(args);
  445. checkException();
  446. return result;
  447. }
  448. jlong CallLongMethod(jobject obj, jmethodID methodID, ...) {
  449. va_list args;
  450. jlong result;
  451. va_start(args,methodID);
  452. result = functions->CallLongMethodV(this,obj,methodID,args);
  453. va_end(args);
  454. checkException();
  455. return result;
  456. }
  457. jboolean CallBooleanMethod(jobject obj, jmethodID methodID, ...)
  458. {
  459. va_list args;
  460. jboolean result;
  461. va_start(args,methodID);
  462. result = JNIEnv::CallBooleanMethodV(obj,methodID,args);
  463. va_end(args);
  464. checkException();
  465. return result;
  466. }
  467. jobject CallObjectMethod(jobject obj, jmethodID methodID, ...)
  468. {
  469. va_list args;
  470. jobject result;
  471. va_start(args,methodID);
  472. result = JNIEnv::CallObjectMethodV(obj,methodID,args);
  473. va_end(args);
  474. checkException();
  475. return result;
  476. }
  477. jchar CallCharMethodA(jobject obj, jmethodID methodID,
  478. const jvalue * args) {
  479. return checkException(functions->CallCharMethodA(this,obj,methodID,args));
  480. }
  481. jboolean CallBooleanMethodA(jobject obj, jmethodID methodID,
  482. const jvalue * args) {
  483. return checkException(functions->CallBooleanMethodA(this,obj,methodID, args));
  484. }
  485. jshort CallShortMethodA(jobject obj, jmethodID methodID,
  486. const jvalue * args) {
  487. return checkException(functions->CallShortMethodA(this,obj,methodID,args));
  488. }
  489. jlong CallLongMethodA(jobject obj, jmethodID methodID,
  490. const jvalue * args) {
  491. return checkException(functions->CallLongMethodA(this,obj,methodID,args));
  492. }
  493. jfloat CallFloatMethodA(jobject obj, jmethodID methodID,
  494. const jvalue * args) {
  495. return checkException(functions->CallFloatMethodA(this,obj,methodID,args));
  496. }
  497. jdouble CallDoubleMethodA(jobject obj, jmethodID methodID,
  498. const jvalue * args) {
  499. return checkException(functions->CallDoubleMethodA(this,obj,methodID,args));
  500. }
  501. jint CallIntMethodA(jobject obj, jmethodID methodID,
  502. const jvalue * args) {
  503. return checkException(functions->CallIntMethodA(this,obj,methodID,args));
  504. }
  505. jbyte CallByteMethodA(jobject obj, jmethodID methodID,
  506. const jvalue * args) {
  507. return checkException(functions->CallByteMethodA(this,obj,methodID,args));
  508. }
  509. jobject CallObjectMethodA(jobject obj, jmethodID methodID,
  510. const jvalue * args) {
  511. return checkException(functions->CallObjectMethodA(this,obj,methodID,args));
  512. }
  513. void CallVoidMethodA(jobject obj, jmethodID methodID,
  514. const jvalue * args) {
  515. functions->CallVoidMethodA(this,obj,methodID,args);
  516. return checkException();
  517. }
  518. jchar CallStaticCharMethodA(jclass clazz, jmethodID methodID,
  519. const jvalue * args) {
  520. return checkException(functions->CallStaticCharMethodA(this,clazz,methodID,args));
  521. }
  522. jboolean CallStaticBooleanMethodA(jclass clazz, jmethodID methodID,
  523. const jvalue * args) {
  524. return checkException(functions->CallStaticBooleanMethodA(this,clazz,methodID, args));
  525. }
  526. jshort CallStaticShortMethodA(jclass clazz, jmethodID methodID,
  527. const jvalue * args) {
  528. return checkException(functions->CallStaticShortMethodA(this,clazz,methodID,args));
  529. }
  530. jlong CallStaticLongMethodA(jclass clazz, jmethodID methodID,
  531. const jvalue * args) {
  532. return checkException(functions->CallStaticLongMethodA(this,clazz,methodID,args));
  533. }
  534. jfloat CallStaticFloatMethodA(jclass clazz, jmethodID methodID,
  535. const jvalue * args) {
  536. return checkException(functions->CallStaticFloatMethodA(this,clazz,methodID,args));
  537. }
  538. jdouble CallStaticDoubleMethodA(jclass clazz, jmethodID methodID,
  539. const jvalue * args) {
  540. return checkException(functions->CallStaticDoubleMethodA(this,clazz,methodID,args));
  541. }
  542. jint CallStaticIntMethodA(jclass clazz, jmethodID methodID,
  543. const jvalue * args) {
  544. return checkException(functions->CallStaticIntMethodA(this,clazz,methodID,args));
  545. }
  546. jbyte CallStaticByteMethodA(jclass clazz, jmethodID methodID,
  547. const jvalue * args) {
  548. return checkException(functions->CallStaticByteMethodA(this,clazz,methodID,args));
  549. }
  550. jobject CallStaticObjectMethodA(jclass clazz, jmethodID methodID,
  551. const jvalue * args) {
  552. return checkException(functions->CallStaticObjectMethodA(this,clazz,methodID,args));
  553. }
  554. void CallStaticVoidMethodA(jclass clazz, jmethodID methodID,
  555. const jvalue * args) {
  556. functions->CallStaticVoidMethodA(this,clazz,methodID,args);
  557. return checkException();
  558. }
  559. jobject NewObjectA(jclass clazz, jmethodID methodID, const jvalue *args)
  560. {
  561. return checkException(JNIEnv::NewObjectA(clazz,methodID,args));
  562. }
  563. jobject NewObject(jclass clazz, jmethodID methodID, ...)
  564. {
  565. va_list args;
  566. jobject result;
  567. va_start(args, methodID);
  568. result = JNIEnv::NewObjectV(clazz,methodID,args);
  569. va_end(args);
  570. checkException();
  571. return result;
  572. }
  573. jstring NewStringUTF(const char *utf)
  574. {
  575. return checkException(JNIEnv::NewStringUTF(utf));
  576. }
  577. jfieldID FromReflectedField(jobject field)
  578. {
  579. return checkException(JNIEnv::FromReflectedField(field));
  580. }
  581. using JNIEnv::PushLocalFrame;
  582. using JNIEnv::PopLocalFrame;
  583. using JNIEnv::DeleteLocalRef;
  584. using JNIEnv::ExceptionClear;
  585. using JNIEnv::ExceptionCheck;
  586. using JNIEnv::GetObjectRefType;
  587. #ifdef TRACE_GLOBALREF
  588. void DeleteGlobalRef(jobject val)
  589. {
  590. DBGLOG("DeleteGlobalRef %p", val);
  591. JNIEnv::DeleteGlobalRef(val);
  592. #ifdef FORCE_GC
  593. forceGC(this);
  594. #endif
  595. }
  596. jobject NewGlobalRef(jobject val, const char *why)
  597. {
  598. jobject ret = JNIEnv::NewGlobalRef(val);
  599. DBGLOG("NewGlobalRef %p (%s) returns %p", val, why, ret);
  600. return ret;
  601. }
  602. #else
  603. inline void DeleteGlobalRef(jobject val)
  604. {
  605. JNIEnv::DeleteGlobalRef(val);
  606. #ifdef FORCE_GC
  607. forceGC(this);
  608. #endif
  609. }
  610. inline jobject NewGlobalRef(jobject val, const char *)
  611. {
  612. return JNIEnv::NewGlobalRef(val);
  613. }
  614. #endif
  615. };
  616. static bool printNameForClass(CheckedJNIEnv *JNIenv, jobject clsObj)
  617. {
  618. if (!clsObj)
  619. {
  620. printf("Object %p is null\n", clsObj);
  621. return false;
  622. }
  623. jclass cls = JNIenv->GetObjectClass(clsObj);
  624. jmethodID mid = JNIenv->GetMethodID(cls, "getName", "()Ljava/lang/String;");
  625. jstring strObj = (jstring) JNIenv->CallObjectMethod(clsObj, mid);
  626. const char* str = JNIenv->GetStringUTFChars(strObj, NULL);
  627. printf("class %s\n", str);
  628. bool ret = streq(str, "java.lang.Class");
  629. JNIenv->ReleaseStringUTFChars(strObj, str);
  630. return ret;
  631. }
  632. static void printClassForObject(CheckedJNIEnv *JNIenv, jobject obj)
  633. {
  634. printf("Object %p ", obj);
  635. if (!obj)
  636. {
  637. printf("is null\n");
  638. return;
  639. }
  640. jclass objClass = JNIenv->GetObjectClass(obj);
  641. jmethodID mid = JNIenv->GetMethodID(objClass, "getClass", "()Ljava/lang/Class;");
  642. jobject clsObj = JNIenv->CallObjectMethod(obj, mid);
  643. if (printNameForClass(JNIenv, clsObj))
  644. {
  645. printf(" ");
  646. printNameForClass(JNIenv, obj);
  647. }
  648. }
  649. static StringBuffer &getClassNameForObject(CheckedJNIEnv *JNIenv, StringBuffer &ret, jobject obj)
  650. {
  651. if (obj)
  652. {
  653. jclass objClass = JNIenv->GetObjectClass(obj);
  654. jmethodID mid = JNIenv->GetMethodID(objClass, "getClass", "()Ljava/lang/Class;");
  655. jobject clsObj = JNIenv->CallObjectMethod(obj, mid);
  656. jclass cls = JNIenv->GetObjectClass(clsObj);
  657. mid = JNIenv->GetMethodID(cls, "getName", "()Ljava/lang/String;");
  658. jstring strObj = (jstring) JNIenv->CallObjectMethod(clsObj, mid);
  659. const char* str = JNIenv->GetStringUTFChars(strObj, NULL);
  660. ret.append(str);
  661. JNIenv->ReleaseStringUTFChars(strObj, str);
  662. }
  663. return ret;
  664. }
  665. static jobject getClassLoader(CheckedJNIEnv *JNIenv, jclass obj)
  666. {
  667. jclass objClass = JNIenv->GetObjectClass(obj);
  668. jmethodID mid = JNIenv->GetMethodID(objClass, "getClassLoader", "()Ljava/lang/ClassLoader;");
  669. jobject classloader = JNIenv->CallObjectMethod(obj, mid);
  670. return classloader;
  671. }
  672. static StringBuffer helperLibraryName;
  673. static CheckedJNIEnv *queryJNIEnv();
  674. // Some global objects setup at load time for efficiency and code readability
  675. static jclass customLoaderClass;
  676. static jmethodID clc_newInstance;
  677. static jmethodID clc_getSignature;
  678. static jclass hpccIteratorClass;
  679. static jmethodID hi_constructor;
  680. static jmethodID hi_load;
  681. static jclass utilIteratorClass;
  682. static jclass langIterableClass;
  683. static jmethodID iterable_iterator;
  684. static jclass systemClass;
  685. static jmethodID system_gc;
  686. static jclass javaLangClassLoaderClass;
  687. static jmethodID cl_getSystemClassLoader;
  688. static jclass javaLangThreadClass;
  689. static jmethodID thread_currentThread;
  690. static jmethodID thread_getContextClassLoader;
  691. static jmethodID thread_setContextClassLoader;
  692. static jclass langObjectClass;
  693. static jmethodID object_toString;
  694. static jclass arrayListClass;
  695. static jmethodID arrayList_toArray;
  696. static jmethodID arrayList_constructor;
  697. static jmethodID arrayList_add;
  698. static jclass langStringClass;
  699. static jclass netURLClass;
  700. static jmethodID netURL_constructor;
  701. static jclass throwableClass;
  702. //static jmethodID throwable_toString; and others declared above
  703. static jclass stackTraceElementClass;
  704. static jclass langIllegalArgumentExceptionClass;
  705. static void forceGC(CheckedJNIEnv* JNIenv)
  706. {
  707. JNIenv->CallStaticVoidMethod(systemClass, system_gc);
  708. }
  709. static void setupGlobals(CheckedJNIEnv *J)
  710. {
  711. try
  712. {
  713. // Load this first as we can't report errors on the others sensibly if this one not loaded!
  714. throwableClass = J->FindGlobalClass("java/lang/Throwable");
  715. throwable_toString = J->GetMethodID(throwableClass, "toString", "()Ljava/lang/String;");
  716. throwable_getStackTrace = J->GetMethodID(throwableClass, "getStackTrace", "()[Ljava/lang/StackTraceElement;");
  717. throwable_getCause = J->GetMethodID(throwableClass, "getCause", "()Ljava/lang/Throwable;");
  718. stackTraceElementClass = J->FindGlobalClass("java/lang/StackTraceElement");
  719. frame_toString = J->GetMethodID(stackTraceElementClass, "toString", "()Ljava/lang/String;");
  720. systemClass = J->FindGlobalClass("java/lang/System");
  721. system_gc = J->GetStaticMethodID(systemClass, "gc", "()V");
  722. javaLangClassLoaderClass = J->FindGlobalClass("java/lang/ClassLoader");
  723. cl_getSystemClassLoader = J->GetStaticMethodID(javaLangClassLoaderClass, "getSystemClassLoader", "()Ljava/lang/ClassLoader;");
  724. javaLangThreadClass = J->FindGlobalClass("java/lang/Thread");
  725. thread_currentThread = J->GetStaticMethodID(javaLangThreadClass, "currentThread", "()Ljava/lang/Thread;");
  726. thread_getContextClassLoader = J->GetMethodID(javaLangThreadClass, "getContextClassLoader", "()Ljava/lang/ClassLoader;");
  727. thread_setContextClassLoader = J->GetMethodID(javaLangThreadClass, "setContextClassLoader", "(Ljava/lang/ClassLoader;)V");
  728. langObjectClass = J->FindGlobalClass("java/lang/Object");
  729. object_toString = J->GetMethodID(langObjectClass, "toString", "()Ljava/lang/String;");
  730. arrayListClass = J->FindGlobalClass("java/util/ArrayList");
  731. arrayList_constructor = J->GetMethodID(arrayListClass, "<init>", "()V");
  732. arrayList_add = J->GetMethodID(arrayListClass, "add", "(Ljava/lang/Object;)Z");
  733. arrayList_toArray = J->GetMethodID(arrayListClass, "toArray", "()[Ljava/lang/Object;" );
  734. langStringClass = J->FindGlobalClass("java/lang/String");
  735. langIterableClass = J->FindGlobalClass("java/lang/Iterable");
  736. iterable_iterator = J->GetMethodID(langIterableClass, "iterator", "()Ljava/util/Iterator;");
  737. utilIteratorClass = J->FindGlobalClass("java/util/Iterator");
  738. langIllegalArgumentExceptionClass = J->FindGlobalClass("java/lang/IllegalArgumentException");
  739. }
  740. catch (IException *E)
  741. {
  742. Owned<IException> e = E;
  743. throw makeWrappedExceptionV(E, E->errorCode(), "javaembed: Unable to load Java system classes - is classpath set properly?");
  744. }
  745. try
  746. {
  747. customLoaderClass = J->FindGlobalClass("com/HPCCSystems/HpccClassLoader");
  748. clc_newInstance = J->GetStaticMethodID(customLoaderClass, "newInstance","(Ljava/lang/String;Ljava/lang/ClassLoader;IJLjava/lang/String;)Lcom/HPCCSystems/HpccClassLoader;");
  749. clc_getSignature = J->GetStaticMethodID(customLoaderClass, "getSignature","(Ljava/lang/Class;Ljava/lang/String;)Ljava/lang/String;");
  750. hpccIteratorClass = J->FindGlobalClass("com/HPCCSystems/HpccUtils");
  751. hi_constructor = J->GetMethodID(hpccIteratorClass, "<init>", "(JLjava/lang/String;)V");
  752. hi_load = J->GetStaticMethodID(hpccIteratorClass, "load", "(Ljava/lang/String;)V");
  753. J->CallStaticVoidMethod(hpccIteratorClass, hi_load, J->NewStringUTF(helperLibraryName));
  754. }
  755. catch (IException *E)
  756. {
  757. Owned<IException> e = E;
  758. throw makeWrappedExceptionV(E, E->errorCode(), "javaembed: Unable to find HPCC classes - is classpath set properly?");
  759. }
  760. }
  761. static StringAttr & getSignature(StringAttr &ret, CheckedJNIEnv *J, jclass clazz, const char *funcName)
  762. {
  763. StringBuffer sig;
  764. jstring result = (jstring) J->CallStaticObjectMethod(customLoaderClass, clc_getSignature, clazz, J->NewStringUTF(funcName));
  765. J->getString(sig, result);
  766. sig.replace('.', '/');
  767. ret.set(sig);
  768. return ret;
  769. }
  770. /**
  771. * The following classes are used to ensure that the code in loadFunction that creates an instance
  772. * that is shared between multiple callers is only called on one thread, while other threads will wait
  773. * and use the instance created by the first thread.
  774. *
  775. */
  776. class PersistedObject : public MappingBase
  777. {
  778. public:
  779. PersistedObject(const char *_name) : name(_name) {}
  780. ~PersistedObject()
  781. {
  782. if (instance)
  783. {
  784. #ifdef TRACE_GLOBALREF
  785. DBGLOG("DeleteGlobalRef(singleton): %p", instance);
  786. #endif
  787. queryJNIEnv()->DeleteGlobalRef(instance);
  788. }
  789. }
  790. CriticalSection crit;
  791. jobject instance = nullptr;
  792. StringAttr name;
  793. virtual const void * getKey() const { return name; }
  794. };
  795. class PersistedObjectCriticalBlock
  796. {
  797. PersistedObject *obj = nullptr;
  798. public:
  799. inline PersistedObjectCriticalBlock()
  800. {
  801. }
  802. inline ~PersistedObjectCriticalBlock()
  803. {
  804. if (obj)
  805. obj->crit.leave();
  806. }
  807. inline void enter(PersistedObject *_obj)
  808. {
  809. // Note that the object should be locked before we are called
  810. assertex(!obj);
  811. obj = _obj;
  812. }
  813. inline void leave(jobject instance = nullptr)
  814. {
  815. if (obj)
  816. {
  817. if (instance)
  818. obj->instance = instance;
  819. obj->crit.leave();
  820. obj = nullptr;
  821. }
  822. }
  823. inline bool locked()
  824. {
  825. return obj != nullptr;
  826. }
  827. jobject getInstance()
  828. {
  829. assertex(obj);
  830. return obj->instance;
  831. }
  832. };
  833. // Use a global object to ensure that the Java VM is initialized once only.
  834. // We would like to create it lazily for two reasons:
  835. // 1. So that we only get a JVM if we need one (even if we have loaded the plugin)
  836. // 2. It's important for the JVM to be initialized AFTER we have set up signal handlers, as it
  837. // likes to set its own (in particular, it seems to intercept and ignore some SIGSEGV during the
  838. // garbage collection).
  839. // Unfortunately, it seems that the design of the JNI interface is such that JNI_CreateJavaVM has to be called on the 'main thread'.
  840. // So we can't achieve 1, and 2 requires that we create via the INIT_MODULE mechanism (rather than just a static object), and that
  841. // any engines that call InitModuleObjects() or load plugins dynamically do so AFTER setting any signal handlers or calling
  842. // EnableSEHtoExceptionMapping
  843. //
  844. static StringBuffer &appendClassPath(StringBuffer &classPath)
  845. {
  846. const IProperties &conf = queryEnvironmentConf();
  847. if (conf.hasProp("classpath"))
  848. {
  849. conf.getProp("classpath", classPath);
  850. classPath.append(ENVSEPCHAR);
  851. }
  852. else
  853. {
  854. classPath.append(hpccBuildInfo.installDir).append(PATHSEPCHAR).append("classes").append(ENVSEPCHAR);
  855. }
  856. return classPath;
  857. }
  858. static class JavaGlobalState
  859. {
  860. public:
  861. JavaGlobalState() : persistedObjects(false)
  862. {
  863. JavaVMInitArgs vm_args; /* JDK/JRE 6 VM initialization arguments */
  864. StringArray optionStrings;
  865. const char* origPath = getenv("CLASSPATH");
  866. StringBuffer newPath;
  867. newPath.append("-Djava.class.path=");
  868. if (origPath && *origPath)
  869. {
  870. newPath.append(origPath).append(ENVSEPCHAR);
  871. }
  872. appendClassPath(newPath);
  873. newPath.append(".");
  874. optionStrings.append(newPath);
  875. const IProperties &conf = queryEnvironmentConf();
  876. if (conf.hasProp("jvmlibpath"))
  877. {
  878. StringBuffer libPath;
  879. libPath.append("-Djava.library.path=");
  880. conf.getProp("jvmlibpath", libPath);
  881. optionStrings.append(libPath);
  882. }
  883. // Options we should set (but allow for override with jvmoptions below)
  884. optionStrings.append("-XX:-UseLargePages");
  885. if (conf.hasProp("jvmoptions"))
  886. {
  887. // Use space as field sep as ':' and ';' are valid
  888. optionStrings.appendList(conf.queryProp("jvmoptions"), " ");
  889. }
  890. // Options we know we always want set
  891. optionStrings.append("-Xrs");
  892. #ifdef RLIMIT_STACK
  893. // JVM has a habit of reducing the stack limit on main thread to 1M - probably dates back to when it was actually an increase...
  894. StringBuffer stackOption("-Xss");
  895. struct rlimit limit;
  896. rlim_t slim = 0;
  897. if (getrlimit (RLIMIT_STACK, &limit)==0)
  898. slim = limit.rlim_cur;
  899. if (!slim)
  900. slim = 8*1024*1024;
  901. if (slim >= 1*1024*1024)
  902. {
  903. stackOption.append((__uint64) slim);
  904. optionStrings.append(stackOption);
  905. }
  906. #endif
  907. // These may be useful for debugging
  908. #ifdef CHECK_JNI
  909. optionStrings.append("-Xcheck:jni");
  910. optionStrings.append("-verbose:jni");
  911. optionStrings.append("-XX:+TraceClassLoading");
  912. #endif
  913. JavaVMOption* options = new JavaVMOption[optionStrings.length()];
  914. ForEachItemIn(idx, optionStrings)
  915. {
  916. // DBGLOG("javaembed: Setting JVM option: %s",(char *)optionStrings.item(idx));
  917. options[idx].optionString = (char *) optionStrings.item(idx);
  918. options[idx].extraInfo = NULL;
  919. }
  920. vm_args.nOptions = optionStrings.length();
  921. vm_args.options = options;
  922. vm_args.ignoreUnrecognized = true;
  923. vm_args.version = JNI_VERSION_1_8;
  924. /* load and initialize a Java VM, return a JNI interface pointer in env */
  925. JNIEnv *env; /* receives pointer to native method interface */
  926. int createResult = JNI_CreateJavaVM(&javaVM, (void**)&env, &vm_args);
  927. delete [] options;
  928. if (createResult != 0)
  929. throw MakeStringException(0, "javaembed: Unable to initialize JVM (%d)",createResult);
  930. setupGlobals((CheckedJNIEnv *) env);
  931. // DBGLOG("JNI environment version %x loaded", env->GetVersion()); // Comes out a bit too early
  932. }
  933. ~JavaGlobalState()
  934. {
  935. /* We could release global persisted classes here but not a lot of point. Code would look something like this:
  936. HashIterator it(persistedObjects);
  937. ForEach(it)
  938. {
  939. IMapping &entry = it.query();
  940. jobject *pObj = persistedObjects.mapToValue(&entry);
  941. if (pClass)
  942. queryJNIEnv()->DeleteGlobalRef(*pObj);
  943. }
  944. */
  945. // This function is never called anyway...
  946. // We don't attempt to destroy the Java VM, as it's buggy...
  947. }
  948. PersistedObject *getGlobalObject(CheckedJNIEnv *JNIenv, const char *name)
  949. {
  950. PersistedObject *p;
  951. {
  952. CriticalBlock b(hashCrit);
  953. p = persistedObjects.find(name);
  954. if (!p)
  955. {
  956. p = new PersistedObject(name);
  957. persistedObjects.replaceOwn(*p);
  958. }
  959. }
  960. p->crit.enter(); // outside the hashCrit block, otherwise I think there is a possibility of deadlock
  961. return p;
  962. }
  963. void doUnregister(const char *key)
  964. {
  965. CriticalBlock b(hashCrit);
  966. persistedObjects.remove(key);
  967. }
  968. static void unregister(const char *key);
  969. JavaVM *javaVM; /* denotes a Java VM */
  970. private:
  971. CriticalSection hashCrit;
  972. StringMapOf<PersistedObject> persistedObjects;
  973. } *globalState;
  974. void JavaGlobalState::unregister(const char *key)
  975. {
  976. // Remove a class that was persisted via : PERSIST options - it has come to the end of its life
  977. globalState->doUnregister(key);
  978. }
  979. #ifdef _WIN32
  980. EXTERN_C IMAGE_DOS_HEADER __ImageBase;
  981. #endif
  982. MODULE_INIT(INIT_PRIORITY_STANDARD)
  983. {
  984. // Make sure we are never unloaded (as JVM does not support it)
  985. // we do this by doing a dynamic load of the javaembed library
  986. #ifdef _WIN32
  987. char ln[_MAX_PATH];
  988. ::GetModuleFileName((HINSTANCE)&__ImageBase, ln, _MAX_PATH);
  989. if (strstr(path, "javaembed"))
  990. {
  991. HINSTANCE h = LoadSharedObject(ln, false, false);
  992. helperLibraryName.set(ln);
  993. DBGLOG("LoadSharedObject returned %p", h);
  994. }
  995. #else
  996. if (findLoadedModule(helperLibraryName, "javaembed"))
  997. {
  998. HINSTANCE h = LoadSharedObject(helperLibraryName, false, false);
  999. // Deliberately leak this handle
  1000. }
  1001. #endif
  1002. globalState = new JavaGlobalState;
  1003. return true;
  1004. }
  1005. MODULE_EXIT()
  1006. {
  1007. // We don't attempt to destroy the Java VM, as it's buggy...
  1008. // delete globalState;
  1009. // globalState = NULL;
  1010. }
  1011. static void checkType(type_t javatype, size32_t javasize, type_t ecltype, size32_t eclsize)
  1012. {
  1013. if (javatype != ecltype || javasize != eclsize)
  1014. throw MakeStringException(0, "javaembed: Type mismatch"); // MORE - could provide some details!
  1015. }
  1016. enum PersistMode
  1017. {
  1018. persistNone,
  1019. persistSupplied,
  1020. persistThread,
  1021. persistChannel,
  1022. persistWorkunit,
  1023. persistQuery,
  1024. persistGlobal
  1025. };
  1026. static PersistMode getPersistMode(const char *val, StringAttr &globalScope)
  1027. {
  1028. StringAttr trimmed;
  1029. const char *colon = strchr(val, ':');
  1030. if (colon)
  1031. {
  1032. globalScope.set(colon+1);
  1033. trimmed.set(val, colon-val);
  1034. val = trimmed;
  1035. }
  1036. if (isEmptyString(val) || strieq(val, "none"))
  1037. return persistNone;
  1038. else if (strieq(val, "thread"))
  1039. return persistThread;
  1040. else if (strieq(val, "channel"))
  1041. return persistChannel;
  1042. else if (strieq(val, "workunit"))
  1043. return persistWorkunit;
  1044. else if (strieq(val, "query"))
  1045. return persistQuery;
  1046. else if (strieq(val, "global"))
  1047. return persistGlobal;
  1048. else
  1049. throw MakeStringException(MSGAUD_user, 0, "javaembed: Unrecognized persist mode %s", val);
  1050. }
  1051. //-------------------------------------------
  1052. // A JavaObject accessor has common functionality shared by both the builders below (Java-> ECL and ECL->Java)
  1053. class JavaObjectAccessor : public CInterface
  1054. {
  1055. protected:
  1056. JavaObjectAccessor(CheckedJNIEnv *_JNIenv, const RtlFieldInfo *_outerRow, jobject _row)
  1057. : JNIenv(_JNIenv), row(_row), outerRow(_outerRow), idx(0), limit(0), inSet(false), inDataSet(false)
  1058. {
  1059. Class = (jclass) JNIenv->NewGlobalRef(JNIenv->GetObjectClass(row), "Class");
  1060. }
  1061. JavaObjectAccessor(CheckedJNIEnv *_JNIenv, const RtlFieldInfo *_outerRow, jclass _Class)
  1062. : JNIenv(_JNIenv), outerRow(_outerRow), idx(0), limit(0), inSet(false), inDataSet(false)
  1063. {
  1064. row = NULL;
  1065. Class = (jclass) JNIenv->NewGlobalRef(_Class, "Class");
  1066. }
  1067. ~JavaObjectAccessor()
  1068. {
  1069. // Unwind anything left on the stack (in case we had exceptions), to make sure the Class we release is the global one
  1070. if (stack.length())
  1071. Class = (jclass) stack.item(0);
  1072. if (Class)
  1073. JNIenv->DeleteGlobalRef(Class);
  1074. }
  1075. void push()
  1076. {
  1077. stack.append(Class);
  1078. stack.append(row);
  1079. }
  1080. void pop()
  1081. {
  1082. row = (jobject) stack.popGet();
  1083. Class = (jclass) stack.popGet();
  1084. }
  1085. jfieldID checkCharField(const RtlFieldInfo * field)
  1086. {
  1087. return JNIenv->GetFieldIDUnchecked(Class, field->name, inSet ? "[C" : "C");
  1088. }
  1089. jfieldID getFieldId(const RtlFieldInfo * field, const char *sig, const char *expected)
  1090. {
  1091. // MORE - if we are going to stream a dataset we really should be caching these somehow
  1092. try
  1093. {
  1094. jfieldID fieldId = 0;
  1095. if (sig)
  1096. {
  1097. if (inSet)
  1098. {
  1099. VStringBuffer arraySig("[%s", sig);
  1100. fieldId = JNIenv->GetFieldID(Class, field->name, arraySig.str());
  1101. }
  1102. else
  1103. fieldId = JNIenv->GetFieldID(Class, field->name, sig);
  1104. }
  1105. else
  1106. {
  1107. // Do it the hard way via reflection API
  1108. // Equivalent java:
  1109. // Field field = object.getClass().getDeclaredField(fieldName);
  1110. jclass classClass =JNIenv->GetObjectClass(Class);
  1111. jmethodID getDeclaredField = JNIenv->GetMethodID(classClass, "getDeclaredField", "(Ljava/lang/String;)Ljava/lang/reflect/Field;" );
  1112. jstring fieldName = JNIenv->NewStringUTF(field->name);
  1113. jobject reflectedField = JNIenv->CallObjectMethod(Class, getDeclaredField, fieldName);
  1114. fieldId = JNIenv->FromReflectedField(reflectedField);
  1115. }
  1116. return fieldId;
  1117. }
  1118. catch (IException *E)
  1119. {
  1120. ::Release(E);
  1121. throw MakeStringException(0, "javaembed: Unable to retrieve field %s of type %s", field->name, expected);
  1122. }
  1123. }
  1124. CheckedJNIEnv *JNIenv;
  1125. jobject row;
  1126. const RtlFieldInfo *outerRow;
  1127. jclass Class;
  1128. ConstPointerArray stack;
  1129. unsigned idx;
  1130. UnsignedArray idxStack;
  1131. unsigned limit;
  1132. bool inSet;
  1133. bool inDataSet;
  1134. };
  1135. // A JavaRowBuilder object is used to construct an ECL row from a Java object
  1136. class JavaRowBuilder : public JavaObjectAccessor, implements IFieldSource
  1137. {
  1138. public:
  1139. IMPLEMENT_IINTERFACE;
  1140. JavaRowBuilder(CheckedJNIEnv *_JNIenv, const RtlFieldInfo *_outerRow, jobject _row)
  1141. : JavaObjectAccessor(_JNIenv, _outerRow, _row)
  1142. {
  1143. }
  1144. virtual bool getBooleanResult(const RtlFieldInfo *field)
  1145. {
  1146. jboolean b;
  1147. if (inSet)
  1148. {
  1149. JNIenv->GetBooleanArrayRegion((jbooleanArray) row, idx, 1, &b);
  1150. }
  1151. else
  1152. {
  1153. jfieldID fieldId = getFieldId(field, "Z", "boolean");
  1154. b = JNIenv->GetBooleanField(row, fieldId);
  1155. }
  1156. return b;
  1157. }
  1158. virtual void getDataResult(const RtlFieldInfo *field, size32_t &__len, void * &__result)
  1159. {
  1160. jbyteArray array;
  1161. if (inSet)
  1162. {
  1163. array = (jbyteArray) JNIenv->GetObjectArrayElement((jobjectArray) row, idx);
  1164. }
  1165. else
  1166. {
  1167. jfieldID fieldId = getFieldId(field, "[B", "DATA");
  1168. array = (jbyteArray) JNIenv->GetObjectField(row, fieldId);
  1169. }
  1170. __len = (array != NULL ? JNIenv->GetArrayLength(array) : 0);
  1171. __result = (__len > 0 ? rtlMalloc(__len) : NULL);
  1172. if (__result)
  1173. JNIenv->GetByteArrayRegion(array, 0, __len, (jbyte *) __result);
  1174. }
  1175. virtual double getRealResult(const RtlFieldInfo *field)
  1176. {
  1177. double d;
  1178. if (inSet)
  1179. {
  1180. float f;
  1181. switch (field->size(NULL, NULL))
  1182. {
  1183. case 4:
  1184. JNIenv->GetFloatArrayRegion((jfloatArray) row, idx, 1, &f);
  1185. d = f;
  1186. break;
  1187. case 8:
  1188. JNIenv->GetDoubleArrayRegion((jdoubleArray) row, idx, 1, &d);
  1189. break;
  1190. default:
  1191. throwUnexpected();
  1192. }
  1193. }
  1194. else
  1195. {
  1196. jfieldID fieldId;
  1197. switch (field->size(NULL, NULL))
  1198. {
  1199. case 4:
  1200. fieldId = getFieldId(field, "F", "float");
  1201. d = JNIenv->GetFloatField(row, fieldId);
  1202. break;
  1203. case 8:
  1204. fieldId = getFieldId(field, "D", "double");
  1205. d = JNIenv->GetDoubleField(row, fieldId);
  1206. break;
  1207. default:
  1208. throwUnexpected();
  1209. }
  1210. }
  1211. return d;
  1212. }
  1213. virtual __int64 getSignedResult(const RtlFieldInfo *field)
  1214. {
  1215. __int64 ret;
  1216. if (inSet)
  1217. {
  1218. jbyte b;
  1219. jshort s;
  1220. jint i;
  1221. jlong l;
  1222. switch (field->size(NULL, NULL))
  1223. {
  1224. case 1:
  1225. JNIenv->GetByteArrayRegion((jbyteArray) row, idx, 1, &b);
  1226. ret = b;
  1227. break;
  1228. case 2:
  1229. JNIenv->GetShortArrayRegion((jshortArray) row, idx, 1, &s);
  1230. ret = s;
  1231. break;
  1232. case 4:
  1233. JNIenv->GetIntArrayRegion((jintArray) row, idx, 1, &i);
  1234. ret = i;
  1235. break;
  1236. case 8:
  1237. JNIenv->GetLongArrayRegion((jlongArray) row, idx, 1, &l);
  1238. ret = l;
  1239. break;
  1240. default:
  1241. UNSUPPORTED("non-standard integer sizes");
  1242. }
  1243. }
  1244. else
  1245. {
  1246. jfieldID fieldId;
  1247. switch (field->size(NULL, NULL))
  1248. {
  1249. case 1:
  1250. fieldId = getFieldId(field, "B", "byte");
  1251. ret = JNIenv->GetByteField(row, fieldId);
  1252. break;
  1253. case 2:
  1254. fieldId = getFieldId(field, "S", "short");
  1255. ret = JNIenv->GetShortField(row, fieldId);
  1256. break;
  1257. case 4:
  1258. fieldId = getFieldId(field, "I", "int");
  1259. ret = JNIenv->GetIntField(row, fieldId);
  1260. break;
  1261. case 8:
  1262. fieldId = getFieldId(field, "J", "long");
  1263. ret = JNIenv->GetLongField(row, fieldId);
  1264. break;
  1265. default:
  1266. UNSUPPORTED("non-standard integer sizes");
  1267. }
  1268. }
  1269. return ret;
  1270. }
  1271. virtual unsigned __int64 getUnsignedResult(const RtlFieldInfo *field)
  1272. {
  1273. UNSUPPORTED("unsigned fields"); // No unsigned types in Java
  1274. }
  1275. virtual void getStringResult(const RtlFieldInfo *field, size32_t &__len, char * &__result)
  1276. {
  1277. jstring result;
  1278. if (inSet)
  1279. {
  1280. // MORE - set of string1 mapping to Java array of char ? Not sure it's worth it.
  1281. result = (jstring) JNIenv->GetObjectArrayElement((jobjectArray) row, idx);
  1282. }
  1283. else
  1284. {
  1285. if (field->isFixedSize() && field->size(NULL, NULL)==1)
  1286. {
  1287. // See if there's a char field
  1288. jfieldID charFieldId = checkCharField(field);
  1289. if (charFieldId)
  1290. {
  1291. jchar resultChar = JNIenv->GetCharField(row, charFieldId);
  1292. rtlUnicodeToStrX(__len, __result, 1, &resultChar);
  1293. return;
  1294. }
  1295. }
  1296. jfieldID fieldId = getFieldId(field, "Ljava/lang/String;", "String");
  1297. result = (jstring) JNIenv->GetObjectField(row, fieldId);
  1298. }
  1299. if (!result)
  1300. {
  1301. __len = 0;
  1302. __result = NULL;
  1303. return;
  1304. }
  1305. size_t size = JNIenv->GetStringUTFLength(result); // in bytes
  1306. const char *text = JNIenv->GetStringUTFChars(result, NULL);
  1307. size32_t chars = rtlUtf8Length(size, text);
  1308. rtlUtf8ToStrX(__len, __result, chars, text);
  1309. JNIenv->ReleaseStringUTFChars(result, text);
  1310. JNIenv->DeleteLocalRef(result);
  1311. }
  1312. virtual void getUTF8Result(const RtlFieldInfo *field, size32_t &__len, char * &__result)
  1313. {
  1314. jstring result;
  1315. if (inSet)
  1316. {
  1317. // MORE - set of string1 mapping to Java array of char ? Not sure it's worth it.
  1318. result = (jstring) JNIenv->GetObjectArrayElement((jobjectArray) row, idx);
  1319. }
  1320. else
  1321. {
  1322. if (field->isFixedSize() && field->size(NULL, NULL)==1)
  1323. {
  1324. // See if there's a char field
  1325. jfieldID charFieldId = checkCharField(field);
  1326. if (charFieldId)
  1327. {
  1328. jchar resultChar = JNIenv->GetCharField(row, charFieldId);
  1329. rtlUnicodeToUtf8X(__len, __result, 1, &resultChar);
  1330. return;
  1331. }
  1332. }
  1333. jfieldID fieldId = getFieldId(field, "Ljava/lang/String;", "String");
  1334. result = (jstring) JNIenv->GetObjectField(row, fieldId);
  1335. }
  1336. if (!result)
  1337. {
  1338. __len = 0;
  1339. __result = NULL;
  1340. return;
  1341. }
  1342. size_t size = JNIenv->GetStringUTFLength(result); // in bytes
  1343. const char *text = JNIenv->GetStringUTFChars(result, NULL);
  1344. size32_t chars = rtlUtf8Length(size, text);
  1345. rtlUtf8ToUtf8X(__len, __result, chars, text);
  1346. JNIenv->ReleaseStringUTFChars(result, text);
  1347. JNIenv->DeleteLocalRef(result);
  1348. }
  1349. virtual void getUnicodeResult(const RtlFieldInfo *field, size32_t &__len, UChar * &__result)
  1350. {
  1351. jstring result;
  1352. if (inSet)
  1353. {
  1354. // MORE - set of string1 mapping to Java array of char ? Not sure it's worth it.
  1355. result = (jstring) JNIenv->GetObjectArrayElement((jobjectArray) row, idx);
  1356. }
  1357. else
  1358. {
  1359. if (field->isFixedSize() && field->size(NULL, NULL)==1)
  1360. {
  1361. // See if there's a char field
  1362. jfieldID charFieldId = checkCharField(field);
  1363. if (charFieldId)
  1364. {
  1365. jchar resultChar = JNIenv->GetCharField(row, charFieldId);
  1366. rtlUnicodeToUnicodeX(__len, __result, 1, &resultChar);
  1367. return;
  1368. }
  1369. }
  1370. jfieldID fieldId = getFieldId(field, "Ljava/lang/String;", "String");
  1371. result = (jstring) JNIenv->GetObjectField(row, fieldId);
  1372. }
  1373. if (!result)
  1374. {
  1375. __len = 0;
  1376. __result = NULL;
  1377. return;
  1378. }
  1379. size_t size = JNIenv->GetStringUTFLength(result); // in bytes
  1380. const char *text = JNIenv->GetStringUTFChars(result, NULL);
  1381. size32_t chars = rtlUtf8Length(size, text);
  1382. rtlUtf8ToUnicodeX(__len, __result, chars, text);
  1383. JNIenv->ReleaseStringUTFChars(result, text);
  1384. JNIenv->DeleteLocalRef(result);
  1385. }
  1386. virtual void getDecimalResult(const RtlFieldInfo *field, Decimal &value)
  1387. {
  1388. double ret = getRealResult(field);
  1389. value.setReal(ret);
  1390. }
  1391. virtual void processBeginSet(const RtlFieldInfo * field, bool &isAll)
  1392. {
  1393. isAll = false; // No concept of an 'all' set in Java
  1394. push();
  1395. jfieldID fieldId = getFieldId(field, NULL, "object"); // We assume it will be an array, but not sure of what...
  1396. row = JNIenv->GetObjectField(row, fieldId);
  1397. inSet = true;
  1398. idx = -1; // First call to next() increments it to 0
  1399. limit = row != NULL ? JNIenv->GetArrayLength((jarray) row) : 0;
  1400. }
  1401. virtual bool processNextSet(const RtlFieldInfo * field)
  1402. {
  1403. assertex(inSet);
  1404. idx++;
  1405. return idx < limit;
  1406. }
  1407. virtual void processBeginDataset(const RtlFieldInfo * field)
  1408. {
  1409. push();
  1410. jfieldID fieldId = getFieldId(field, NULL, "object"); // We assume it will be an array, but not sure of what...
  1411. row = JNIenv->GetObjectField(row, fieldId);
  1412. inDataSet = true;
  1413. idx = -1; // First call to next() increments it to 0
  1414. limit = row != NULL ? JNIenv->GetArrayLength((jarray) row) : 0;
  1415. }
  1416. virtual void processBeginRow(const RtlFieldInfo * field)
  1417. {
  1418. if (field != outerRow)
  1419. {
  1420. push();
  1421. if (inDataSet)
  1422. {
  1423. row = JNIenv->GetObjectArrayElement((jobjectArray) row, idx);
  1424. }
  1425. else
  1426. {
  1427. jfieldID fieldId = getFieldId(field, NULL, "object");
  1428. row = JNIenv->GetObjectField(row, fieldId);
  1429. }
  1430. if (!row)
  1431. rtlFail(0, "javaembed: child dataset object should not be NULL");
  1432. Class = JNIenv->GetObjectClass(row);
  1433. }
  1434. }
  1435. virtual bool processNextRow(const RtlFieldInfo * field)
  1436. {
  1437. assertex(inDataSet);
  1438. idx++;
  1439. return idx < limit;
  1440. }
  1441. virtual void processEndSet(const RtlFieldInfo * field)
  1442. {
  1443. inSet = false;
  1444. JNIenv->DeleteLocalRef(row);
  1445. pop();
  1446. }
  1447. virtual void processEndDataset(const RtlFieldInfo * field)
  1448. {
  1449. inDataSet = false;
  1450. JNIenv->DeleteLocalRef(row);
  1451. pop();
  1452. }
  1453. virtual void processEndRow(const RtlFieldInfo * field)
  1454. {
  1455. if (field != outerRow)
  1456. {
  1457. JNIenv->DeleteLocalRef(row);
  1458. JNIenv->DeleteLocalRef(Class);
  1459. pop();
  1460. }
  1461. }
  1462. };
  1463. //-------------------------------------------
  1464. // A JavaObjectBuilder object is used to construct a Java object from an ECL row
  1465. class JavaObjectBuilder : public JavaObjectAccessor, implements IFieldProcessor
  1466. {
  1467. public:
  1468. IMPLEMENT_IINTERFACE;
  1469. JavaObjectBuilder(CheckedJNIEnv *_JNIenv, const RtlFieldInfo *_outerRow, jclass _Class)
  1470. : JavaObjectAccessor(_JNIenv, _outerRow, _Class)
  1471. {
  1472. setConstructor();
  1473. }
  1474. virtual void processString(unsigned numchars, const char *text, const RtlFieldInfo * field)
  1475. {
  1476. if (field->isFixedSize() && field->size(NULL, NULL)==1 && !inSet) // SET OF STRING1 is not mapped to array of char...
  1477. {
  1478. // See if there's a char field
  1479. jfieldID charFieldId = checkCharField(field);
  1480. if (charFieldId)
  1481. {
  1482. assertex(numchars==1);
  1483. jchar c;
  1484. rtlStrToUnicode(1, &c, 1, text);
  1485. JNIenv->SetCharField(row, charFieldId, c);
  1486. return;
  1487. }
  1488. }
  1489. jfieldID fieldId = getFieldId(field, "Ljava/lang/String;", "String");
  1490. size32_t numchars16;
  1491. rtlDataAttr unicode16;
  1492. rtlStrToUnicodeX(numchars16, unicode16.refustr(), numchars, text);
  1493. jstring value = JNIenv->NewString(unicode16.getustr(), numchars16);
  1494. if (inSet)
  1495. JNIenv->SetObjectArrayElement((jobjectArray) row, idx, value);
  1496. else
  1497. JNIenv->SetObjectField(row, fieldId, value);
  1498. JNIenv->DeleteLocalRef(value);
  1499. }
  1500. virtual void processBool(bool value, const RtlFieldInfo * field)
  1501. {
  1502. jfieldID fieldId = getFieldId(field, "Z", "boolean");
  1503. JNIenv->SetBooleanField(row, fieldId, value);
  1504. }
  1505. virtual void processData(unsigned len, const void *value, const RtlFieldInfo * field)
  1506. {
  1507. jfieldID fieldId = getFieldId(field, "[B", "data");
  1508. jbyteArray javaData = JNIenv->NewByteArray(len);
  1509. JNIenv->SetByteArrayRegion(javaData, 0, len, (jbyte *) value);
  1510. if (inSet)
  1511. JNIenv->SetObjectArrayElement((jobjectArray) row, idx, javaData);
  1512. else
  1513. JNIenv->SetObjectField(row, fieldId, javaData);
  1514. }
  1515. virtual void processInt(__int64 value, const RtlFieldInfo * field)
  1516. {
  1517. jfieldID fieldId;
  1518. switch (field->size(NULL, NULL))
  1519. {
  1520. case 1:
  1521. fieldId = getFieldId(field, "B", "byte");
  1522. JNIenv->SetByteField(row, fieldId, value);
  1523. break;
  1524. case 2:
  1525. fieldId = getFieldId(field, "S", "short");
  1526. JNIenv->SetShortField(row, fieldId, value);
  1527. break;
  1528. case 4:
  1529. fieldId = getFieldId(field, "I", "int");
  1530. JNIenv->SetIntField(row, fieldId, value);
  1531. break;
  1532. case 8:
  1533. fieldId = getFieldId(field, "J", "long");
  1534. JNIenv->SetLongField(row, fieldId, value);
  1535. break;
  1536. default:
  1537. UNSUPPORTED("non-standard integer sizes");
  1538. break;
  1539. }
  1540. }
  1541. virtual void processUInt(unsigned __int64 value, const RtlFieldInfo * field)
  1542. {
  1543. UNSUPPORTED("unsigned fields"); // No unsigned types in Java
  1544. }
  1545. virtual void processReal(double value, const RtlFieldInfo * field)
  1546. {
  1547. jfieldID fieldId;
  1548. switch (field->size(NULL, NULL))
  1549. {
  1550. case 4:
  1551. fieldId = getFieldId(field, "F", "float");
  1552. JNIenv->SetFloatField(row, fieldId, (float) value);
  1553. break;
  1554. case 8:
  1555. fieldId = getFieldId(field, "D", "double");
  1556. JNIenv->SetDoubleField(row, fieldId, value);
  1557. break;
  1558. default:
  1559. throwUnexpected();
  1560. }
  1561. }
  1562. virtual void processDecimal(const void *value, unsigned digits, unsigned precision, const RtlFieldInfo * field)
  1563. {
  1564. // we could map to doubles, but probably better to let the ECL programmer do that themselves
  1565. UNSUPPORTED("DECIMAL fields");
  1566. }
  1567. virtual void processUDecimal(const void *value, unsigned digits, unsigned precision, const RtlFieldInfo * field)
  1568. {
  1569. UNSUPPORTED("UDECIMAL fields");
  1570. }
  1571. virtual void processUnicode(unsigned numchars, const UChar *text, const RtlFieldInfo * field)
  1572. {
  1573. jfieldID fieldId = getFieldId(field, "Ljava/lang/String;", "String");
  1574. jstring value = JNIenv->NewString(text, numchars);
  1575. if (inSet)
  1576. JNIenv->SetObjectArrayElement((jobjectArray) row, idx, value);
  1577. else
  1578. JNIenv->SetObjectField(row, fieldId, value);
  1579. JNIenv->DeleteLocalRef(value);
  1580. }
  1581. virtual void processQString(unsigned len, const char *value, const RtlFieldInfo * field)
  1582. {
  1583. size32_t charCount;
  1584. rtlDataAttr text;
  1585. rtlQStrToStrX(charCount, text.refstr(), len, value);
  1586. processString(charCount, text.getstr(), field);
  1587. }
  1588. virtual void processUtf8(unsigned numchars, const char *text, const RtlFieldInfo * field)
  1589. {
  1590. jfieldID fieldId = getFieldId(field, "Ljava/lang/String;", "String");
  1591. size32_t numchars16;
  1592. rtlDataAttr unicode16;
  1593. rtlUtf8ToUnicodeX(numchars16, unicode16.refustr(), numchars, text);
  1594. jstring value = JNIenv->NewString(unicode16.getustr(), numchars16);
  1595. if (inSet)
  1596. JNIenv->SetObjectArrayElement((jobjectArray) row, idx, value);
  1597. else
  1598. JNIenv->SetObjectField(row, fieldId, value);
  1599. JNIenv->DeleteLocalRef(value);
  1600. }
  1601. virtual bool processBeginSet(const RtlFieldInfo * field, unsigned numElems, bool isAll, const byte *data)
  1602. {
  1603. push();
  1604. idx = 0;
  1605. limit = numElems;
  1606. const char *javaTypeSignature = NULL;
  1607. bool processElements = false;
  1608. // row needs to be created as an array of <whatever>
  1609. if (isAll)
  1610. UNSUPPORTED("ALL sets");
  1611. const RtlTypeInfo *childType = field->type->queryChildType();
  1612. jobject newRow;
  1613. switch(childType->fieldType & RFTMkind)
  1614. {
  1615. case type_boolean:
  1616. newRow = JNIenv->NewBooleanArray(numElems);
  1617. JNIenv->SetBooleanArrayRegion((jbooleanArray) newRow, 0, numElems, (jboolean *) data);
  1618. javaTypeSignature = "[Z";
  1619. break;
  1620. case type_int:
  1621. if (childType->fieldType & RFTMunsigned)
  1622. UNSUPPORTED("unsigned integers");
  1623. switch (childType->length)
  1624. {
  1625. case 1:
  1626. newRow = JNIenv->NewByteArray(numElems);
  1627. JNIenv->SetByteArrayRegion((jbyteArray) newRow, 0, numElems, (jbyte *) data);
  1628. javaTypeSignature = "[B";
  1629. break;
  1630. case 2:
  1631. newRow = JNIenv->NewShortArray(numElems);
  1632. JNIenv->SetShortArrayRegion((jshortArray) newRow, 0, numElems, (jshort *) data);
  1633. javaTypeSignature = "[S";
  1634. break;
  1635. case 4:
  1636. newRow = JNIenv->NewIntArray(numElems);
  1637. JNIenv->SetIntArrayRegion((jintArray) newRow, 0, numElems, (jint *) data);
  1638. javaTypeSignature = "[I";
  1639. break;
  1640. case 8:
  1641. newRow = JNIenv->NewLongArray(numElems);
  1642. JNIenv->SetLongArrayRegion((jlongArray) newRow, 0, numElems, (jlong *) data);
  1643. javaTypeSignature = "[J";
  1644. break;
  1645. default:
  1646. UNSUPPORTED("non-standard integer sizes");
  1647. break;
  1648. }
  1649. break;
  1650. case type_real:
  1651. switch (childType->length)
  1652. {
  1653. case 4:
  1654. newRow = JNIenv->NewFloatArray(numElems);
  1655. JNIenv->SetFloatArrayRegion((jfloatArray) newRow, 0, numElems, (float *) data);
  1656. javaTypeSignature = "[F";
  1657. break;
  1658. case 8:
  1659. newRow = JNIenv->NewDoubleArray(numElems);
  1660. JNIenv->SetDoubleArrayRegion((jdoubleArray) newRow, 0, numElems, (double *) data);
  1661. javaTypeSignature = "[D";
  1662. break;
  1663. default:
  1664. throwUnexpected();
  1665. break;
  1666. }
  1667. break;
  1668. case type_string:
  1669. case type_varstring:
  1670. case type_unicode:
  1671. case type_utf8:
  1672. newRow = JNIenv->NewObjectArray(numElems, langStringClass, NULL);
  1673. javaTypeSignature = "[Ljava/lang/String;";
  1674. processElements = true;
  1675. break;
  1676. case type_data:
  1677. newRow = JNIenv->NewObjectArray(numElems, JNIenv->FindClass("[B"), NULL);
  1678. javaTypeSignature = "[[B";
  1679. processElements = true;
  1680. break;
  1681. default:
  1682. throwUnexpected();
  1683. }
  1684. jfieldID fieldId = getFieldId(field, javaTypeSignature, "Array");
  1685. JNIenv->SetObjectField(row, fieldId, newRow);
  1686. row = newRow;
  1687. inSet = true;
  1688. return processElements;
  1689. }
  1690. virtual bool processBeginDataset(const RtlFieldInfo * field, unsigned numRows)
  1691. {
  1692. push();
  1693. idxStack.append(idx);
  1694. idx = 0;
  1695. inDataSet = true;
  1696. // Create an empty array
  1697. jfieldID childId = getFieldId( field, NULL, "RECORD");
  1698. jobject newRow = NULL;
  1699. if (numRows)
  1700. {
  1701. jclass arrayClass = getClassForChild(childId);
  1702. jmethodID isArrayMethod = JNIenv->GetMethodID(JNIenv->GetObjectClass(arrayClass), "isArray", "()Z" );
  1703. if (!JNIenv->CallBooleanMethod(arrayClass, isArrayMethod))
  1704. {
  1705. JNIenv->ExceptionClear();
  1706. VStringBuffer message("javaembed: Array expected for field %s", field->name);
  1707. rtlFail(0, message.str());
  1708. }
  1709. // Set up constructor etc for the child rows, so we don't do it per row
  1710. jmethodID getTypeMethod = JNIenv->GetMethodID(JNIenv->GetObjectClass(arrayClass), "getComponentType", "()Ljava/lang/Class;" );
  1711. Class = (jclass) JNIenv->CallObjectMethod(arrayClass, getTypeMethod);
  1712. setConstructor();
  1713. // Now we need to create the array
  1714. newRow = JNIenv->NewObjectArray(numRows, Class, NULL);
  1715. }
  1716. JNIenv->SetObjectField(row, childId, newRow);
  1717. row = newRow;
  1718. return true;
  1719. }
  1720. virtual bool processBeginRow(const RtlFieldInfo * field)
  1721. {
  1722. if (field == outerRow)
  1723. row = JNIenv->NewObject(Class, constructor);
  1724. else
  1725. {
  1726. push();
  1727. stack.append(constructor);
  1728. // Now we have to create the child object
  1729. jobject newRow = NULL;
  1730. if (inDataSet)
  1731. {
  1732. newRow = JNIenv->NewObject(Class, constructor);
  1733. JNIenv->SetObjectArrayElement((jobjectArray) row, idx++, newRow);
  1734. }
  1735. else
  1736. {
  1737. // All this is done once per dataset in the nested dataset case. But for embedded record case we have to do it here
  1738. jfieldID childId = getFieldId( field, NULL, "RECORD");
  1739. Class = getClassForChild(childId);
  1740. setConstructor();
  1741. newRow = JNIenv->NewObject(Class, constructor);
  1742. JNIenv->SetObjectField(row, childId, newRow);
  1743. }
  1744. row = newRow;
  1745. }
  1746. return true;
  1747. }
  1748. virtual void processEndSet(const RtlFieldInfo * field)
  1749. {
  1750. JNIenv->DeleteLocalRef(row);
  1751. pop();
  1752. inSet = false;
  1753. }
  1754. virtual void processEndDataset(const RtlFieldInfo * field)
  1755. {
  1756. inDataSet = false;
  1757. idx = idxStack.popGet();
  1758. pop();
  1759. }
  1760. virtual void processEndRow(const RtlFieldInfo * field)
  1761. {
  1762. if (field != outerRow)
  1763. {
  1764. constructor = (jmethodID) stack.popGet();
  1765. JNIenv->DeleteLocalRef(row);
  1766. pop();
  1767. }
  1768. }
  1769. inline jobject getObject()
  1770. {
  1771. return row;
  1772. }
  1773. protected:
  1774. jclass getClassForChild(jfieldID childId)
  1775. {
  1776. jobject reflectedField = JNIenv->ToReflectedField(Class, childId, false);
  1777. jclass fieldClass =JNIenv->GetObjectClass(reflectedField);
  1778. jmethodID getTypeMethod = JNIenv->GetMethodID(fieldClass, "getType", "()Ljava/lang/Class;" );
  1779. jclass result = (jclass) JNIenv->CallObjectMethod(reflectedField, getTypeMethod);
  1780. JNIenv->DeleteLocalRef(reflectedField);
  1781. JNIenv->DeleteLocalRef(fieldClass);
  1782. return result;
  1783. }
  1784. void setConstructor()
  1785. {
  1786. constructor = JNIenv->GetMethodIDUnchecked(Class, "<init>", "()V");
  1787. if (!constructor)
  1788. {
  1789. jmethodID getNameMethod = JNIenv->GetMethodID(JNIenv->GetObjectClass(Class), "getName", "()Ljava/lang/String;" );
  1790. jstring name = (jstring) JNIenv->CallObjectMethod(Class, getNameMethod);
  1791. const char *nameText = JNIenv->GetStringUTFChars(name, NULL);
  1792. VStringBuffer message("javaembed: no suitable constructor for class %s", nameText);
  1793. JNIenv->ReleaseStringUTFChars(name, nameText);
  1794. rtlFail(0, message.str());
  1795. }
  1796. }
  1797. jmethodID constructor;
  1798. };
  1799. //----------------------------------------------------------------------
  1800. // Wrap an IRowStream into a Java Iterator
  1801. class ECLDatasetIterator : public CInterfaceOf<IInterface>
  1802. {
  1803. public:
  1804. ECLDatasetIterator(CheckedJNIEnv *JNIenv, const RtlTypeInfo *_typeInfo, jclass className, IRowStream * _val)
  1805. : typeInfo(_typeInfo), val(_val),
  1806. dummyField("<row>", NULL, typeInfo),
  1807. javaBuilder(JNIenv, &dummyField, className)
  1808. {
  1809. nextRead = false;
  1810. }
  1811. bool hasNext()
  1812. {
  1813. if (!nextRead)
  1814. {
  1815. nextPending.setown(val->ungroupedNextRow());
  1816. nextRead = true;
  1817. if (!nextPending)
  1818. val->stop();
  1819. }
  1820. return nextPending != NULL;
  1821. }
  1822. jobject next()
  1823. {
  1824. if (!hasNext())
  1825. return NULL;
  1826. typeInfo->process((const byte *) nextPending.get(), (const byte *) nextPending.get(), &dummyField, javaBuilder); // Creates a java object from the incoming ECL row
  1827. nextRead = false;
  1828. return javaBuilder.getObject();
  1829. }
  1830. protected:
  1831. const RtlTypeInfo *typeInfo; // Not linked (or linkable)
  1832. Linked<IRowStream> val;
  1833. RtlFieldStrInfo dummyField;
  1834. JavaObjectBuilder javaBuilder;
  1835. roxiemem::OwnedConstRoxieRow nextPending;
  1836. bool nextRead;
  1837. };
  1838. //-------------------------------------------
  1839. // A Java function that returns a dataset will return a JavaRowStream object that can be
  1840. // interrogated to return each row of the result in turn
  1841. class JavaLocalFrame
  1842. {
  1843. public:
  1844. JavaLocalFrame(CheckedJNIEnv *_JNIenv, unsigned size = 16) : JNIenv(_JNIenv)
  1845. {
  1846. JNIenv->PushLocalFrame(size);
  1847. }
  1848. ~JavaLocalFrame()
  1849. {
  1850. JNIenv->PopLocalFrame(NULL);
  1851. }
  1852. private:
  1853. CheckedJNIEnv *JNIenv;
  1854. };
  1855. class JavaRowStream : public CInterfaceOf<IRowStream>
  1856. {
  1857. public:
  1858. JavaRowStream(jobject _iterator, IEngineRowAllocator *_resultAllocator)
  1859. : resultAllocator(_resultAllocator)
  1860. {
  1861. CheckedJNIEnv *JNIenv = queryJNIEnv();
  1862. iterator = JNIenv->NewGlobalRef(_iterator, "iterator");
  1863. iterClass = (jclass) JNIenv->NewGlobalRef(JNIenv->GetObjectClass(iterator), "iterClass");
  1864. hasNextMethod = JNIenv->GetMethodID(iterClass, "hasNext", "()Z" );
  1865. nextMethod = JNIenv->GetMethodID(iterClass, "next", "()Ljava/lang/Object;" );
  1866. // Note that we can't save the JNIEnv value here - calls may be made on different threads (though not at the same time).
  1867. }
  1868. ~JavaRowStream()
  1869. {
  1870. stop();
  1871. }
  1872. virtual const void *nextRow()
  1873. {
  1874. if (!iterator)
  1875. return NULL;
  1876. CheckedJNIEnv *JNIenv = queryJNIEnv();
  1877. JavaLocalFrame lf(JNIenv);
  1878. // Java code would be
  1879. // if (!iterator.hasNext)
  1880. // {
  1881. // stop();
  1882. // return NULL;
  1883. // }
  1884. // result = iterator.next();
  1885. jboolean hasNext = JNIenv->CallBooleanMethod(iterator, hasNextMethod);
  1886. if (!hasNext)
  1887. {
  1888. stop();
  1889. return NULL;
  1890. }
  1891. jobject result = JNIenv->CallObjectMethod(iterator, nextMethod);
  1892. RtlDynamicRowBuilder rowBuilder(resultAllocator);
  1893. const RtlTypeInfo *typeInfo = resultAllocator->queryOutputMeta()->queryTypeInfo();
  1894. assertex(typeInfo);
  1895. RtlFieldStrInfo dummyField("<row>", NULL, typeInfo);
  1896. JavaRowBuilder javaRowBuilder(queryJNIEnv(), &dummyField, result);
  1897. size32_t len = typeInfo->build(rowBuilder, 0, &dummyField, javaRowBuilder);
  1898. return rowBuilder.finalizeRowClear(len);
  1899. }
  1900. virtual void stop()
  1901. {
  1902. resultAllocator.clear();
  1903. CheckedJNIEnv *JNIenv = queryJNIEnv();
  1904. if (JNIenv)
  1905. {
  1906. if (iterator)
  1907. {
  1908. JNIenv->DeleteGlobalRef(iterator);
  1909. iterator = NULL;
  1910. }
  1911. if (iterClass)
  1912. {
  1913. JNIenv->DeleteGlobalRef(iterClass);
  1914. iterClass = NULL;
  1915. }
  1916. hasNextMethod = nullptr;
  1917. nextMethod = nullptr;
  1918. }
  1919. }
  1920. protected:
  1921. Linked<IEngineRowAllocator> resultAllocator;
  1922. jobject iterator = nullptr;
  1923. jclass iterClass = nullptr;
  1924. jmethodID hasNextMethod = nullptr;
  1925. jmethodID nextMethod = nullptr;
  1926. };
  1927. const char *esdl2JavaSig(IEsdlDefinition &esdl, const char *esdlType)
  1928. {
  1929. EsdlBasicElementType t = esdl.translateSimpleType(esdlType);
  1930. switch (t)
  1931. {
  1932. case ESDLT_INT16:
  1933. case ESDLT_UINT16:
  1934. return "Ljava/lang/Short;";
  1935. case ESDLT_INT32:
  1936. case ESDLT_UINT32:
  1937. return "Ljava/lang/Integer;";
  1938. case ESDLT_INT64:
  1939. case ESDLT_UINT64:
  1940. return "Ljava/math/BigInteger;";
  1941. case ESDLT_BOOL:
  1942. return "Ljava/lang/Boolean;";
  1943. case ESDLT_FLOAT:
  1944. return "Ljava/lang/Float;";
  1945. case ESDLT_DOUBLE:
  1946. return "Ljava/lang/Double;";
  1947. case ESDLT_INT8:
  1948. case ESDLT_UINT8:
  1949. case ESDLT_BYTE:
  1950. case ESDLT_UBYTE:
  1951. return "Ljava/lang/Byte;";
  1952. case ESDLT_STRING:
  1953. return "Ljava/lang/String;";
  1954. case ESDLT_UNKOWN:
  1955. case ESDLT_STRUCT:
  1956. case ESDLT_REQUEST:
  1957. case ESDLT_RESPONSE:
  1958. case ESDLT_COMPLEX:
  1959. default:
  1960. return NULL;
  1961. }
  1962. }
  1963. const char *esdl2JavaFullClassName(IEsdlDefinition &esdl, const char *esdlType)
  1964. {
  1965. EsdlBasicElementType t = esdl.translateSimpleType(esdlType);
  1966. switch (t)
  1967. {
  1968. case ESDLT_INT16:
  1969. case ESDLT_UINT16:
  1970. return "java/lang/Short";
  1971. case ESDLT_INT32:
  1972. case ESDLT_UINT32:
  1973. return "java/lang/Integer";
  1974. case ESDLT_INT64:
  1975. case ESDLT_UINT64:
  1976. return "java/math/BigInteger";
  1977. case ESDLT_BOOL:
  1978. return "java/lang/Boolean";
  1979. case ESDLT_FLOAT:
  1980. return "java/lang/Float";
  1981. case ESDLT_DOUBLE:
  1982. return "java/lang/Double";
  1983. case ESDLT_INT8:
  1984. case ESDLT_UINT8:
  1985. case ESDLT_BYTE:
  1986. case ESDLT_UBYTE:
  1987. return "java/lang/Byte";
  1988. case ESDLT_STRING:
  1989. return "java/lang/String";
  1990. case ESDLT_UNKOWN:
  1991. case ESDLT_STRUCT:
  1992. case ESDLT_REQUEST:
  1993. case ESDLT_RESPONSE:
  1994. case ESDLT_COMPLEX:
  1995. default:
  1996. return NULL;
  1997. }
  1998. }
  1999. class JavaObjectXmlWriter : public CInterface
  2000. {
  2001. public:
  2002. JavaObjectXmlWriter(CheckedJNIEnv *_JNIenv, jobject _obj, const char *_reqType, IEsdlDefinition &_esdl, const char *_esdlService, IXmlWriter &_writer)
  2003. : JNIenv(_JNIenv), obj(_obj), writer(_writer), esdl(_esdl), esdlService(_esdlService), reqType(_reqType)
  2004. {
  2005. Class = (jclass) JNIenv->NewGlobalRef(JNIenv->GetObjectClass(obj), "class");
  2006. }
  2007. ~JavaObjectXmlWriter()
  2008. {
  2009. if (Class)
  2010. JNIenv->DeleteGlobalRef(Class);
  2011. HashIterator it(javaClasses);
  2012. ForEach(it)
  2013. {
  2014. IMapping &entry = it.query();
  2015. jclass *pClass = javaClasses.mapToValue(&entry);
  2016. if (pClass)
  2017. JNIenv->DeleteGlobalRef(*pClass);
  2018. }
  2019. }
  2020. void writeSimpleType(const char *fieldname, jobject fieldObj)
  2021. {
  2022. jstring fieldStr = (jstring) JNIenv->CallObjectMethod(fieldObj, object_toString);
  2023. if (!fieldStr)
  2024. return;
  2025. const char *text = JNIenv->GetStringUTFChars(fieldStr, NULL);
  2026. if (text)
  2027. writer.outputCString(text, fieldname);
  2028. JNIenv->ReleaseStringUTFChars(fieldStr, text);
  2029. JNIenv->DeleteLocalRef(fieldStr);
  2030. }
  2031. void writeSimpleType(jclass parentClass, jobject parentObject, const char *fieldname, const char *javaSig)
  2032. {
  2033. if (!fieldname || !*fieldname)
  2034. return;
  2035. if (!javaSig || !*javaSig)
  2036. return;
  2037. jfieldID fieldId = JNIenv->GetFieldID(parentClass, fieldname, javaSig);
  2038. if (!fieldId)
  2039. return;
  2040. jobject fieldObj = (jobject) JNIenv->GetObjectField(parentObject, fieldId);
  2041. if (!fieldObj)
  2042. return;
  2043. writeSimpleType(fieldname, fieldObj);
  2044. JNIenv->DeleteLocalRef(fieldObj);
  2045. }
  2046. void writeSimpleType(jclass parentClass, jobject parentObject, IEsdlDefObject &defObject)
  2047. {
  2048. const char *fieldname = defObject.queryName();
  2049. const char *javaSig = esdl2JavaSig(esdl, defObject.queryProp("type"));
  2050. writeSimpleType(parentClass, parentObject, fieldname, javaSig);
  2051. }
  2052. void writeEnumType(jclass parentClass, jobject parentObject, IEsdlDefObject &defObject)
  2053. {
  2054. const char *fieldname = defObject.queryName();
  2055. VStringBuffer javaSig("L%s/%s;", esdlService.str(), defObject.queryProp("enum_type"));
  2056. jfieldID fieldId = JNIenv->GetFieldID(parentClass, fieldname, javaSig);
  2057. if (!fieldId)
  2058. return;
  2059. jobject fieldObj = (jobject) JNIenv->GetObjectField(parentObject, fieldId);
  2060. if (!fieldObj)
  2061. return;
  2062. jstring fieldStr = (jstring) JNIenv->CallObjectMethod(fieldObj, object_toString);
  2063. const char *text = JNIenv->GetStringUTFChars(fieldStr, NULL);
  2064. if (text)
  2065. writer.outputCString(text, defObject.queryName());
  2066. JNIenv->ReleaseStringUTFChars(fieldStr, text);
  2067. }
  2068. void writeComplexType(jclass parentClass, jobject parentObject, IEsdlDefObject &defObject)
  2069. {
  2070. IEsdlDefStruct *defStruct = esdl.queryStruct(defObject.queryProp("complex_type"));
  2071. if (!defStruct)
  2072. return;
  2073. const char *fieldname = defObject.queryName();
  2074. VStringBuffer javaSig("L%s/%s;", esdlService.str(), defObject.queryProp("complex_type"));
  2075. jfieldID fieldId = JNIenv->GetFieldID(parentClass, fieldname, javaSig); //tbd cache this
  2076. if (!fieldId)
  2077. return;
  2078. jobject fieldObj = (jobject) JNIenv->GetObjectField(parentObject, fieldId);
  2079. if (!fieldObj)
  2080. return;
  2081. writer.outputBeginNested(fieldname, true);
  2082. writeChildren(JNIenv->GetObjectClass(fieldObj), fieldObj, defStruct);
  2083. writer.outputEndNested(fieldname);
  2084. }
  2085. void writeSimpleArray(jobjectArray arrayObj, jint count, const char *name, const char *item_tag)
  2086. {
  2087. writer.outputBeginNested(name, true);
  2088. writer.outputBeginArray(item_tag);
  2089. for (jint i=0; i < count; i++)
  2090. {
  2091. jobject elementObj = JNIenv->GetObjectArrayElement(arrayObj, i);
  2092. writeSimpleType(item_tag, elementObj);
  2093. JNIenv->DeleteLocalRef(elementObj);
  2094. }
  2095. writer.outputEndArray(item_tag);
  2096. writer.outputEndNested(name);
  2097. }
  2098. void writeComplexArray(jobjectArray arrayObj, jint count, const char *name, const char *item_tag, const char *itemTypeName)
  2099. {
  2100. writer.outputBeginNested(name, true);
  2101. writer.outputBeginArray(item_tag);
  2102. {
  2103. VStringBuffer javaClassName("%s/%s", esdlService.str(), itemTypeName);
  2104. jclass elementClass = FindClass(javaClassName);
  2105. if (!elementClass)
  2106. return;
  2107. IEsdlDefStruct *defStruct = esdl.queryStruct(itemTypeName);
  2108. if (!defStruct)
  2109. return;
  2110. for (jint i=0; i < count; i++)
  2111. {
  2112. jobject elementObj = JNIenv->GetObjectArrayElement(arrayObj, i);
  2113. writer.outputBeginNested(item_tag, true);
  2114. writeChildren(elementClass, elementObj, defStruct);
  2115. writer.outputEndNested(item_tag);
  2116. JNIenv->DeleteLocalRef(elementObj);
  2117. }
  2118. }
  2119. writer.outputEndArray(item_tag);
  2120. writer.outputEndNested(name);
  2121. }
  2122. void writeArray(jclass parentClass, jobject parentObject, IEsdlDefObject &defObject)
  2123. {
  2124. const char *itemTypeName = defObject.queryProp("type");
  2125. if (!itemTypeName)
  2126. return;
  2127. const char *item_tag = defObject.queryProp("item_tag");
  2128. if (!item_tag)
  2129. return;
  2130. const char *fieldname = defObject.queryName();
  2131. jfieldID fieldId = JNIenv->GetFieldID(parentClass, fieldname, "Ljava/util/ArrayList;");
  2132. if (!fieldId)
  2133. return;
  2134. jobject arrayListObj = (jobject) JNIenv->GetObjectField(parentObject, fieldId);
  2135. if (!arrayListObj)
  2136. return;
  2137. jobjectArray arrayObj = (jobjectArray) JNIenv->CallObjectMethod(arrayListObj, arrayList_toArray);
  2138. if (arrayObj)
  2139. {
  2140. jint count = JNIenv->GetArrayLength(arrayObj);
  2141. if (count)
  2142. {
  2143. if (esdl2JavaSig(esdl, itemTypeName))
  2144. writeSimpleArray(arrayObj, count, defObject.queryName(), item_tag);
  2145. else
  2146. writeComplexArray(arrayObj, count, defObject.queryName(), item_tag, itemTypeName);
  2147. }
  2148. JNIenv->DeleteLocalRef(arrayObj);
  2149. }
  2150. JNIenv->DeleteLocalRef(arrayListObj);
  2151. }
  2152. void writeChildren(jclass javaClass, jobject javaObject, IEsdlDefStruct *defStruct)
  2153. {
  2154. Owned<IEsdlDefObjectIterator> children = defStruct->getChildren();
  2155. ForEach (*children)
  2156. {
  2157. IEsdlDefObject &child = children->query();
  2158. if (child.getEsdlType()==EsdlTypeElement)
  2159. {
  2160. if (child.hasProp("type"))
  2161. writeSimpleType(javaClass, javaObject, child);
  2162. else if (child.hasProp("complex_type"))
  2163. writeComplexType(javaClass, javaObject, child);
  2164. }
  2165. else if (child.getEsdlType()==EsdlTypeEnumRef)
  2166. {
  2167. writeEnumType(javaClass, javaObject, child);
  2168. }
  2169. else if (child.getEsdlType()==EsdlTypeArray)
  2170. {
  2171. writeArray(javaClass, javaObject, child);
  2172. }
  2173. }
  2174. }
  2175. void write()
  2176. {
  2177. IEsdlDefStruct *reqStruct = esdl.queryStruct(reqType);
  2178. const char *name = reqStruct->queryName();
  2179. writer.outputBeginNested("Response", true);
  2180. writer.outputBeginNested("Results", true);
  2181. writer.outputBeginNested("Result", true);
  2182. writer.outputBeginDataset(name, true);
  2183. writer.outputBeginArray("Row");
  2184. writer.outputBeginNested("Row", true);
  2185. writeChildren(Class, obj, reqStruct);
  2186. writer.outputEndNested("Row");
  2187. writer.outputEndArray("Row");
  2188. writer.outputEndDataset(name);
  2189. writer.outputEndNested("Result");
  2190. writer.outputEndNested("Results");
  2191. writer.outputEndNested("Response");
  2192. }
  2193. jclass FindClass(const char *name)
  2194. {
  2195. jclass *pClass = javaClasses.getValue(name);
  2196. if (pClass)
  2197. return *pClass;
  2198. jclass localClass = JNIenv->FindClass(name);
  2199. if (!localClass)
  2200. return 0;
  2201. jclass Class = (jclass) JNIenv->NewGlobalRef(localClass, "class");
  2202. javaClasses.setValue(name, Class);
  2203. JNIenv->DeleteLocalRef(localClass);
  2204. return Class;
  2205. }
  2206. CheckedJNIEnv *JNIenv;
  2207. MapStringTo<jclass> javaClasses;
  2208. jclass Class;
  2209. jobject obj;
  2210. IXmlWriter &writer;
  2211. IEsdlDefinition &esdl;
  2212. StringAttr reqType;
  2213. StringAttr esdlService;
  2214. };
  2215. //-------------------------------------------
  2216. // There is a singleton JavaThreadContext per thread. This handles the interaction between
  2217. // the C++ thread and the java threading library, ensuring that we register/unregister as needed,
  2218. // and that any thread_local function contexts are destroyed before we detach from the java thread
  2219. interface IJavaEmbedFunctionContext : public IEmbedFunctionContext
  2220. {
  2221. virtual void endThread() = 0;
  2222. };
  2223. class JavaThreadContext
  2224. {
  2225. public:
  2226. CheckedJNIEnv *JNIenv; /* receives pointer to native method interface */
  2227. public:
  2228. JavaThreadContext()
  2229. {
  2230. jint res = globalState->javaVM->AttachCurrentThread((void **) &JNIenv, NULL);
  2231. assertex(res >= 0);
  2232. setThreadClassLoader(getSystemClassLoader());
  2233. }
  2234. ~JavaThreadContext()
  2235. {
  2236. // Make sure all thread-local function contexts and saved objects are destroyed before we detach from
  2237. // the Java thread
  2238. contexts.kill();
  2239. persistedObjects.kill();
  2240. loaders.kill();
  2241. // According to the Java VM 1.7 docs, "A native thread attached to
  2242. // the VM must call DetachCurrentThread() to detach itself before
  2243. // exiting."
  2244. globalState->javaVM->DetachCurrentThread();
  2245. }
  2246. void endThread()
  2247. {
  2248. persistedObjects.kill();
  2249. ForEachItemIn(idx, contexts)
  2250. {
  2251. auto &context = contexts.item(idx);
  2252. context.endThread();
  2253. }
  2254. }
  2255. jobject getSystemClassLoader()
  2256. {
  2257. jobject systemClassLoaderObj = JNIenv->CallStaticObjectMethod(javaLangClassLoaderClass, cl_getSystemClassLoader);
  2258. assertex(systemClassLoaderObj);
  2259. return systemClassLoaderObj;
  2260. }
  2261. void setThreadClassLoader(jobject classLoader)
  2262. {
  2263. jobject threadObj = JNIenv->CallStaticObjectMethod(javaLangThreadClass, thread_currentThread);
  2264. JNIenv->CallObjectMethod(threadObj, thread_setContextClassLoader, classLoader);
  2265. }
  2266. jobject getThreadClassLoader()
  2267. {
  2268. JNIenv->ExceptionClear();
  2269. jobject threadObj = JNIenv->CallStaticObjectMethod(javaLangThreadClass, thread_currentThread);
  2270. jobject contextClassLoaderObj = JNIenv->CallObjectMethod(threadObj, thread_getContextClassLoader);
  2271. assertex(contextClassLoaderObj);
  2272. return contextClassLoaderObj;
  2273. }
  2274. void writeObjectResult(jobject result, IEsdlDefinition *esdl, const char *esdlservice, const char *name, IXmlWriter *writer)
  2275. {
  2276. JavaObjectXmlWriter x(JNIenv, result, name, *esdl, esdlservice, *writer);
  2277. x.write();
  2278. }
  2279. void registerContext(IJavaEmbedFunctionContext *ctx)
  2280. {
  2281. // Note - this object is thread-local so no need for a critsec
  2282. contexts.append(*ctx);
  2283. }
  2284. PersistedObject *getLocalObject(CheckedJNIEnv *JNIenv, const char *name)
  2285. {
  2286. // Note - this object is thread-local so no need for a critsec
  2287. PersistedObject *p;
  2288. p = persistedObjects.find(name);
  2289. if (!p)
  2290. {
  2291. p = new PersistedObject(name);
  2292. persistedObjects.replaceOwn(*p);
  2293. }
  2294. p->crit.enter(); // needed to keep code common between local/global cases
  2295. return p;
  2296. }
  2297. jobject createThreadClassLoader(const char *classPath, const char *classname, size32_t bytecodeLen, const byte *bytecode)
  2298. {
  2299. if (bytecodeLen || (classPath && *classPath))
  2300. {
  2301. jstring jClassPath = (classPath && *classPath) ? JNIenv->NewStringUTF(classPath) : nullptr;
  2302. jobject helperName = JNIenv->NewStringUTF(helperLibraryName);
  2303. jobject contextClassLoaderObj = JNIenv->CallStaticObjectMethod(customLoaderClass, clc_newInstance, jClassPath, getSystemClassLoader(), bytecodeLen, (uint64_t) bytecode, helperName);
  2304. assertex(contextClassLoaderObj);
  2305. return contextClassLoaderObj;
  2306. }
  2307. else
  2308. {
  2309. return getSystemClassLoader();
  2310. }
  2311. }
  2312. jobject getThreadClassLoader(const char *classPath, const char *classname, size32_t bytecodeLen, const byte *bytecode)
  2313. {
  2314. StringBuffer key(classname);
  2315. if (classPath && *classPath)
  2316. key.append('!').append(classPath);
  2317. PersistedObject *p;
  2318. p = loaders.find(key);
  2319. if (!p)
  2320. {
  2321. p = new PersistedObject(key);
  2322. p->instance = JNIenv->NewGlobalRef(createThreadClassLoader(classPath, classname, bytecodeLen, bytecode), "cachedClassLoader");
  2323. loaders.replaceOwn(*p);
  2324. }
  2325. return p->instance;
  2326. }
  2327. private:
  2328. IArrayOf<IJavaEmbedFunctionContext> contexts;
  2329. StringMapOf<PersistedObject> persistedObjects = { false };
  2330. StringMapOf<PersistedObject> loaders = { false };
  2331. };
  2332. class JavaXmlBuilder : implements IXmlWriterExt, public CInterface
  2333. {
  2334. public:
  2335. IMPLEMENT_IINTERFACE;
  2336. JavaXmlBuilder(CheckedJNIEnv *_JNIenv, IEsdlDefinition *esdl_, const char *esdlservice, const char *esdltype_)
  2337. : JNIenv(_JNIenv), esdl(esdl_), javaPackage(esdlservice), esdlType(esdltype_)
  2338. {
  2339. }
  2340. ~JavaXmlBuilder()
  2341. {
  2342. while (defStack.length())
  2343. popDefStackEntry(JNIenv);
  2344. HashIterator it(javaClasses);
  2345. ForEach(it)
  2346. {
  2347. IMapping &entry = it.query();
  2348. jclass *pClass = javaClasses.mapToValue(&entry);
  2349. if (pClass)
  2350. JNIenv->DeleteGlobalRef(*pClass);
  2351. }
  2352. }
  2353. void initWriter()
  2354. {
  2355. }
  2356. IXmlWriterExt & clear()
  2357. {
  2358. throwUnexpected();
  2359. }
  2360. virtual size32_t length() const
  2361. {
  2362. return 0;
  2363. }
  2364. virtual const char *str() const
  2365. {
  2366. throwUnexpected();
  2367. }
  2368. virtual void finalize() override
  2369. {
  2370. }
  2371. virtual void checkDelimiter() override {}
  2372. virtual IInterface *saveLocation() const {return nullptr;}
  2373. virtual void rewindTo(IInterface *loc)
  2374. {
  2375. //needs to be a no-op because it is used, but the way its used to trim empty xml sections I think we're fairly safe.
  2376. //revisit cleaning up any empty objects later.
  2377. }
  2378. inline IEsdlDefStruct *queryCurrentEsdlStruct()
  2379. {
  2380. if (!defStack.length() || !defStack.tos().defType)
  2381. return NULL;
  2382. return dynamic_cast<IEsdlDefStruct*>(defStack.tos().defType.get());
  2383. }
  2384. inline jobject getObject()
  2385. {
  2386. if (!defStack.length())
  2387. return 0;
  2388. return defStack.item(0).obj;
  2389. }
  2390. inline jobject getCurJavaObject()
  2391. {
  2392. if (!defStack.length())
  2393. return 0;
  2394. return defStack.tos().obj;
  2395. }
  2396. inline jclass getCurJavaClass()
  2397. {
  2398. if (!defStack.length())
  2399. return 0;
  2400. return defStack.tos().Class;
  2401. }
  2402. inline jmethodID getCurJavaConstructor()
  2403. {
  2404. if (!defStack.length())
  2405. return 0;
  2406. return defStack.tos().constructor;
  2407. }
  2408. virtual void outputEnumString(unsigned size, const char *text, const char *fieldname, IEsdlDefObject *defField)
  2409. {
  2410. const char *enum_type = defField->queryProp("enum_type");
  2411. if (!enum_type || !*enum_type)
  2412. return;
  2413. VStringBuffer enumClassName("%s/%s", javaPackage.str(), enum_type);
  2414. VStringBuffer enumSig("L%s;", enumClassName.str());
  2415. jfieldID fieldId = JNIenv->GetFieldID(getCurJavaClass(), fieldname, enumSig);
  2416. if (!fieldId)
  2417. return;
  2418. jclass enumClass = FindClass(enumClassName);
  2419. jmethodID fromString = JNIenv->GetStaticMethodID(enumClass, "fromString", "(Ljava/lang/String;)LEsdlExample/AddressType;"); //All types currently used for ESDL mapping have string constructors
  2420. StringAttr s(text, size);
  2421. jstring strvalue = JNIenv->NewStringUTF(s);
  2422. jobject value = JNIenv->CallStaticObjectMethod(enumClass, fromString, strvalue);
  2423. JNIenv->DeleteLocalRef(strvalue);
  2424. JNIenv->SetObjectField(getCurJavaObject(), fieldId, value);
  2425. JNIenv->DeleteLocalRef(value);
  2426. }
  2427. virtual void outputString(unsigned size, const char *text, const char *fieldname)
  2428. {
  2429. DefStackEntry *parent = defStack.length() ? &defStack.tos() : NULL;
  2430. if (!parent)
  2431. return;
  2432. const char *defTypeName = NULL;
  2433. bool isArray = (parent->defObj && parent->defObj->getEsdlType()==EsdlTypeArray);
  2434. if (isArray)
  2435. defTypeName = parent->defObj->queryProp("type");
  2436. else
  2437. {
  2438. IEsdlDefStruct *defStruct = queryCurrentEsdlStruct();
  2439. if (!defStruct)
  2440. return;
  2441. IEsdlDefObject *defField = defStruct->queryChild(fieldname);
  2442. if (!defField)
  2443. return;
  2444. if (defField->getEsdlType()==EsdlTypeEnumRef)
  2445. return outputEnumString(size, text, fieldname, defField);
  2446. defTypeName = defField->queryProp("type");
  2447. }
  2448. if (!defTypeName)
  2449. return;
  2450. const char *javaSig = esdl2JavaSig(*esdl, defTypeName);
  2451. if (!javaSig)
  2452. return;
  2453. const char *fieldClassName = esdl2JavaFullClassName(*esdl, defTypeName);
  2454. jclass typeClass = FindClass(fieldClassName);
  2455. jmethodID typeStringConstructor = JNIenv->GetMethodID(typeClass, "<init>", "(Ljava/lang/String;)V"); //All types currently used for ESDL mapping have string constructors
  2456. StringAttr s(text, size);
  2457. jstring strvalue = JNIenv->NewStringUTF(s);
  2458. jobject value = JNIenv->NewObject(typeClass, typeStringConstructor, strvalue);
  2459. JNIenv->DeleteLocalRef(strvalue);
  2460. if (!value)
  2461. return;
  2462. if (isArray)
  2463. JNIenv->CallObjectMethod(parent->obj, parent->append, value);
  2464. else
  2465. {
  2466. jfieldID fieldId = JNIenv->GetFieldID(getCurJavaClass(), fieldname, javaSig);
  2467. if (fieldId)
  2468. JNIenv->SetObjectField(getCurJavaObject(), fieldId, value);
  2469. }
  2470. JNIenv->DeleteLocalRef(value);
  2471. }
  2472. void outputString(const char *text, const char *fieldname)
  2473. {
  2474. outputString((unsigned)strlen(text), text, fieldname);
  2475. }
  2476. virtual void outputNumericString(const char *field, const char *fieldname)
  2477. {
  2478. outputString(field, fieldname);
  2479. }
  2480. virtual void outputBool(bool value, const char *fieldname)
  2481. {
  2482. outputString(value ? "true" : "false", fieldname);
  2483. }
  2484. virtual void outputUInt(unsigned __int64 field, unsigned size, const char *fieldname)
  2485. {
  2486. StringBuffer value;
  2487. value.append(field);
  2488. outputString(value.length(), value, fieldname);
  2489. }
  2490. virtual void outputInt(__int64 field, unsigned size, const char *fieldname)
  2491. {
  2492. StringBuffer value;
  2493. value.append(field);
  2494. outputString(value.length(), value, fieldname);
  2495. }
  2496. virtual void outputReal(double field, const char *fieldname)
  2497. {
  2498. StringBuffer value;
  2499. value.append(field);
  2500. outputString(value.length(), value, fieldname);
  2501. }
  2502. virtual void outputDecimal(const void *field, unsigned size, unsigned precision, const char *fieldname)
  2503. {
  2504. Decimal d;
  2505. d.setDecimal(size, precision, field);
  2506. outputString(d.getCString(), fieldname);
  2507. }
  2508. virtual void outputUDecimal(const void *field, unsigned size, unsigned precision, const char *fieldname)
  2509. {
  2510. Decimal d;
  2511. d.setUDecimal(size, precision, field);
  2512. outputString(d.getCString(), fieldname);
  2513. }
  2514. virtual void outputQString(unsigned len, const char *field, const char *fieldname)
  2515. {
  2516. MemoryAttr tempBuffer;
  2517. char * temp;
  2518. if (len <= 100)
  2519. temp = (char *)alloca(len);
  2520. else
  2521. temp = (char *)tempBuffer.allocate(len);
  2522. rtlQStrToStr(len, temp, len, field);
  2523. outputString(len, temp, fieldname);
  2524. }
  2525. virtual void outputUnicode(unsigned len, const UChar *field, const char *fieldname)
  2526. {
  2527. char * buff = 0;
  2528. unsigned bufflen = 0;
  2529. rtlUnicodeToCodepageX(bufflen, buff, len, field, "utf-8");
  2530. outputString(bufflen, buff, fieldname);
  2531. rtlFree(buff);
  2532. }
  2533. virtual void outputUtf8(unsigned len, const char *field, const char *fieldname)
  2534. {
  2535. outputString(len, field, fieldname);
  2536. }
  2537. virtual void outputData(unsigned len, const void *value, const char *fieldname)
  2538. {
  2539. }
  2540. virtual void outputQuoted(const char *text) //would have to let beginNested represent simple types with content set using this?
  2541. {
  2542. }
  2543. virtual void outputBeginDataset(const char *dsname, bool nestChildren) //not used by ESDL engine
  2544. {
  2545. }
  2546. virtual void outputEndDataset(const char *dsname)
  2547. {
  2548. }
  2549. inline IEsdlDefObject *queryChildStructDefObj(IEsdlDefObject *child)
  2550. {
  2551. if (child)
  2552. {
  2553. switch (child->getEsdlType())
  2554. {
  2555. case EsdlTypeArray:
  2556. {
  2557. const char *structType = child->queryProp("type");
  2558. if (structType)
  2559. return esdl->queryObj(structType);
  2560. break;
  2561. }
  2562. case EsdlTypeElement:
  2563. {
  2564. const char *structType = child->queryProp("complex_type");
  2565. if (structType)
  2566. return esdl->queryObj(structType);
  2567. break;
  2568. }
  2569. default:
  2570. break;
  2571. }
  2572. }
  2573. return NULL;
  2574. }
  2575. virtual void outputBeginNested(const char *fieldname, bool nestChildren)
  2576. {
  2577. IEsdlDefStruct *defStruct = NULL;
  2578. IEsdlDefObject *defField = NULL;
  2579. IEsdlDefObject *defType = NULL;
  2580. if (!defStack.length())
  2581. {
  2582. defType = esdl->queryObj(fieldname);
  2583. }
  2584. else
  2585. {
  2586. DefStackEntry &parent = defStack.tos();
  2587. if (parent.defObj && parent.defObj->getEsdlType()==EsdlTypeArray)
  2588. {
  2589. defType = parent.defType;
  2590. }
  2591. else
  2592. {
  2593. defStruct = queryCurrentEsdlStruct();
  2594. if (defStruct)
  2595. {
  2596. defField = defStruct->queryChild(fieldname);
  2597. if (defField)
  2598. defType = queryChildStructDefObj(defField);
  2599. }
  2600. }
  2601. }
  2602. pushDefStackEntry(JNIenv, javaPackage, fieldname, defType, defField);
  2603. }
  2604. virtual void outputEndNested(const char *fieldname)
  2605. {
  2606. if (defStack.length()<=1) //don't destroy root object yet
  2607. return;
  2608. if (!streq(fieldname, defStack.tos().name)) //should be exception? or forgive and forget?
  2609. return;
  2610. popDefStackEntry(JNIenv);
  2611. }
  2612. virtual void outputSetAll()
  2613. {
  2614. }
  2615. virtual void outputBeginArray(const char *fieldname)
  2616. {
  2617. }
  2618. virtual void outputEndArray(const char *fieldname)
  2619. {
  2620. }
  2621. virtual void outputInlineXml(const char *text)
  2622. {
  2623. }
  2624. virtual void outputXmlns(const char *name, const char *uri)
  2625. {
  2626. }
  2627. virtual void cutFrom(IInterface *location, StringBuffer& databuf)
  2628. {
  2629. }
  2630. virtual void outputInline(const char* text)
  2631. {
  2632. }
  2633. virtual void flushContent(bool close)
  2634. {
  2635. };
  2636. public:
  2637. CheckedJNIEnv *JNIenv;
  2638. Linked<IEsdlDefinition> esdl;
  2639. StringAttr javaPackage;
  2640. StringAttr esdlType;
  2641. class DefStackEntry : public CInterface
  2642. {
  2643. public:
  2644. DefStackEntry(const char *fieldname, IEsdlDefObject *_defType, IEsdlDefObject *_defObj) : name(fieldname), defType(_defType), defObj(_defObj), Class(0), obj(0), constructor(0), append(0), fieldId(0)
  2645. {
  2646. }
  2647. ~DefStackEntry()
  2648. {
  2649. }
  2650. public:
  2651. Linked<IEsdlDefObject> defType;
  2652. Linked<IEsdlDefObject> defObj;
  2653. StringAttr name;
  2654. jclass Class;
  2655. jmethodID constructor;
  2656. jmethodID append;
  2657. jfieldID fieldId;
  2658. jobject obj;
  2659. };
  2660. jobject MakeObjectGlobal(jobject local)
  2661. {
  2662. if (!local)
  2663. return 0;
  2664. jobject global = JNIenv->NewGlobalRef(local, "makeObjectGlobal");
  2665. JNIenv->DeleteLocalRef(local);
  2666. return global;
  2667. }
  2668. jclass FindClass(const char *name)
  2669. {
  2670. jclass *pClass = javaClasses.getValue(name);
  2671. if (pClass)
  2672. return *pClass;
  2673. jclass Class = (jclass) MakeObjectGlobal(JNIenv->FindClass(name));
  2674. javaClasses.setValue(name, Class); //even save if result has no class
  2675. return Class;
  2676. }
  2677. void popDefStackEntry(CheckedJNIEnv *JNIenv)
  2678. {
  2679. if (!defStack.length())
  2680. return;
  2681. Owned<DefStackEntry> entry = &defStack.popGet();
  2682. if (entry->obj)
  2683. JNIenv->DeleteGlobalRef(entry->obj);
  2684. }
  2685. void pushDefStackEntry(CheckedJNIEnv *JNIenv, const char *package, const char *fieldname, IEsdlDefObject *defType, IEsdlDefObject *defObject)
  2686. {
  2687. DefStackEntry *parent = defStack.length() ? &defStack.tos() : NULL;
  2688. Owned<DefStackEntry> entry = new DefStackEntry(fieldname, defType, defObject);
  2689. JNIenv->ExceptionClear();
  2690. if (defObject && defObject->getEsdlType()==EsdlTypeArray)
  2691. {
  2692. entry->Class = arrayListClass;
  2693. entry->constructor = arrayList_constructor;
  2694. entry->append = arrayList_add;
  2695. entry->obj = MakeObjectGlobal(JNIenv->NewObject(entry->Class, entry->constructor));
  2696. if (entry->obj)
  2697. {
  2698. if (parent && parent->Class)
  2699. {
  2700. entry->fieldId = JNIenv->GetFieldID(parent->Class, fieldname, "Ljava/util/ArrayList;");
  2701. if (parent->obj && entry->fieldId)
  2702. JNIenv->SetObjectField(parent->obj, entry->fieldId, entry->obj);
  2703. }
  2704. }
  2705. }
  2706. else if (defType)
  2707. {
  2708. VStringBuffer javaClassName("%s/%s", package, defType->queryName());
  2709. entry->Class = FindClass(javaClassName);
  2710. if (entry->Class)
  2711. {
  2712. entry->constructor = JNIenv->GetMethodID(entry->Class, "<init>", "()V");
  2713. entry->obj = MakeObjectGlobal(JNIenv->NewObject(entry->Class, entry->constructor));
  2714. if (entry->obj)
  2715. {
  2716. if (parent)
  2717. {
  2718. if (parent->defObj && parent->defObj->getEsdlType()==EsdlTypeArray)
  2719. JNIenv->CallObjectMethod(parent->obj, parent->append, entry->obj);
  2720. else if (parent->Class)
  2721. {
  2722. VStringBuffer javaSig("L%s;", javaClassName.str());
  2723. entry->fieldId = JNIenv->GetFieldID(parent->Class, fieldname, javaSig);
  2724. if (parent->obj && entry->fieldId)
  2725. JNIenv->SetObjectField(parent->obj, entry->fieldId, entry->obj);
  2726. }
  2727. }
  2728. }
  2729. }
  2730. }
  2731. defStack.append(*entry.getClear());
  2732. }
  2733. CIArrayOf<DefStackEntry> defStack;
  2734. MapStringTo<jclass> javaClasses;
  2735. };
  2736. // Each call to a Java function will use a new JavaEmbedScriptContext object
  2737. #define MAX_JNI_ARGS 10
  2738. class JavaClassReader
  2739. {
  2740. public:
  2741. JavaClassReader(const char *filename)
  2742. {
  2743. // Pull apart a class file to see its name and signature.
  2744. /* From https://docs.oracle.com/javase/specs/jvms/se7/html/jvms-4.html#jvms-4.1
  2745. ClassFile {
  2746. u4 magic;
  2747. u2 minor_version;
  2748. u2 major_version;
  2749. u2 constant_pool_count;
  2750. cp_info constant_pool[constant_pool_count-1];
  2751. u2 access_flags;
  2752. u2 this_class;
  2753. u2 super_class;
  2754. u2 interfaces_count;
  2755. u2 interfaces[interfaces_count];
  2756. u2 fields_count;
  2757. field_info fields[fields_count];
  2758. u2 methods_count;
  2759. method_info methods[methods_count];
  2760. u2 attributes_count;
  2761. attribute_info attributes[attributes_count];
  2762. }
  2763. */
  2764. #ifdef TRACE_CLASSFILE
  2765. DBGLOG("Reading class file created in %s", filename);
  2766. #endif
  2767. Owned<IFile> file = createIFile(filename);
  2768. OwnedIFileIO io = file->open(IFOread);
  2769. assertex(io);
  2770. read(io, 0, (size32_t)-1, b);
  2771. b.setEndian(__BIG_ENDIAN);
  2772. uint32_t magic;
  2773. b.read(magic);
  2774. if (magic != 0xcafebabe)
  2775. throwUnexpected();
  2776. uint16_t major, minor, cpc;
  2777. b.read(major);
  2778. b.read(minor);
  2779. b.read(cpc);
  2780. constOffsets = new unsigned[cpc];
  2781. constOffsets[0] = 0;
  2782. for (int i = 0; i < cpc-1; i++) // There are only cpc-1 entries, for reasons best known to the java designers
  2783. {
  2784. constOffsets[i+1] = b.getPos();
  2785. byte tag;
  2786. b.read(tag);
  2787. switch (tag)
  2788. {
  2789. case CONSTANT_Class:
  2790. uint16_t idx;
  2791. b.read(idx);
  2792. #ifdef TRACE_CLASSFILE
  2793. DBGLOG("%u: Class %u", i+1, idx);
  2794. #endif
  2795. break;
  2796. case CONSTANT_Fieldref:
  2797. case CONSTANT_Methodref:
  2798. case CONSTANT_InterfaceMethodref:
  2799. uint16_t classIdx;
  2800. uint16_t nametypeIdx;
  2801. b.read(classIdx);
  2802. b.read(nametypeIdx);
  2803. #ifdef TRACE_CLASSFILE
  2804. DBGLOG("%u: ref(%u) class %u nametype %u", i+1, tag, classIdx, nametypeIdx);
  2805. #endif
  2806. break;
  2807. case CONSTANT_String:
  2808. #ifdef TRACE_CLASSFILE
  2809. DBGLOG("%u: Tag %u", i+1, tag);
  2810. #endif
  2811. b.skip(2);
  2812. break;
  2813. case CONSTANT_Integer:
  2814. case CONSTANT_Float:
  2815. #ifdef TRACE_CLASSFILE
  2816. DBGLOG("%u: Tag %u", i+1, tag);
  2817. #endif
  2818. b.skip(4);
  2819. break;
  2820. case CONSTANT_Long:
  2821. case CONSTANT_Double:
  2822. #ifdef TRACE_CLASSFILE
  2823. DBGLOG("%u: Tag %u", i+1, tag);
  2824. #endif
  2825. b.skip(8);
  2826. break;
  2827. case CONSTANT_NameAndType:
  2828. uint16_t nameIdx;
  2829. uint16_t descIdx;
  2830. b.read(nameIdx);
  2831. b.read(descIdx);
  2832. #ifdef TRACE_CLASSFILE
  2833. DBGLOG("%u: NameAndType(%u) name %u desc %u", i+1, tag, nameIdx, descIdx);
  2834. #endif
  2835. break;
  2836. case CONSTANT_Utf8:
  2837. // length-prefixed
  2838. uint16_t length;
  2839. b.read(length);
  2840. const byte *val;
  2841. val = b.readDirect(length);
  2842. #ifdef TRACE_CLASSFILE
  2843. DBGLOG("%u: %.*s", i+1, length, val);
  2844. #endif
  2845. break;
  2846. case CONSTANT_MethodHandle:
  2847. #ifdef TRACE_CLASSFILE
  2848. DBGLOG("%u: Tag %u", i+1, tag);
  2849. #endif
  2850. b.skip(3);
  2851. break;
  2852. case CONSTANT_MethodType:
  2853. #ifdef TRACE_CLASSFILE
  2854. DBGLOG("%u: Tag %u", i+1, tag);
  2855. #endif
  2856. b.skip(2);
  2857. break;
  2858. case CONSTANT_InvokeDynamic:
  2859. #ifdef TRACE_CLASSFILE
  2860. DBGLOG("%u: Tag %u", i+1, tag);
  2861. #endif
  2862. b.skip(4);
  2863. break;
  2864. default:
  2865. DBGLOG("Unexpected tag %u reading bytecode file", tag);
  2866. throwUnexpected();
  2867. }
  2868. }
  2869. uint16_t access_flags; b.read(access_flags);
  2870. uint16_t this_class; b.read(this_class);
  2871. uint16_t super_class; b.read(super_class);
  2872. uint16_t interfaces_count; b.read(interfaces_count);
  2873. b.skip(interfaces_count*sizeof(uint16_t));
  2874. uint16_t fields_count; b.read(fields_count);
  2875. #ifdef TRACE_CLASSFILE
  2876. DBGLOG("Access flags %x this_class=%u super_class=%u interfaces_count=%u fields_count=%u", access_flags, this_class, super_class, interfaces_count, fields_count);
  2877. #endif
  2878. for (unsigned i = 0; i < fields_count; i++)
  2879. {
  2880. b.skip(6);
  2881. uint16_t attr_count;
  2882. b.read(attr_count);
  2883. for (unsigned j = 0; j < attr_count; j++)
  2884. {
  2885. b.skip(2);
  2886. uint32_t attr_length;
  2887. b.read(attr_length);
  2888. b.skip(attr_length);
  2889. }
  2890. }
  2891. uint16_t methods_count; b.read(methods_count);
  2892. #ifdef TRACE_CLASSFILE
  2893. DBGLOG("methods_count %u", methods_count);
  2894. #endif
  2895. for (unsigned i = 0; i < methods_count; i++)
  2896. {
  2897. uint16_t flags; b.read(flags);
  2898. uint16_t name; b.read(name);
  2899. uint16_t desc; b.read(desc);
  2900. #ifdef TRACE_CLASSFILE
  2901. DBGLOG("Method %u name %u desc %u flags %x", i, name, desc, flags);
  2902. #endif
  2903. if (flags & ACC_PUBLIC)
  2904. {
  2905. StringAttr thisName;
  2906. readUtf(thisName, name);
  2907. StringAttr thisSig;
  2908. readUtf(thisSig, desc);
  2909. methodNames.append(thisName);
  2910. methodSigs.append(thisSig);
  2911. methodFlags.append(flags);
  2912. }
  2913. uint16_t attr_count;
  2914. b.read(attr_count);
  2915. for (unsigned j = 0; j < attr_count; j++)
  2916. {
  2917. uint16_t attr_name_idx; b.read(attr_name_idx);
  2918. StringAttr attrName;
  2919. readUtf(attrName, attr_name_idx);
  2920. uint32_t attr_length;
  2921. b.read(attr_length);
  2922. if (streq(attrName, "Signature") && attr_length==2)
  2923. {
  2924. uint16_t ext_sig_idx; b.read(ext_sig_idx);
  2925. StringAttr extSig;
  2926. readUtf(extSig, ext_sig_idx);
  2927. #ifdef TRACE_CLASSFILE
  2928. DBGLOG("Seen extended signature %s", extSig.str());
  2929. #endif
  2930. if (flags & ACC_PUBLIC)
  2931. {
  2932. methodSigs.pop();
  2933. methodSigs.append(extSig);
  2934. }
  2935. }
  2936. else
  2937. b.skip(attr_length);
  2938. }
  2939. }
  2940. /* Don't bother reading attributes as they are not really interesting to us
  2941. uint16_t attributes_count; b.read(attributes_count);
  2942. #ifdef TRACE_CLASSFILE
  2943. DBGLOG("attributes_count %u", attributes_count);
  2944. #endif
  2945. for (unsigned i = 0; i < attributes_count; i++)
  2946. {
  2947. b.skip(2);
  2948. uint32_t attr_length;
  2949. b.read(attr_length);
  2950. b.skip(attr_length);
  2951. }
  2952. #ifdef TRACE_CLASSFILE
  2953. DBGLOG("%u of %u bytes remaining", b.remaining(), b.length());
  2954. #endif
  2955. */
  2956. // Now we can find this class name
  2957. readTag(this_class, CONSTANT_Class);
  2958. readUtf(className, readIdx());
  2959. }
  2960. ~JavaClassReader()
  2961. {
  2962. delete [] constOffsets;
  2963. }
  2964. StringBuffer & getSignature(StringBuffer &ret, unsigned idx) const
  2965. {
  2966. if (!methodNames.isItem(idx))
  2967. throw makeStringException(0, "No public static method found");
  2968. ret.appendf("%s.%s:", className.get(), methodNames.item(idx));
  2969. if ((methodFlags[idx] & ACC_STATIC) == 0)
  2970. ret.append('@');
  2971. return ret.append(methodSigs.item(idx));
  2972. }
  2973. const char *queryClassName() const
  2974. {
  2975. return className.get();
  2976. }
  2977. MemoryBuffer &getEmbedData(MemoryBuffer &result, const char *funcName, bool mainClass) const
  2978. {
  2979. result.setEndian(__BIG_ENDIAN);
  2980. StringBuffer signature;
  2981. if (mainClass)
  2982. {
  2983. unsigned methodIdx = getFunctionIdx(funcName);
  2984. getSignature(signature, methodIdx);
  2985. }
  2986. else
  2987. signature.set(className);
  2988. result.append((size32_t) signature.length());
  2989. result.append(signature.length(), signature.str());
  2990. result.append((size32_t) b.length());
  2991. result.append(b);
  2992. return result;
  2993. }
  2994. enum access_flag : uint16_t
  2995. {
  2996. ACC_PUBLIC = 0x0001, // Declared public; may be accessed from outside its package.
  2997. ACC_PRIVATE = 0x0002, // Declared private; accessible only within the defining class.
  2998. ACC_PROTECTED = 0x0004, // Declared protected; may be accessed within subclasses.
  2999. ACC_STATIC = 0x0008, // Declared static.
  3000. ACC_FINAL = 0x0010, // Declared final; must not be overridden (§5.4.5).
  3001. ACC_SYNCHRONIZED = 0x0020, // Declared synchronized; invocation is wrapped by a monitor use.
  3002. ACC_BRIDGE = 0x0040, // A bridge method, generated by the compiler.
  3003. ACC_VARARGS = 0x0080, // Declared with variable number of arguments.
  3004. ACC_NATIVE = 0x0100, // Declared native; implemented in a language other than Java.
  3005. ACC_ABSTRACT = 0x0400, // Declared abstract; no implementation is provided.
  3006. ACC_STRICT = 0x0800, // Declared strictfp; floating-point mode is FP-strict.
  3007. ACC_SYNTHETIC = 0x1000, // Declared synthetic; not present in the source code.
  3008. };
  3009. unsigned getFlags(const char *funcName) const
  3010. {
  3011. unsigned idx = getFunctionIdx(funcName);
  3012. return methodFlags[idx];
  3013. }
  3014. private:
  3015. bool isConstructor(const char *name) const
  3016. {
  3017. const char *shortClass = strrchr(className, '/');
  3018. if (shortClass)
  3019. shortClass++;
  3020. else
  3021. shortClass = className;
  3022. return streq(shortClass, name);
  3023. }
  3024. unsigned getFunctionIdx(const char *funcName) const
  3025. {
  3026. if (isConstructor(funcName))
  3027. funcName = "<init>";
  3028. unsigned methodIdx = (unsigned) -1;
  3029. ForEachItemIn(idx, methodNames)
  3030. {
  3031. if (streq(funcName, methodNames[idx]))
  3032. {
  3033. if (methodIdx != (unsigned) -1)
  3034. throw makeStringExceptionV(0, "Embedded java has multiple public methods called %s", funcName);
  3035. methodIdx = idx;
  3036. }
  3037. }
  3038. if (methodIdx == (unsigned) -1)
  3039. throw makeStringExceptionV(0, "Embedded java should export a public method %s", funcName);
  3040. return methodIdx;
  3041. }
  3042. uint16_t readIdx()
  3043. {
  3044. uint16_t idx;
  3045. b.read(idx);
  3046. return idx;
  3047. }
  3048. void readTag(unsigned idx, byte expected)
  3049. {
  3050. b.reset(constOffsets[idx]);
  3051. byte tag;
  3052. b.read(tag);
  3053. assertex(tag == expected);
  3054. }
  3055. void readUtf(StringAttr &dest, unsigned idx)
  3056. {
  3057. auto savepos = b.getPos();
  3058. readTag(idx, CONSTANT_Utf8);
  3059. uint16_t length;
  3060. b.read(length);
  3061. dest.set((const char *) b.readDirect(length), length);
  3062. b.reset(savepos);
  3063. }
  3064. enum const_type
  3065. {
  3066. CONSTANT_Class = 7,
  3067. CONSTANT_Fieldref = 9,
  3068. CONSTANT_Methodref = 10,
  3069. CONSTANT_InterfaceMethodref = 11,
  3070. CONSTANT_String = 8,
  3071. CONSTANT_Integer = 3,
  3072. CONSTANT_Float = 4,
  3073. CONSTANT_Long = 5,
  3074. CONSTANT_Double = 6,
  3075. CONSTANT_NameAndType = 12,
  3076. CONSTANT_Utf8 = 1,
  3077. CONSTANT_MethodHandle = 15,
  3078. CONSTANT_MethodType = 16,
  3079. CONSTANT_InvokeDynamic = 18
  3080. };
  3081. MemoryBuffer b;
  3082. unsigned *constOffsets = nullptr;
  3083. StringAttr className;
  3084. StringArray methodNames;
  3085. StringArray methodSigs;
  3086. UnsignedArray methodFlags;
  3087. };
  3088. // Objects of class JavaEmbedImportContext are created locally for each call of a function, or thread-local to persist from one call to the next.
  3089. // Methods in here do not need to be thread-safe
  3090. class JavaEmbedImportContext : public CInterfaceOf<IJavaEmbedFunctionContext>
  3091. {
  3092. public:
  3093. JavaEmbedImportContext(ICodeContext *codeCtx, JavaThreadContext *_sharedCtx, jobject _instance, unsigned _flags, const char *options, const IThorActivityContext *_activityContext)
  3094. : sharedCtx(_sharedCtx), JNIenv(sharedCtx->JNIenv), instance(_instance), flags(_flags), activityContext(_activityContext)
  3095. {
  3096. argcount = 0;
  3097. argsig = NULL;
  3098. nonStatic = (instance != nullptr);
  3099. javaClass = nullptr;
  3100. StringArray opts;
  3101. opts.appendList(options, ",");
  3102. if (codeCtx)
  3103. {
  3104. engine = codeCtx->queryEngineContext();
  3105. try {
  3106. nodeNum = codeCtx->getNodeNum();
  3107. }
  3108. catch (IException *E)
  3109. {
  3110. E->Release(); // We may get an error if calling on the master - we just want to ignore it
  3111. }
  3112. }
  3113. StringBuffer lclassPath;
  3114. if (engine)
  3115. {
  3116. const StringArray &manifestJars = engine->queryManifestFiles("jar");
  3117. ForEachItemIn(idx, manifestJars)
  3118. {
  3119. lclassPath.append(';').append(manifestJars.item(idx));
  3120. }
  3121. }
  3122. ForEachItemIn(idx, opts)
  3123. {
  3124. const char *opt = opts.item(idx);
  3125. const char *val = strchr(opt, '=');
  3126. if (val)
  3127. {
  3128. StringBuffer optName(val-opt, opt);
  3129. val++;
  3130. if (stricmp(optName, "classpath")==0)
  3131. lclassPath.append(';').append(val);
  3132. else if (strieq(optName, "globalscope"))
  3133. globalScopeKey.set(val);
  3134. else if (strieq(optName, "persist"))
  3135. {
  3136. if (persistMode != persistNone)
  3137. throw MakeStringException(MSGAUD_user, 0, "javaembed: Persist option specified more than once");
  3138. persistMode = getPersistMode(val, globalScopeKey);
  3139. switch (persistMode)
  3140. {
  3141. case persistChannel:
  3142. case persistWorkunit:
  3143. case persistQuery:
  3144. if (!engine)
  3145. throw MakeStringException(MSGAUD_user, 0, "javaembed: Persist mode '%s' not supported here", val);
  3146. break;
  3147. }
  3148. }
  3149. else
  3150. throw MakeStringException(0, "javaembed: Unknown option %s", optName.str());
  3151. }
  3152. }
  3153. if (lclassPath.length()>1)
  3154. classpath.set(lclassPath.str()+1);
  3155. if (flags & EFthreadlocal)
  3156. sharedCtx->registerContext(this); // Do at end - otherwise an exception thrown during construction will leave this reference dangling
  3157. }
  3158. ~JavaEmbedImportContext()
  3159. {
  3160. if (javaClass)
  3161. JNIenv->DeleteGlobalRef(javaClass);
  3162. if (classLoader)
  3163. JNIenv->DeleteGlobalRef(classLoader);
  3164. }
  3165. virtual void endThread() override
  3166. {
  3167. instance = nullptr;
  3168. if (javaClass)
  3169. {
  3170. JNIenv->DeleteGlobalRef(javaClass);
  3171. javaClass = nullptr;
  3172. }
  3173. if (classLoader)
  3174. {
  3175. JNIenv->DeleteGlobalRef(classLoader);
  3176. classLoader = nullptr;
  3177. }
  3178. javaMethodID = nullptr;
  3179. }
  3180. virtual bool getBooleanResult()
  3181. {
  3182. switch (*returnType)
  3183. {
  3184. case 'Z': return result.z;
  3185. case 'L':
  3186. {
  3187. // Result should be of class 'Boolean'
  3188. if (!result.l)
  3189. return false;
  3190. jmethodID getVal = JNIenv->GetMethodID(JNIenv->GetObjectClass(result.l), "booleanValue", "()Z");
  3191. if (!getVal)
  3192. throw MakeStringException(MSGAUD_user, 0, "javaembed: Type mismatch on result");
  3193. bool ret=JNIenv->CallBooleanMethod(result.l, getVal);
  3194. return ret;
  3195. }
  3196. default:
  3197. throw MakeStringException(MSGAUD_user, 0, "javaembed: Type mismatch on result");
  3198. }
  3199. }
  3200. virtual void getDataResult(size32_t &__len, void * &__result)
  3201. {
  3202. if (strcmp(returnType, "[B")!=0)
  3203. throw resultMismatchException("data");
  3204. jbyteArray array = (jbyteArray) result.l;
  3205. __len = (array != NULL ? JNIenv->GetArrayLength(array) : 0);
  3206. __result = (__len > 0 ? rtlMalloc(__len) : NULL);
  3207. if (__result)
  3208. JNIenv->GetByteArrayRegion(array, 0, __len, (jbyte *) __result);
  3209. }
  3210. virtual double getRealResult()
  3211. {
  3212. switch (*returnType)
  3213. {
  3214. case 'D': return result.d;
  3215. case 'F': return result.f;
  3216. case 'L':
  3217. {
  3218. // Result should be of class 'Number'
  3219. if (!result.l)
  3220. return 0;
  3221. jmethodID getVal = JNIenv->GetMethodID(JNIenv->GetObjectClass(result.l), "doubleValue", "()D");
  3222. if (!getVal)
  3223. throw resultMismatchException("real");
  3224. double ret = JNIenv->CallDoubleMethod(result.l, getVal);
  3225. return ret;
  3226. }
  3227. default:
  3228. throw resultMismatchException("real");
  3229. }
  3230. }
  3231. virtual __int64 getSignedResult()
  3232. {
  3233. switch (*returnType)
  3234. {
  3235. case 'B': return result.b;
  3236. case 'S': return result.s;
  3237. case 'I': return result.i;
  3238. case 'J': return result.j;
  3239. case 'L':
  3240. {
  3241. // Result should be of class 'Number'
  3242. if (!result.l)
  3243. return 0;
  3244. jmethodID getVal = JNIenv->GetMethodID(JNIenv->GetObjectClass(result.l), "longValue", "()J");
  3245. if (!getVal)
  3246. throw resultMismatchException("integer");
  3247. __int64 ret = JNIenv->CallLongMethod(result.l, getVal);
  3248. return ret;
  3249. }
  3250. default:
  3251. throw resultMismatchException("integer");
  3252. }
  3253. }
  3254. virtual unsigned __int64 getUnsignedResult()
  3255. {
  3256. if (*returnType=='V' && strieq(methodName, "<init>"))
  3257. return (unsigned __int64) result.l;
  3258. if (*returnType=='L' && JNIenv->IsSameObject(result.l, instance) && persistMode != persistNone)
  3259. return (unsigned __int64) instance;
  3260. StringBuffer s;
  3261. throw makeStringExceptionV(MSGAUD_user, 0, "javaembed: In method %s: Unsigned results not supported", getReportName(s).str()); // Java doesn't support unsigned
  3262. }
  3263. virtual void getStringResult(size32_t &__len, char * &__result)
  3264. {
  3265. switch (*returnType)
  3266. {
  3267. case 'C': // Single char returned, prototyped as STRING or STRING1 in ECL
  3268. rtlUnicodeToStrX(__len, __result, 1, &result.c);
  3269. break;
  3270. case 'L':
  3271. {
  3272. jstring sresult = (jstring) result.l;
  3273. if (sresult)
  3274. {
  3275. size_t size = JNIenv->GetStringUTFLength(sresult); // in bytes
  3276. const char *text = JNIenv->GetStringUTFChars(sresult, NULL);
  3277. size32_t chars = rtlUtf8Length(size, text);
  3278. rtlUtf8ToStrX(__len, __result, chars, text);
  3279. JNIenv->ReleaseStringUTFChars(sresult, text);
  3280. }
  3281. else
  3282. {
  3283. __len = 0;
  3284. __result = NULL;
  3285. }
  3286. break;
  3287. }
  3288. default:
  3289. throw resultMismatchException("string");
  3290. }
  3291. }
  3292. virtual void getUTF8Result(size32_t &__chars, char * &__result)
  3293. {
  3294. switch (*returnType)
  3295. {
  3296. case 'C': // Single jchar returned, prototyped as UTF8 in ECL
  3297. rtlUnicodeToUtf8X(__chars, __result, 1, &result.c);
  3298. break;
  3299. case 'L':
  3300. {
  3301. jstring sresult = (jstring) result.l;
  3302. if (sresult)
  3303. {
  3304. size_t size = JNIenv->GetStringUTFLength(sresult); // Returns length in bytes (not chars)
  3305. const char * text = JNIenv->GetStringUTFChars(sresult, NULL);
  3306. rtlUtf8ToUtf8X(__chars, __result, rtlUtf8Length(size, text), text);
  3307. JNIenv->ReleaseStringUTFChars(sresult, text);
  3308. }
  3309. else
  3310. {
  3311. __chars = 0;
  3312. __result = NULL;
  3313. }
  3314. break;
  3315. }
  3316. default:
  3317. throw resultMismatchException("utf8");
  3318. }
  3319. }
  3320. virtual void getUnicodeResult(size32_t &__chars, UChar * &__result)
  3321. {
  3322. switch (*returnType)
  3323. {
  3324. case 'C': // Single jchar returned, prototyped as UNICODE or UNICODE1 in ECL
  3325. rtlUnicodeToUnicodeX(__chars, __result, 1, &result.c);
  3326. break;
  3327. case 'L':
  3328. {
  3329. jstring sresult = (jstring) result.l;
  3330. if (sresult)
  3331. {
  3332. size_t size = JNIenv->GetStringUTFLength(sresult); // in bytes
  3333. const char *text = JNIenv->GetStringUTFChars(sresult, NULL);
  3334. size32_t chars = rtlUtf8Length(size, text);
  3335. rtlUtf8ToUnicodeX(__chars, __result, chars, text);
  3336. JNIenv->ReleaseStringUTFChars(sresult, text);
  3337. }
  3338. else
  3339. {
  3340. __chars = 0;
  3341. __result = NULL;
  3342. }
  3343. break;
  3344. }
  3345. default:
  3346. throw resultMismatchException("unicode");
  3347. }
  3348. }
  3349. virtual void getSetResult(bool & __isAllResult, size32_t & __resultBytes, void * & __result, int _elemType, size32_t elemSize)
  3350. {
  3351. if (*returnType != '[')
  3352. throw resultMismatchException("array");
  3353. type_t elemType = (type_t) _elemType;
  3354. jarray array = (jarray) result.l;
  3355. int numResults = (array != NULL ? JNIenv->GetArrayLength(array) : 0);
  3356. rtlRowBuilder out;
  3357. byte *outData = NULL;
  3358. size32_t outBytes = 0;
  3359. if (numResults > 0)
  3360. {
  3361. if (elemSize != UNKNOWN_LENGTH)
  3362. {
  3363. out.ensureAvailable(numResults * elemSize); // MORE - check for overflow?
  3364. outData = out.getbytes();
  3365. }
  3366. switch(returnType[1])
  3367. {
  3368. case 'Z':
  3369. checkType(type_boolean, sizeof(jboolean), elemType, elemSize);
  3370. JNIenv->GetBooleanArrayRegion((jbooleanArray) array, 0, numResults, (jboolean *) outData);
  3371. break;
  3372. case 'B':
  3373. checkType(type_int, sizeof(jbyte), elemType, elemSize);
  3374. JNIenv->GetByteArrayRegion((jbyteArray) array, 0, numResults, (jbyte *) outData);
  3375. break;
  3376. case 'C':
  3377. // we COULD map to a set of string1, but is there any point?
  3378. {
  3379. StringBuffer s;
  3380. throw MakeStringException(0, "javaembed: In method %s: Return type mismatch (char[] not supported)", getReportName(s).str());
  3381. break;
  3382. }
  3383. case 'S':
  3384. checkType(type_int, sizeof(jshort), elemType, elemSize);
  3385. JNIenv->GetShortArrayRegion((jshortArray) array, 0, numResults, (jshort *) outData);
  3386. break;
  3387. case 'I':
  3388. checkType(type_int, sizeof(jint), elemType, elemSize);
  3389. JNIenv->GetIntArrayRegion((jintArray) array, 0, numResults, (jint *) outData);
  3390. break;
  3391. case 'J':
  3392. checkType(type_int, sizeof(jlong), elemType, elemSize);
  3393. JNIenv->GetLongArrayRegion((jlongArray) array, 0, numResults, (jlong *) outData);
  3394. break;
  3395. case 'F':
  3396. checkType(type_real, sizeof(jfloat), elemType, elemSize);
  3397. JNIenv->GetFloatArrayRegion((jfloatArray) array, 0, numResults, (jfloat *) outData);
  3398. break;
  3399. case 'D':
  3400. checkType(type_real, sizeof(jdouble), elemType, elemSize);
  3401. JNIenv->GetDoubleArrayRegion((jdoubleArray) array, 0, numResults, (jdouble *) outData);
  3402. break;
  3403. case 'L':
  3404. if (strcmp(returnType, "[Ljava/lang/String;") == 0)
  3405. {
  3406. for (int i = 0; i < numResults; i++)
  3407. {
  3408. jstring elem = (jstring) JNIenv->GetObjectArrayElement((jobjectArray) array, i);
  3409. size_t lenBytes = JNIenv->GetStringUTFLength(elem); // in bytes
  3410. const char *text = JNIenv->GetStringUTFChars(elem, NULL);
  3411. switch (elemType)
  3412. {
  3413. case type_string:
  3414. if (elemSize == UNKNOWN_LENGTH)
  3415. {
  3416. out.ensureAvailable(outBytes + lenBytes + sizeof(size32_t));
  3417. outData = out.getbytes() + outBytes;
  3418. * (size32_t *) outData = lenBytes;
  3419. rtlStrToStr(lenBytes, outData+sizeof(size32_t), lenBytes, text);
  3420. outBytes += lenBytes + sizeof(size32_t);
  3421. }
  3422. else
  3423. rtlStrToStr(elemSize, outData, lenBytes, text);
  3424. break;
  3425. case type_varstring:
  3426. if (elemSize == UNKNOWN_LENGTH)
  3427. {
  3428. out.ensureAvailable(outBytes + lenBytes + 1);
  3429. outData = out.getbytes() + outBytes;
  3430. rtlStrToVStr(0, outData, lenBytes, text);
  3431. outBytes += lenBytes + 1;
  3432. }
  3433. else
  3434. rtlStrToVStr(elemSize, outData, lenBytes, text); // Fixed size null terminated strings... weird.
  3435. break;
  3436. case type_utf8:
  3437. case type_unicode:
  3438. {
  3439. size32_t numchars = rtlUtf8Length(lenBytes, text);
  3440. if (elemType == type_utf8)
  3441. {
  3442. assertex (elemSize == UNKNOWN_LENGTH);
  3443. out.ensureAvailable(outBytes + lenBytes + sizeof(size32_t));
  3444. outData = out.getbytes() + outBytes;
  3445. * (size32_t *) outData = numchars;
  3446. rtlStrToStr(lenBytes, outData+sizeof(size32_t), lenBytes, text);
  3447. outBytes += lenBytes + sizeof(size32_t);
  3448. }
  3449. else
  3450. {
  3451. if (elemSize == UNKNOWN_LENGTH)
  3452. {
  3453. // You can't assume that number of chars in utf8 matches number in unicode16 ...
  3454. size32_t numchars16;
  3455. rtlDataAttr unicode16;
  3456. rtlUtf8ToUnicodeX(numchars16, unicode16.refustr(), numchars, text);
  3457. out.ensureAvailable(outBytes + numchars16*sizeof(UChar) + sizeof(size32_t));
  3458. outData = out.getbytes() + outBytes;
  3459. * (size32_t *) outData = numchars16;
  3460. rtlUnicodeToUnicode(numchars16, (UChar *) (outData+sizeof(size32_t)), numchars16, unicode16.getustr());
  3461. outBytes += numchars16*sizeof(UChar) + sizeof(size32_t);
  3462. }
  3463. else
  3464. rtlUtf8ToUnicode(elemSize / sizeof(UChar), (UChar *) outData, numchars, text);
  3465. }
  3466. break;
  3467. }
  3468. default:
  3469. JNIenv->ReleaseStringUTFChars(elem, text);
  3470. StringBuffer s;
  3471. throw MakeStringException(0, "javaembed: In method %s: Return type mismatch (ECL string type expected)", getReportName(s).str());
  3472. }
  3473. JNIenv->ReleaseStringUTFChars(elem, text);
  3474. JNIenv->DeleteLocalRef(elem);
  3475. if (elemSize != UNKNOWN_LENGTH)
  3476. outData += elemSize;
  3477. }
  3478. }
  3479. else
  3480. {
  3481. StringBuffer s;
  3482. throw MakeStringException(0, "javaembed: In method %s: Return type mismatch (%s[] not supported)", getReportName(s).str(), returnType+2);
  3483. }
  3484. break;
  3485. }
  3486. }
  3487. __isAllResult = false;
  3488. __resultBytes = elemSize == UNKNOWN_LENGTH ? outBytes : elemSize * numResults;
  3489. __result = out.detachdata();
  3490. }
  3491. virtual IRowStream *getDatasetResult(IEngineRowAllocator * _resultAllocator)
  3492. {
  3493. jclass iterClass =JNIenv->GetObjectClass(result.l);
  3494. if (!JNIenv->IsAssignableFrom(iterClass, utilIteratorClass))
  3495. {
  3496. if (JNIenv->IsAssignableFrom(iterClass, langIterableClass))
  3497. {
  3498. result.l = JNIenv->CallObjectMethod(result.l, iterable_iterator);
  3499. }
  3500. else
  3501. {
  3502. StringBuffer s;
  3503. throw MakeStringException(0, "javaembed: In method %s: Java code should return an iterator or iterable object", getReportName(s).str());
  3504. }
  3505. }
  3506. return new JavaRowStream(result.l, _resultAllocator);
  3507. }
  3508. virtual byte * getRowResult(IEngineRowAllocator * _resultAllocator)
  3509. {
  3510. RtlDynamicRowBuilder rowBuilder(_resultAllocator);
  3511. size32_t len = getRowResult(result.l, rowBuilder);
  3512. return (byte *) rowBuilder.finalizeRowClear(len);
  3513. }
  3514. virtual size32_t getTransformResult(ARowBuilder & builder)
  3515. {
  3516. return getRowResult(result.l, builder);
  3517. }
  3518. virtual void bindBooleanParam(const char *name, bool val)
  3519. {
  3520. if (*argsig != 'Z')
  3521. typeError("BOOLEAN");
  3522. argsig++;
  3523. jvalue v;
  3524. v.z = val;
  3525. addArg(v);
  3526. }
  3527. virtual void bindDataParam(const char *name, size32_t len, const void *val)
  3528. {
  3529. if (argsig[0] != '[' || argsig[1] != 'B')
  3530. typeError("DATA");
  3531. argsig += 2;
  3532. jvalue v;
  3533. jbyteArray javaData = JNIenv->NewByteArray(len);
  3534. JNIenv->SetByteArrayRegion(javaData, 0, len, (jbyte *) val);
  3535. v.l = javaData;
  3536. addArg(v);
  3537. }
  3538. virtual void bindFloatParam(const char *name, float val)
  3539. {
  3540. // Could argue that the size should match...
  3541. jvalue v;
  3542. switch(*argsig)
  3543. {
  3544. case 'D':
  3545. v.d = val;
  3546. break;
  3547. case 'F':
  3548. v.f = val;
  3549. break;
  3550. default:
  3551. typeError("REAL");
  3552. break;
  3553. }
  3554. argsig++;
  3555. addArg(v);
  3556. }
  3557. virtual void bindRealParam(const char *name, double val)
  3558. {
  3559. jvalue v;
  3560. switch(*argsig)
  3561. {
  3562. case 'D':
  3563. v.d = val;
  3564. break;
  3565. case 'F':
  3566. v.f = val;
  3567. break;
  3568. default:
  3569. typeError("REAL");
  3570. break;
  3571. }
  3572. argsig++;
  3573. addArg(v);
  3574. }
  3575. virtual void bindSignedSizeParam(const char *name, int size, __int64 val)
  3576. {
  3577. bindSignedParam(name, val);
  3578. }
  3579. virtual void bindSignedParam(const char *name, __int64 val)
  3580. {
  3581. jvalue v;
  3582. switch(*argsig)
  3583. {
  3584. case 'I':
  3585. v.i = val;
  3586. break;
  3587. case 'J':
  3588. v.j = val;
  3589. break;
  3590. case 'S':
  3591. v.s = val;
  3592. break;
  3593. case 'B':
  3594. v.b = val;
  3595. break;
  3596. default:
  3597. typeError("INTEGER");
  3598. break;
  3599. }
  3600. argsig++;
  3601. addArg(v);
  3602. }
  3603. virtual void bindUnsignedSizeParam(const char *name, int size, unsigned __int64 val)
  3604. {
  3605. StringBuffer s;
  3606. throw MakeStringException(MSGAUD_user, 0, "javaembed: In method %s: Unsigned parameters not supported", getReportName(s).str()); // Java doesn't support unsigned
  3607. }
  3608. virtual void bindUnsignedParam(const char *name, unsigned __int64 val)
  3609. {
  3610. if (!strchr(importName, '.') && argcount==0) // Could require a flag, or a special parameter name...
  3611. {
  3612. if (!val)
  3613. {
  3614. StringBuffer s;
  3615. throw MakeStringException(MSGAUD_user, 0, "javaembed: In method %s: Null value passed for \"this\"", getReportName(s).str());
  3616. }
  3617. instance = (jobject) val;
  3618. persistMode = persistSupplied;
  3619. if (JNIenv->GetObjectRefType(instance) != JNIGlobalRefType)
  3620. {
  3621. StringBuffer s;
  3622. throw MakeStringException(MSGAUD_user, 0, "javaembed: In method %s: Invalid value passed for \"this\"", getReportName(s).str());
  3623. }
  3624. jclass newJavaClass = JNIenv->GetObjectClass(instance);
  3625. if (!JNIenv->IsSameObject(newJavaClass, javaClass))
  3626. {
  3627. if (javaClass)
  3628. {
  3629. JNIenv->DeleteGlobalRef(javaClass);
  3630. javaClass = nullptr;
  3631. }
  3632. if (classLoader)
  3633. {
  3634. JNIenv->DeleteGlobalRef(classLoader);
  3635. classLoader = nullptr;
  3636. }
  3637. loadFunction(classpath, 0, nullptr);
  3638. }
  3639. reinit();
  3640. }
  3641. else
  3642. {
  3643. // We could match a java class, to allow objects returned from one embed to be passed as parameters to another
  3644. StringBuffer s;
  3645. throw MakeStringException(MSGAUD_user, 0, "javaembed: In method %s: Unsigned parameters not supported", getReportName(s).str()); // Java doesn't support unsigned
  3646. }
  3647. }
  3648. virtual void bindStringParam(const char *name, size32_t len, const char *val)
  3649. {
  3650. jvalue v;
  3651. switch(*argsig)
  3652. {
  3653. case 'C':
  3654. rtlStrToUnicode(1, &v.c, len, val);
  3655. argsig++;
  3656. break;
  3657. case 'L':
  3658. if (strncmp(argsig, "Ljava/lang/String;", 18) == 0)
  3659. {
  3660. argsig += 18;
  3661. unsigned unicodeChars;
  3662. UChar *unicode;
  3663. rtlStrToUnicodeX(unicodeChars, unicode, len, val);
  3664. v.l = JNIenv->NewString(unicode, unicodeChars);
  3665. rtlFree(unicode);
  3666. break;
  3667. }
  3668. // fall into ...
  3669. default:
  3670. typeError("STRING");
  3671. break;
  3672. }
  3673. addArg(v);
  3674. }
  3675. virtual void bindVStringParam(const char *name, const char *val)
  3676. {
  3677. bindStringParam(name, strlen(val), val);
  3678. }
  3679. virtual void bindUTF8Param(const char *name, size32_t numchars, const char *val)
  3680. {
  3681. jvalue v;
  3682. switch(*argsig)
  3683. {
  3684. case 'C':
  3685. rtlUtf8ToUnicode(1, &v.c, numchars, val);
  3686. argsig++;
  3687. break;
  3688. case 'L':
  3689. if (strncmp(argsig, "Ljava/lang/String;", 18) == 0)
  3690. {
  3691. argsig += 18;
  3692. unsigned unicodeChars;
  3693. UChar *unicode;
  3694. rtlUtf8ToUnicodeX(unicodeChars, unicode, numchars, val);
  3695. v.l = JNIenv->NewString(unicode, unicodeChars);
  3696. rtlFree(unicode);
  3697. break;
  3698. }
  3699. // fall into ...
  3700. default:
  3701. typeError("UTF8");
  3702. break;
  3703. }
  3704. addArg(v);
  3705. }
  3706. virtual void bindUnicodeParam(const char *name, size32_t numchars, const UChar *val)
  3707. {
  3708. jvalue v;
  3709. switch(*argsig)
  3710. {
  3711. case 'C':
  3712. rtlUnicodeToUnicode(1, &v.c, numchars, val);
  3713. argsig++;
  3714. break;
  3715. case 'L':
  3716. if (strncmp(argsig, "Ljava/lang/String;", 18) == 0)
  3717. {
  3718. argsig += 18;
  3719. v.l = JNIenv->NewString(val, numchars);
  3720. break;
  3721. }
  3722. // fall into ...
  3723. default:
  3724. typeError("UNICODE");
  3725. break;
  3726. }
  3727. addArg(v);
  3728. }
  3729. virtual void bindSetParam(const char *name, int _elemType, size32_t elemSize, bool isAll, size32_t totalBytes, const void *setData)
  3730. {
  3731. jvalue v;
  3732. if (*argsig != '[')
  3733. typeError("SET");
  3734. argsig++;
  3735. type_t elemType = (type_t) _elemType;
  3736. int numElems = totalBytes / elemSize;
  3737. switch(*argsig)
  3738. {
  3739. case 'Z':
  3740. checkType(type_boolean, sizeof(jboolean), elemType, elemSize);
  3741. v.l = JNIenv->NewBooleanArray(numElems);
  3742. JNIenv->SetBooleanArrayRegion((jbooleanArray) v.l, 0, numElems, (jboolean *) setData);
  3743. break;
  3744. case 'B':
  3745. checkType(type_int, sizeof(jbyte), elemType, elemSize);
  3746. v.l = JNIenv->NewByteArray(numElems);
  3747. JNIenv->SetByteArrayRegion((jbyteArray) v.l, 0, numElems, (jbyte *) setData);
  3748. break;
  3749. case 'C':
  3750. // we COULD map to a set of string1, but is there any point?
  3751. typeError("");
  3752. break;
  3753. case 'S':
  3754. checkType(type_int, sizeof(jshort), elemType, elemSize);
  3755. v.l = JNIenv->NewShortArray(numElems);
  3756. JNIenv->SetShortArrayRegion((jshortArray) v.l, 0, numElems, (jshort *) setData);
  3757. break;
  3758. case 'I':
  3759. checkType(type_int, sizeof(jint), elemType, elemSize);
  3760. v.l = JNIenv->NewIntArray(numElems);
  3761. JNIenv->SetIntArrayRegion((jintArray) v.l, 0, numElems, (jint *) setData);
  3762. break;
  3763. case 'J':
  3764. checkType(type_int, sizeof(jlong), elemType, elemSize);
  3765. v.l = JNIenv->NewLongArray(numElems);
  3766. JNIenv->SetLongArrayRegion((jlongArray) v.l, 0, numElems, (jlong *) setData);
  3767. break;
  3768. case 'F':
  3769. checkType(type_real, sizeof(jfloat), elemType, elemSize);
  3770. v.l = JNIenv->NewFloatArray(numElems);
  3771. JNIenv->SetFloatArrayRegion((jfloatArray) v.l, 0, numElems, (jfloat *) setData);
  3772. break;
  3773. case 'D':
  3774. checkType(type_real, sizeof(jdouble), elemType, elemSize);
  3775. v.l = JNIenv->NewDoubleArray(numElems);
  3776. JNIenv->SetDoubleArrayRegion((jdoubleArray) v.l, 0, numElems, (jdouble *) setData);
  3777. break;
  3778. case 'L':
  3779. if (strncmp(argsig, "Ljava/lang/String;", 18) == 0)
  3780. {
  3781. argsig += 17; // Yes, 17, because we increment again at the end of the case
  3782. const byte *inData = (const byte *) setData;
  3783. const byte *endData = inData + totalBytes;
  3784. if (elemSize == UNKNOWN_LENGTH)
  3785. {
  3786. numElems = 0;
  3787. // Will need 2 passes to work out how many elements there are in the set :(
  3788. while (inData < endData)
  3789. {
  3790. int thisSize;
  3791. switch (elemType)
  3792. {
  3793. case type_varstring:
  3794. thisSize = strlen((const char *) inData) + 1;
  3795. break;
  3796. case type_string:
  3797. thisSize = * (size32_t *) inData + sizeof(size32_t);
  3798. break;
  3799. case type_unicode:
  3800. thisSize = (* (size32_t *) inData) * sizeof(UChar) + sizeof(size32_t);
  3801. break;
  3802. case type_utf8:
  3803. thisSize = rtlUtf8Size(* (size32_t *) inData, inData + sizeof(size32_t)) + sizeof(size32_t);;
  3804. break;
  3805. default:
  3806. typeError("STRING");
  3807. }
  3808. inData += thisSize;
  3809. numElems++;
  3810. }
  3811. inData = (const byte *) setData;
  3812. }
  3813. int idx = 0;
  3814. v.l = JNIenv->NewObjectArray(numElems, langStringClass, NULL);
  3815. while (inData < endData)
  3816. {
  3817. jstring thisElem;
  3818. size32_t thisSize = elemSize;
  3819. switch (elemType)
  3820. {
  3821. case type_varstring:
  3822. {
  3823. size32_t numChars = strlen((const char *) inData);
  3824. unsigned unicodeChars;
  3825. rtlDataAttr unicode;
  3826. rtlStrToUnicodeX(unicodeChars, unicode.refustr(), numChars, (const char *) inData);
  3827. thisElem = JNIenv->NewString(unicode.getustr(), unicodeChars);
  3828. if (elemSize == UNKNOWN_LENGTH)
  3829. thisSize = numChars + 1;
  3830. break;
  3831. }
  3832. case type_string:
  3833. {
  3834. if (elemSize == UNKNOWN_LENGTH)
  3835. {
  3836. thisSize = * (size32_t *) inData;
  3837. inData += sizeof(size32_t);
  3838. }
  3839. unsigned unicodeChars;
  3840. rtlDataAttr unicode;
  3841. rtlStrToUnicodeX(unicodeChars, unicode.refustr(), thisSize, (const char *) inData);
  3842. thisElem = JNIenv->NewString(unicode.getustr(), unicodeChars);
  3843. break;
  3844. }
  3845. case type_unicode:
  3846. {
  3847. if (elemSize == UNKNOWN_LENGTH)
  3848. {
  3849. thisSize = (* (size32_t *) inData) * sizeof(UChar); // NOTE - it's in chars...
  3850. inData += sizeof(size32_t);
  3851. }
  3852. thisElem = JNIenv->NewString((const UChar *) inData, thisSize / sizeof(UChar));
  3853. break;
  3854. }
  3855. case type_utf8:
  3856. {
  3857. assertex (elemSize == UNKNOWN_LENGTH);
  3858. size32_t numChars = * (size32_t *) inData;
  3859. inData += sizeof(size32_t);
  3860. unsigned unicodeChars;
  3861. rtlDataAttr unicode;
  3862. rtlUtf8ToUnicodeX(unicodeChars, unicode.refustr(), numChars, (const char *) inData);
  3863. thisElem = JNIenv->NewString(unicode.getustr(), unicodeChars);
  3864. thisSize = rtlUtf8Size(numChars, inData);
  3865. break;
  3866. }
  3867. default:
  3868. typeError("STRING");
  3869. }
  3870. inData += thisSize;
  3871. JNIenv->SetObjectArrayElement((jobjectArray) v.l, idx, thisElem);
  3872. JNIenv->DeleteLocalRef(thisElem);
  3873. idx++;
  3874. }
  3875. }
  3876. else
  3877. typeError("");
  3878. break;
  3879. default:
  3880. throwUnexpected();
  3881. }
  3882. argsig++;
  3883. addArg(v);
  3884. }
  3885. virtual void bindRowParam(const char *name, IOutputMetaData & metaVal, const byte *val) override
  3886. {
  3887. if (*argsig != 'L') // should tell us the type of the object we need to create to pass in
  3888. typeError("RECORD");
  3889. // Class name is from the char after the L up to the first ;
  3890. const char *tail = strchr(argsig, ';');
  3891. if (!tail)
  3892. typeError("RECORD");
  3893. StringAttr className(argsig+1, tail - (argsig+1));
  3894. argsig = tail+1;
  3895. const RtlTypeInfo *typeInfo = metaVal.queryTypeInfo();
  3896. assertex(typeInfo);
  3897. RtlFieldStrInfo dummyField("<row>", NULL, typeInfo);
  3898. JavaObjectBuilder javaBuilder((CheckedJNIEnv *)JNIenv, &dummyField, loadClass(className));
  3899. typeInfo->process(val, val, &dummyField, javaBuilder); // Creates a java object from the incoming ECL row
  3900. jvalue v;
  3901. v.l = javaBuilder.getObject();
  3902. addArg(v);
  3903. }
  3904. virtual IInterface *bindParamWriter(IInterface *esdl, const char *esdlservice, const char *esdltype, const char *name)
  3905. {
  3906. if (*argsig != 'L') // should tell us the type of the object we need to create to pass in
  3907. typeError("OBJECT");
  3908. // Class name is from the char after the L up to the first ;
  3909. const char *tail = strchr(argsig, ';');
  3910. if (!tail)
  3911. typeError("OBJECT");
  3912. StringAttr className(argsig+1, tail - (argsig+1));
  3913. argsig = tail+1;
  3914. Owned<JavaXmlBuilder> writer = new JavaXmlBuilder(JNIenv, dynamic_cast<IEsdlDefinition*>(esdl), esdlservice, esdltype);
  3915. writer->initWriter();
  3916. return (IXmlWriter*)writer.getClear();
  3917. }
  3918. virtual void paramWriterCommit(IInterface *writer)
  3919. {
  3920. JavaXmlBuilder *javaWriter = dynamic_cast<JavaXmlBuilder*>(writer);
  3921. if (!javaWriter)
  3922. throw MakeStringException(0, "javaembed: Invalid object writer for %s", signature.get());
  3923. jvalue v;
  3924. v.l = javaWriter->getObject();
  3925. addArg(v);
  3926. }
  3927. virtual void bindDatasetParam(const char *name, IOutputMetaData & metaVal, IRowStream * val)
  3928. {
  3929. jvalue v;
  3930. char argsigStart = *argsig;
  3931. switch (argsigStart)
  3932. {
  3933. case '[':
  3934. case '<':
  3935. ++argsig;
  3936. break;
  3937. case 'L':
  3938. if (strncmp(argsig, "Ljava/util/Iterator<", 20) == 0)
  3939. {
  3940. argsig += 20;
  3941. break;
  3942. }
  3943. /* no break */
  3944. default:
  3945. typeError("DATASET");
  3946. }
  3947. if (*argsig != 'L') // should tell us the type of the object we need to create to pass in
  3948. typeError("DATASET");
  3949. // Class name is from the char after the L up to the first ;
  3950. const char *tail = strchr(argsig, ';');
  3951. if (!tail)
  3952. typeError("RECORD");
  3953. StringAttr className(argsig+1, tail - (argsig+1));
  3954. argsig = tail+1;
  3955. if (argsigStart=='L')
  3956. {
  3957. if (argsig[0] != '>' || argsig[1] != ';')
  3958. typeError("DATASET");
  3959. argsig += 2;
  3960. }
  3961. if (argsigStart=='[')
  3962. {
  3963. // Pass in an array of objects
  3964. PointerArrayOf<_jobject> allRows;
  3965. const RtlTypeInfo *typeInfo = metaVal.queryTypeInfo();
  3966. assertex(typeInfo);
  3967. RtlFieldStrInfo dummyField("<row>", NULL, typeInfo);
  3968. jclass Class = loadClass(className);
  3969. JavaObjectBuilder javaBuilder((CheckedJNIEnv *) JNIenv, &dummyField, Class);
  3970. for (;;)
  3971. {
  3972. roxiemem::OwnedConstRoxieRow thisRow = val->ungroupedNextRow();
  3973. if (!thisRow)
  3974. break;
  3975. const byte *brow = (const byte *) thisRow.get();
  3976. typeInfo->process(brow, brow, &dummyField, javaBuilder); // Creates a java object from the incoming ECL row
  3977. allRows.append(javaBuilder.getObject());
  3978. }
  3979. jobjectArray array = JNIenv->NewObjectArray(allRows.length(), Class, NULL);
  3980. ForEachItemIn(idx, allRows)
  3981. {
  3982. JNIenv->SetObjectArrayElement(array, idx, allRows.item(idx));
  3983. }
  3984. v.l = array;
  3985. }
  3986. else
  3987. {
  3988. // Pass in an iterator
  3989. // Create a java object of type com.HPCCSystems.HpccUtils - this acts as a proxy for the iterator
  3990. JNIenv->ExceptionClear();
  3991. jvalue param;
  3992. const RtlTypeInfo *typeInfo = metaVal.queryTypeInfo();
  3993. ECLDatasetIterator *iterator = new ECLDatasetIterator((CheckedJNIEnv *) JNIenv, typeInfo, loadClass(className), val);
  3994. param.j = (jlong) iterator;
  3995. iterators.append(*iterator);
  3996. jobject proxy = JNIenv->NewObject(hpccIteratorClass, hi_constructor, param, JNIenv->NewStringUTF(helperLibraryName));
  3997. v.l = proxy;
  3998. }
  3999. addArg(v);
  4000. }
  4001. virtual void writeResult(IInterface *esdl, const char *esdlservice, const char *esdltype, IInterface *writer)
  4002. {
  4003. return sharedCtx->writeObjectResult(result.l, dynamic_cast<IEsdlDefinition*>(esdl), esdlservice, esdltype, dynamic_cast<IXmlWriter*>(writer));
  4004. }
  4005. virtual void importFunction(size32_t lenChars, const char *utf) override
  4006. {
  4007. if (!javaClass)
  4008. {
  4009. size32_t bytes = rtlUtf8Size(lenChars, utf);
  4010. importName.set(utf, bytes);
  4011. if (strchr(importName, '.'))
  4012. loadFunction(classpath, 0, nullptr);
  4013. }
  4014. if (javaClass)
  4015. reinit();
  4016. }
  4017. void bindActivityParam()
  4018. {
  4019. // Note: We don't require that the function takes an activityCtx parameter - if they don't care, they can omit the param
  4020. if (strncmp(argsig, "Lcom/HPCCSystems/ActivityContext;", 33) == 0)
  4021. {
  4022. argsig += 33;
  4023. jvalue v;
  4024. v.l = JNIenv->NewObject(hpccIteratorClass, hi_constructor, activityContext, JNIenv->NewStringUTF(helperLibraryName));
  4025. addArg(v);
  4026. }
  4027. }
  4028. IException *translateException(IException *E)
  4029. {
  4030. StringBuffer msg;
  4031. E->errorMessage(msg);
  4032. const char *text = msg;
  4033. if (strncmp(text, "javaembed: ", 11)==0)
  4034. text += 11;
  4035. auto code = E->errorCode();
  4036. auto aud = E->errorAudience();
  4037. E->Release();
  4038. StringBuffer s;
  4039. return makeStringExceptionV(aud, code, "javaembed: In method %s: %s", getReportName(s).str(), text);
  4040. }
  4041. IException *resultMismatchException(const char *expected)
  4042. {
  4043. StringBuffer s;
  4044. return makeStringExceptionV(0, "javaembed: In method %s: Type mismatch on result (%s expected)", getReportName(s).str(), expected);
  4045. }
  4046. virtual void callFunction()
  4047. {
  4048. try
  4049. {
  4050. if (*argsig != ')')
  4051. throw MakeStringException(0, "Too few ECL parameters passed for Java signature %s", signature.get());
  4052. JNIenv->ExceptionClear();
  4053. if (nonStatic)
  4054. {
  4055. if (streq(methodName, "<init>"))
  4056. {
  4057. if (!instance)
  4058. {
  4059. if (persistMode == persistNone)
  4060. throw MakeStringException(0, "Cannot return object without persist");
  4061. StringBuffer scopeKey;
  4062. getScopeKey(scopeKey);
  4063. PersistedObjectCriticalBlock persistBlock;
  4064. persistBlock.enter(persistMode==persistThread ? sharedCtx->getLocalObject(JNIenv, scopeKey) : globalState->getGlobalObject(JNIenv, scopeKey));
  4065. instance = persistBlock.getInstance();
  4066. if (instance)
  4067. persistBlock.leave();
  4068. else
  4069. {
  4070. instance = JNIenv->NewGlobalRef(JNIenv->NewObjectA(javaClass, javaMethodID, args), "constructor");
  4071. #ifdef TRACE_GLOBALREF
  4072. StringBuffer myClassName;
  4073. getClassNameForObject(JNIenv, myClassName, instance);
  4074. DBGLOG("Constructed object %p of class %s", instance, myClassName.str());
  4075. #endif
  4076. if (persistMode==persistQuery || persistMode==persistWorkunit || persistMode==persistChannel)
  4077. {
  4078. assertex(engine);
  4079. engine->onTermination(JavaGlobalState::unregister, scopeKey.str(), persistMode==persistWorkunit);
  4080. }
  4081. persistBlock.leave(instance);
  4082. }
  4083. }
  4084. result.l = instance;
  4085. return;
  4086. }
  4087. else if (!instance)
  4088. {
  4089. assertex(persistMode == persistNone); // Any other persist mode should have already created the instance
  4090. instance = createInstance(); // Local object, will be released at exit() from function
  4091. }
  4092. assertex(javaMethodID);
  4093. switch (*returnType)
  4094. {
  4095. case 'C': result.c = JNIenv->CallCharMethodA(instance, javaMethodID, args); break;
  4096. case 'Z': result.z = JNIenv->CallBooleanMethodA(instance, javaMethodID, args); break;
  4097. case 'J': result.j = JNIenv->CallLongMethodA(instance, javaMethodID, args); break;
  4098. case 'F': result.f = JNIenv->CallFloatMethodA(instance, javaMethodID, args); break;
  4099. case 'D': result.d = JNIenv->CallDoubleMethodA(instance, javaMethodID, args); break;
  4100. case 'I': result.i = JNIenv->CallIntMethodA(instance, javaMethodID, args); break;
  4101. case 'S': result.s = JNIenv->CallShortMethodA(instance, javaMethodID, args); break;
  4102. case 'B': result.s = JNIenv->CallByteMethodA(instance, javaMethodID, args); break;
  4103. case '[':
  4104. case 'L': result.l = JNIenv->CallObjectMethodA(instance, javaMethodID, args); break;
  4105. case 'V': JNIenv->CallVoidMethodA(instance, javaMethodID, args); result.l = nullptr; break;
  4106. default: throwUnexpected();
  4107. }
  4108. }
  4109. else
  4110. {
  4111. switch (*returnType)
  4112. {
  4113. case 'C': result.c = JNIenv->CallStaticCharMethodA(javaClass, javaMethodID, args); break;
  4114. case 'Z': result.z = JNIenv->CallStaticBooleanMethodA(javaClass, javaMethodID, args); break;
  4115. case 'J': result.j = JNIenv->CallStaticLongMethodA(javaClass, javaMethodID, args); break;
  4116. case 'F': result.f = JNIenv->CallStaticFloatMethodA(javaClass, javaMethodID, args); break;
  4117. case 'D': result.d = JNIenv->CallStaticDoubleMethodA(javaClass, javaMethodID, args); break;
  4118. case 'I': result.i = JNIenv->CallStaticIntMethodA(javaClass, javaMethodID, args); break;
  4119. case 'S': result.s = JNIenv->CallStaticShortMethodA(javaClass, javaMethodID, args); break;
  4120. case 'B': result.s = JNIenv->CallStaticByteMethodA(javaClass, javaMethodID, args); break;
  4121. case '[':
  4122. case 'L': result.l = JNIenv->CallStaticObjectMethodA(javaClass, javaMethodID, args); break;
  4123. case 'V': JNIenv->CallStaticVoidMethodA(javaClass, javaMethodID, args); result.l = nullptr; break;
  4124. default: throwUnexpected();
  4125. }
  4126. }
  4127. }
  4128. catch (IException *E)
  4129. {
  4130. throw translateException(E);
  4131. }
  4132. }
  4133. virtual void compileEmbeddedScript(size32_t lenChars, const char *_script)
  4134. {
  4135. throwUnexpected();
  4136. }
  4137. virtual void loadCompiledScript(size32_t bytecodeLen, const void *bytecode) override
  4138. {
  4139. if (!javaClass)
  4140. {
  4141. MemoryBuffer b;
  4142. b.setBuffer(bytecodeLen, (void *) bytecode, false);
  4143. b.setEndian(__BIG_ENDIAN);
  4144. uint32_t siglen; b.read(siglen);
  4145. const char *sig = (const char *) b.readDirect(siglen);
  4146. size32_t bytes = rtlUtf8Size(siglen, sig); // MORE - check that this size is serialized in chars not bytes!
  4147. importName.set(sig, bytes);
  4148. loadFunction(classpath, bytecodeLen, (const byte *) bytecode);
  4149. }
  4150. reinit();
  4151. }
  4152. virtual void enter() override
  4153. {
  4154. reenter(nullptr);
  4155. }
  4156. virtual void reenter(ICodeContext *codeCtx) override
  4157. {
  4158. // If we rejig codegen to only call loadCompiledScript etc at construction time, then this will need to do the reinit()
  4159. // until we do, it's too early
  4160. if (codeCtx)
  4161. engine = codeCtx->queryEngineContext();
  4162. else if (flags & EFthreadlocal && persistMode > persistThread)
  4163. {
  4164. StringBuffer s;
  4165. throw MakeStringException(0, "javaembed: In method %s: Workunit must be recompiled to support this persist mode", getReportName(s).str());
  4166. }
  4167. // Create a new frame for local references and increase the capacity
  4168. // of those references to 64 (default is 16)
  4169. JNIenv->PushLocalFrame(64);
  4170. }
  4171. virtual void exit() override
  4172. {
  4173. if (persistMode==persistNone)
  4174. instance = 0; // otherwise we leave it for next call as it saves a lot of time looking it up
  4175. iterators.kill();
  4176. #ifdef FORCE_GC
  4177. forceGC(JNIenv);
  4178. #endif
  4179. JNIenv->PopLocalFrame(nullptr);
  4180. }
  4181. protected:
  4182. __declspec(noreturn) void typeError(const char *ECLtype) __attribute__((noreturn))
  4183. {
  4184. const char *javaType;
  4185. int javaLen = 0;
  4186. switch (*argsig)
  4187. {
  4188. case 'Z': javaType = "boolean"; break;
  4189. case 'B': javaType = "byte"; break;
  4190. case 'C': javaType = "char"; break;
  4191. case 'S': javaType = "short"; break;
  4192. case 'I': javaType = "int"; break;
  4193. case 'J': javaType = "long"; break;
  4194. case 'F': javaType = "float"; break;
  4195. case 'D': javaType = "double"; break;
  4196. case '[': javaType = "array"; break;
  4197. case 'L':
  4198. {
  4199. javaType = argsig+1;
  4200. const char *semi = strchr(argsig, ';');
  4201. if (semi)
  4202. javaLen = semi - javaType;
  4203. break;
  4204. }
  4205. case ')':
  4206. {
  4207. StringBuffer s;
  4208. throw MakeStringException(0, "javaembed: In method %s: Too many ECL parameters passed for Java signature", getReportName(s).str());
  4209. }
  4210. default:
  4211. StringBuffer s;
  4212. throw MakeStringException(0, "javaembed: In method %s: Unrecognized character %c in Java signature", getReportName(s).str(), *argsig);
  4213. }
  4214. if (!javaLen)
  4215. javaLen = strlen(javaType);
  4216. StringBuffer s;
  4217. throw MakeStringException(0, "javaembed: In Method %s: ECL type %s cannot be passed to Java type %.*s", getReportName(s).str(), ECLtype, javaLen, javaType);
  4218. }
  4219. void addArg(jvalue &arg)
  4220. {
  4221. assertex(argcount < MAX_JNI_ARGS);
  4222. args[argcount] = arg;
  4223. argcount++;
  4224. }
  4225. size32_t getRowResult(jobject result, ARowBuilder &builder)
  4226. {
  4227. const RtlTypeInfo *typeInfo = builder.queryAllocator()->queryOutputMeta()->queryTypeInfo();
  4228. assertex(typeInfo);
  4229. RtlFieldStrInfo dummyField("<row>", NULL, typeInfo);
  4230. JavaRowBuilder javaRowBuilder((CheckedJNIEnv *) JNIenv, &dummyField, result);
  4231. return typeInfo->build(builder, 0, &dummyField, javaRowBuilder);
  4232. }
  4233. jclass loadClass(const char *className)
  4234. {
  4235. StringBuffer uclassname(className);
  4236. uclassname.replace('/', '.');
  4237. className=uclassname.str();
  4238. JNIenv->ExceptionClear();
  4239. jmethodID loadClassMethod = JNIenv->GetMethodID(JNIenv->GetObjectClass(classLoader), "loadClass","(Ljava/lang/String;)Ljava/lang/Class;");
  4240. jstring classNameString = JNIenv->NewStringUTF(className);
  4241. jclass Class = (jclass) JNIenv->CallObjectMethod(classLoader, loadClassMethod, classNameString);
  4242. return Class;
  4243. }
  4244. StringBuffer &getReportName(StringBuffer &s)
  4245. {
  4246. if (classname.length())
  4247. {
  4248. const char *report = strrchr(classname.str(), '.');
  4249. if (report)
  4250. report++;
  4251. else
  4252. report = classname.str();
  4253. s.append(classname).append('.');
  4254. }
  4255. return s.append(methodName);
  4256. }
  4257. jobject createInstance()
  4258. {
  4259. jmethodID constructor;
  4260. try
  4261. {
  4262. constructor = JNIenv->GetMethodID(javaClass, "<init>", "()V");
  4263. }
  4264. catch (IException *E)
  4265. {
  4266. Owned<IException> e = E;
  4267. throw MakeStringException(0, "parameterless constructor required");
  4268. }
  4269. return JNIenv->NewObject(javaClass, constructor);
  4270. }
  4271. void loadFunction(const char *classpath, size32_t bytecodeLen, const byte *bytecode)
  4272. {
  4273. try
  4274. {
  4275. StringAttr checkedClassName;
  4276. // Name should be in the form class.method:signature
  4277. const char *funcname = strrchr(importName, '.');
  4278. if (funcname)
  4279. {
  4280. classname.clear().append(funcname-importName, importName);
  4281. classname.replace('/', '.');
  4282. funcname++; // skip the '.'
  4283. }
  4284. else
  4285. funcname = importName;
  4286. const char *coloncolon = strstr(funcname, "::");
  4287. if (coloncolon)
  4288. {
  4289. // ClassName::FunctionName syntax - used to check that object passed in is of proper class
  4290. checkedClassName.set(funcname, coloncolon-funcname);
  4291. funcname = coloncolon+2;
  4292. }
  4293. const char *sig = strchr(funcname, ':');
  4294. if (sig)
  4295. {
  4296. methodName.set(funcname, sig-funcname);
  4297. sig++; // skip the ':'
  4298. if (*sig == '@') // indicates a non-static method
  4299. {
  4300. sig++;
  4301. nonStatic = true;
  4302. }
  4303. else
  4304. nonStatic = false;
  4305. signature.set(sig);
  4306. }
  4307. else
  4308. methodName.set(funcname);
  4309. bool isConstructor = streq(methodName, "<init>");
  4310. {
  4311. PersistedObjectCriticalBlock persistBlock;
  4312. StringBuffer scopeKey;
  4313. if (nonStatic && !instance && persistMode >= persistThread && !isConstructor)
  4314. {
  4315. // If a persist scope is specified, we may want to use a pre-existing object. If we do we share its classloader, class, etc.
  4316. assertex(classname.length()); // MORE - what does this imply?
  4317. getScopeKey(scopeKey);
  4318. persistBlock.enter(persistMode==persistThread ? sharedCtx->getLocalObject(JNIenv, scopeKey) : globalState->getGlobalObject(JNIenv, scopeKey));
  4319. instance = persistBlock.getInstance();
  4320. if (instance)
  4321. persistBlock.leave();
  4322. }
  4323. if (instance)
  4324. {
  4325. javaClass = (jclass) JNIenv->NewGlobalRef(JNIenv->GetObjectClass(instance), "javaClass");
  4326. classLoader = JNIenv->NewGlobalRef(getClassLoader(JNIenv, javaClass), "classLoader");
  4327. sharedCtx->setThreadClassLoader(classLoader);
  4328. }
  4329. if (!javaClass)
  4330. {
  4331. if (!classname)
  4332. throw MakeStringException(MSGAUD_user, 0, "Invalid import name - Expected classname.methodname:signature");
  4333. classLoader = JNIenv->NewGlobalRef(sharedCtx->getThreadClassLoader(classpath, classname, bytecodeLen, bytecode), "classLoader");
  4334. sharedCtx->setThreadClassLoader(classLoader);
  4335. jmethodID loadClassMethod = JNIenv->GetMethodID(JNIenv->GetObjectClass(classLoader), "loadClass","(Ljava/lang/String;)Ljava/lang/Class;");
  4336. try
  4337. {
  4338. javaClass = (jclass) JNIenv->CallObjectMethod(classLoader, loadClassMethod, JNIenv->NewStringUTF(classname));
  4339. }
  4340. catch (IException *E)
  4341. {
  4342. Owned<IException> e = E;
  4343. throw makeWrappedExceptionV(E, E->errorCode(), "Failed to resolve class name %s", classname.str());
  4344. }
  4345. javaClass = (jclass) JNIenv->NewGlobalRef(javaClass, "javaClass");
  4346. }
  4347. if (nonStatic && !instance && !isConstructor && persistMode != persistNone)
  4348. {
  4349. instance = createInstance();
  4350. #ifdef TRACE_GLOBALREF
  4351. StringBuffer myClassName;
  4352. getClassNameForObject(JNIenv, myClassName, instance);
  4353. DBGLOG("Created object %p of class %s", instance, myClassName.str());
  4354. #endif
  4355. if (persistBlock.locked()) // I think this should always be true?
  4356. {
  4357. instance = JNIenv->NewGlobalRef(instance, "createInstance");
  4358. if (persistMode==persistQuery || persistMode==persistWorkunit || persistMode==persistChannel)
  4359. {
  4360. assertex(engine);
  4361. engine->onTermination(JavaGlobalState::unregister, scopeKey.str(), persistMode==persistWorkunit);
  4362. }
  4363. persistBlock.leave(instance);
  4364. }
  4365. }
  4366. }
  4367. if (!signature)
  4368. {
  4369. getSignature(signature, JNIenv, javaClass, funcname);
  4370. if (signature.str()[0]=='@')
  4371. {
  4372. nonStatic = true;
  4373. signature.set(signature.str()+1);
  4374. }
  4375. else
  4376. nonStatic = false;
  4377. }
  4378. StringBuffer javaSignature;
  4379. patchSignature(javaSignature, signature);
  4380. if (nonStatic)
  4381. javaMethodID = JNIenv->GetMethodID(javaClass, methodName, javaSignature);
  4382. else
  4383. javaMethodID = JNIenv->GetStaticMethodID(javaClass, methodName, javaSignature);
  4384. if (checkedClassName)
  4385. {
  4386. StringBuffer myClassName;
  4387. getClassNameForObject(JNIenv, myClassName, instance);
  4388. #ifdef CHECK_JNI
  4389. DBGLOG("Checking class name %s for %p matches %s for function %s", myClassName.str(), instance, checkedClassName.str(), methodName.str());
  4390. #endif
  4391. const char *shortClassName = strrchr(myClassName, '.');
  4392. if (shortClassName)
  4393. shortClassName++;
  4394. else
  4395. shortClassName = myClassName;
  4396. if (!streq(checkedClassName, shortClassName))
  4397. throw MakeStringException(0, "Object class %s does not match expected class name %s", shortClassName, checkedClassName.str());
  4398. }
  4399. returnType = strrchr(signature, ')');
  4400. assertex(returnType); // Otherwise how did Java accept it??
  4401. returnType++;
  4402. }
  4403. catch(IException *E)
  4404. {
  4405. throw translateException(E);
  4406. }
  4407. }
  4408. static StringBuffer &patchSignature(StringBuffer &ret, const char *signature)
  4409. {
  4410. // We need to patch up the provided signature - any instances of <classname; need to be replaced by Ljava.utils.iterator
  4411. const char *finger = signature;
  4412. while (finger && *finger)
  4413. {
  4414. if (*finger == '<')
  4415. {
  4416. // If there is a corresponding >, assume it's the 'extended' form and just strip out the bit from < to >
  4417. const char *close = strchr(finger, '>');
  4418. if (close)
  4419. finger = close;
  4420. else
  4421. {
  4422. ret.append("Ljava/util/Iterator;");
  4423. finger = strchr(finger, ';');
  4424. if (!finger)
  4425. throw MakeStringException(MSGAUD_user, 0, "javaembed: Invalid java function signature %s", signature);
  4426. }
  4427. }
  4428. else
  4429. ret.append(*finger);
  4430. finger++;
  4431. }
  4432. return ret;
  4433. }
  4434. StringBuffer &getScopeKey(StringBuffer &ret)
  4435. {
  4436. if (globalScopeKey)
  4437. ret.append(globalScopeKey).append('.');
  4438. ret.append(classname).append('.');
  4439. switch (persistMode)
  4440. {
  4441. case persistThread:
  4442. ret.append(__uint64(GetCurrentThreadId()));
  4443. break;
  4444. case persistGlobal:
  4445. ret.append("global");
  4446. break;
  4447. // Fall into
  4448. case persistWorkunit:
  4449. engine->getQueryId(ret, true);
  4450. break;
  4451. case persistChannel:
  4452. ret.append(nodeNum).append('.');
  4453. case persistQuery:
  4454. engine->getQueryId(ret, false);
  4455. break;
  4456. }
  4457. return ret;
  4458. }
  4459. JavaThreadContext *sharedCtx = nullptr;
  4460. CheckedJNIEnv *JNIenv = nullptr;
  4461. jvalue result = {0};
  4462. StringAttr classpath;
  4463. StringBuffer classname;
  4464. IArrayOf<ECLDatasetIterator> iterators; // to make sure they get freed
  4465. bool nonStatic = false;
  4466. jobject instance = nullptr; // class instance of object to call methods on
  4467. const IThorActivityContext *activityContext = nullptr;
  4468. unsigned flags = 0;
  4469. unsigned nodeNum = 0;
  4470. StringAttr globalScopeKey;
  4471. PersistMode persistMode = persistNone; // Defines the lifetime of the java object for which this is called.
  4472. // The following members are set up the first time a method is called only
  4473. IEngineContext *engine = nullptr;
  4474. jclass javaClass = nullptr;
  4475. jobject classLoader = nullptr;
  4476. jmethodID javaMethodID = nullptr;
  4477. StringAttr methodName;
  4478. StringAttr importName;
  4479. StringAttr signature;
  4480. const char *returnType = nullptr; // A pointer within signature
  4481. // These point to the current arg/signature byte as we are binding
  4482. int argcount = 0;
  4483. jvalue args[MAX_JNI_ARGS];
  4484. const char *argsig = nullptr; // A pointer within signature
  4485. void reinit()
  4486. {
  4487. argcount = 0;
  4488. argsig = signature;
  4489. assertex(*argsig == '(');
  4490. argsig++;
  4491. if (activityContext)
  4492. bindActivityParam();
  4493. }
  4494. };
  4495. static __thread JavaThreadContext* threadContext; // We reuse per thread, for speed
  4496. static bool releaseContext(bool isPooled)
  4497. {
  4498. if (threadContext)
  4499. {
  4500. threadContext->endThread();
  4501. if (!isPooled)
  4502. {
  4503. delete threadContext;
  4504. threadContext = NULL;
  4505. }
  4506. else
  4507. return true;
  4508. }
  4509. return false;
  4510. }
  4511. static JavaThreadContext *queryContext()
  4512. {
  4513. if (!threadContext)
  4514. {
  4515. threadContext = new JavaThreadContext;
  4516. addThreadTermFunc(releaseContext);
  4517. }
  4518. return threadContext;
  4519. }
  4520. static CheckedJNIEnv *queryJNIEnv()
  4521. {
  4522. return queryContext()->JNIenv;
  4523. }
  4524. class JavaEmbedServiceContext : public CInterfaceOf<IEmbedServiceContext>
  4525. {
  4526. public:
  4527. JavaEmbedServiceContext(JavaThreadContext *_sharedCtx, const char *service, const char *_options)
  4528. : sharedCtx(_sharedCtx), Class(0), options(_options), className(service), object(0)
  4529. {
  4530. StringArray opts;
  4531. opts.appendList(options, ",");
  4532. ForEachItemIn(idx, opts)
  4533. {
  4534. const char *opt = opts.item(idx);
  4535. const char *val = strchr(opt, '=');
  4536. if (val)
  4537. {
  4538. StringBuffer optName(val-opt, opt);
  4539. val++;
  4540. if (strieq(optName, "classpath"))
  4541. classpath.set(val);
  4542. else
  4543. throw MakeStringException(0, "javaembed: Unknown option %s", optName.str());
  4544. }
  4545. }
  4546. }
  4547. ~JavaEmbedServiceContext()
  4548. {
  4549. if (object)
  4550. sharedCtx->JNIenv->DeleteGlobalRef(object);
  4551. if (Class)
  4552. sharedCtx->JNIenv->DeleteGlobalRef(Class);
  4553. }
  4554. void init()
  4555. {
  4556. jobject classLoader = sharedCtx->getThreadClassLoader();
  4557. jmethodID loadClassMethod = sharedCtx->JNIenv->GetMethodID(sharedCtx->JNIenv->GetObjectClass(classLoader), "loadClass","(Ljava/lang/String;)Ljava/lang/Class;");
  4558. jstring methodString = sharedCtx->JNIenv->NewStringUTF(className);
  4559. Class = (jclass) sharedCtx->JNIenv->NewGlobalRef(sharedCtx->JNIenv->CallObjectMethod(classLoader, loadClassMethod, methodString), "Class");
  4560. jmethodID constructor = sharedCtx->JNIenv->GetMethodID(Class, "<init>", "()V");
  4561. object = sharedCtx->JNIenv->NewGlobalRef(sharedCtx->JNIenv->NewObject(Class, constructor), "constructed");
  4562. }
  4563. virtual IEmbedFunctionContext *createFunctionContext(const char *function)
  4564. {
  4565. if (!object)
  4566. return NULL;
  4567. Owned<JavaEmbedImportContext> fctx = new JavaEmbedImportContext(nullptr, queryContext(), object, 0, options, nullptr);
  4568. fctx->importFunction(rtlUtf8Length(strlen(function), function), function);
  4569. return fctx.getClear();
  4570. }
  4571. protected:
  4572. JavaThreadContext *sharedCtx;
  4573. StringBuffer className;
  4574. jclass Class;
  4575. jobject object;
  4576. StringAttr classpath;
  4577. StringAttr options;
  4578. };
  4579. class JavaEmbedContext : public CInterfaceOf<IEmbedContext>
  4580. {
  4581. public:
  4582. virtual IEmbedFunctionContext *createFunctionContext(unsigned flags, const char *options) override
  4583. {
  4584. return createFunctionContextEx(nullptr, nullptr, flags, options);
  4585. }
  4586. virtual IEmbedFunctionContext *createFunctionContextEx(ICodeContext * ctx, const IThorActivityContext *activityCtx, unsigned flags, const char *options) override
  4587. {
  4588. return new JavaEmbedImportContext(ctx, queryContext(), nullptr, flags, options, activityCtx);
  4589. }
  4590. virtual IEmbedServiceContext *createServiceContext(const char *service, unsigned flags, const char *options) override
  4591. {
  4592. Owned<JavaEmbedServiceContext> serviceContext = new JavaEmbedServiceContext(queryContext(), service, options);
  4593. serviceContext->init();
  4594. return serviceContext.getClear();
  4595. }
  4596. };
  4597. static JavaEmbedContext embedContext;
  4598. extern DECL_EXPORT IEmbedContext* queryEmbedContext()
  4599. {
  4600. return &embedContext;
  4601. }
  4602. extern DECL_EXPORT IEmbedContext* getEmbedContext()
  4603. {
  4604. return new JavaEmbedContext;
  4605. }
  4606. static bool isValidIdentifier(const char *source)
  4607. {
  4608. return isalnum(*source) || *source=='_' || *source=='$' || ::readUtf8Size(source)>1; // This is not strictly accurate but probably good enough
  4609. }
  4610. static bool isFullClassFile(StringBuffer &className, bool &seenPublic, size32_t len, const char *source)
  4611. {
  4612. // A heuristic to determine whether the supplied embedded source is a full class file or just a single method
  4613. // Basically, if we see keyword "class" before we see { then we assume it's a full file
  4614. // Also track whether the public keyword has been supplied
  4615. bool inLineComment = false;
  4616. bool inBlockComment = false;
  4617. seenPublic = false;
  4618. while (len)
  4619. {
  4620. if (inLineComment)
  4621. {
  4622. if (*source=='\n')
  4623. inLineComment = false;
  4624. }
  4625. else if (inBlockComment)
  4626. {
  4627. if (*source=='*' && len > 1 && source[1]=='/')
  4628. {
  4629. inBlockComment = false;
  4630. len--;
  4631. source++;
  4632. }
  4633. }
  4634. else switch(*source)
  4635. {
  4636. case '/':
  4637. if (len > 1)
  4638. {
  4639. if (source[1]=='*')
  4640. {
  4641. inBlockComment = true;
  4642. len--;
  4643. source++;
  4644. }
  4645. else if (source[1]=='/')
  4646. inLineComment = true;
  4647. }
  4648. break;
  4649. case '{':
  4650. return false;
  4651. default:
  4652. if (isValidIdentifier(source))
  4653. {
  4654. const char *start = source;
  4655. while (len && isValidIdentifier(source))
  4656. {
  4657. source+=::readUtf8Size(source);
  4658. len--;
  4659. }
  4660. if (source-start == 5 && memcmp(start, "class", source-start)==0)
  4661. {
  4662. while (len && isspace(*source)) // MORE - a comment between the keyword and the classname will fail - tough.
  4663. {
  4664. source += ::readUtf8Size(source);
  4665. len--;
  4666. }
  4667. start = source;
  4668. while (len && isValidIdentifier(source))
  4669. {
  4670. source += ::readUtf8Size(source);
  4671. len--;
  4672. }
  4673. className.append(source-start, start);
  4674. return true;
  4675. }
  4676. else if (source-start == 6 && memcmp(start, "public", source-start)==0)
  4677. seenPublic = true;
  4678. }
  4679. if (!len)
  4680. return false;
  4681. break;
  4682. }
  4683. source += ::readUtf8Size(source);
  4684. len--;
  4685. }
  4686. // If we get here then it doesn't have a { at all - we COULD say it needs the prototype too but for now, who knows...
  4687. return false;
  4688. }
  4689. static bool suppressJavaError(const char *err)
  4690. {
  4691. if (!err || !*err)
  4692. return true;
  4693. if (streq(err, "1 error"))
  4694. return true;
  4695. char *rest;
  4696. if (strtoul(err, &rest, 10) && streq(rest, " errors"))
  4697. return true;
  4698. return false;
  4699. }
  4700. static StringBuffer & cleanupJavaError(StringBuffer &ret, StringBuffer &prefix, const char *err, unsigned lineNumberOffset)
  4701. {
  4702. // Remove filename (as it's generated) and fix up line number. Errors that do not have line number use previous error's line number.
  4703. const char *colon = strchr(err, ':');
  4704. // Java errors are a bit of a pain - if you suppress the ones without line numbers you get too little information, if you don't you get too much
  4705. if (colon && isdigit(colon[1]))
  4706. {
  4707. char *end;
  4708. unsigned lineno = strtoul(colon+1, &end, 10) - lineNumberOffset;
  4709. prefix.clear().appendf("(%u,1)", lineno);
  4710. ret.append(prefix).append(end);
  4711. }
  4712. else if (!suppressJavaError(err))
  4713. ret.append(prefix).appendf(": error: %s", err);
  4714. return ret;
  4715. }
  4716. static void cleanupJavaErrors(StringBuffer &ret, const char *errors, unsigned lineNumberOffset)
  4717. {
  4718. StringArray errlines;
  4719. errlines.appendList(errors, "\n", false);
  4720. StringBuffer prefix;
  4721. ForEachItemIn(idx, errlines)
  4722. {
  4723. StringBuffer cleaned;
  4724. cleanupJavaError(cleaned, prefix, errlines.item(idx), lineNumberOffset);
  4725. if (cleaned.length())
  4726. ret.append(cleaned).append('\n');
  4727. }
  4728. }
  4729. static thread_local unsigned prevHash = 0;
  4730. static thread_local MemoryBuffer prevCompile;
  4731. void doPrecompile(size32_t & __lenResult, void * & __result, const char *funcName, size32_t charsBody, const char * body, const char *argNames, const char *compilerOptions, const char *persistOptions, StringBuffer &errors, bool checking)
  4732. {
  4733. unsigned sizeBody = rtlUtf8Size(charsBody, body); // size in bytes
  4734. unsigned hash = rtlHash32Data(sizeBody,body,0xcafebabe);
  4735. if (hash==prevHash) // Reusing result from the syntax check that normally immediately precedes a precompile
  4736. {
  4737. __lenResult = prevCompile.length();
  4738. __result = prevCompile.detachOwn();
  4739. prevHash = 0;
  4740. return;
  4741. }
  4742. StringAttr globalScope;
  4743. PersistMode persistMode = getPersistMode(persistOptions, globalScope);
  4744. StringBuffer tmpDirName;
  4745. getTempFilePath(tmpDirName, "javaembed", nullptr);
  4746. tmpDirName.append(PATHSEPCHAR).append("tmp.XXXXXX");
  4747. if (!mkdtemp((char *) tmpDirName.str()))
  4748. throw makeStringExceptionV(0, "Failed to create temporary directory %s (error %d)", tmpDirName.str(), errno);
  4749. Owned<IFile> tempDir = createIFile(tmpDirName);
  4750. StringBuffer classname;
  4751. bool seenPublic = false;
  4752. bool isFullClass = isFullClassFile(classname, seenPublic, charsBody, body); // note - we pass in length in characters, not bytes
  4753. if (!isFullClass)
  4754. classname.set("embed");
  4755. VStringBuffer javafile("%s" PATHSEPSTR "%s.java", tmpDirName.str(), classname.str());
  4756. FILE *source = fopen(javafile.str(), "wt");
  4757. fprintf(source, "package com.HPCCSystems.embed.x%x;\n", hash);
  4758. unsigned lineNumberOffset = 1; // for the /n above
  4759. if (isFullClass)
  4760. fprintf(source, "%.*s", sizeBody, body);
  4761. else
  4762. {
  4763. if (seenPublic)
  4764. fprintf(source, "public class embed\n{\n %.*s\n}", sizeBody, body);
  4765. else
  4766. fprintf(source, "public class embed\n{\n public static %.*s\n}", sizeBody, body);
  4767. lineNumberOffset += 2; // for the /n's above
  4768. }
  4769. fclose(source);
  4770. MemoryBuffer result;
  4771. Owned<IPipeProcess> pipe = createPipeProcess();
  4772. StringBuffer options(compilerOptions);
  4773. if (isEmptyString(compilerOptions))
  4774. options.append("-g:none");
  4775. appendClassPath(options.append(" -cp "));
  4776. VStringBuffer javac("javac %s %s", options.str(), javafile.str());
  4777. if (!pipe->run("javac", javac, tmpDirName, false, false, true, 0, false))
  4778. {
  4779. throw makeStringException(0, "Failed to run javac");
  4780. }
  4781. else
  4782. {
  4783. StringBuffer javaErrors;
  4784. Owned<ISimpleReadStream> pipeReader = pipe->getErrorStream();
  4785. readSimpleStream(javaErrors, *pipeReader);
  4786. pipe->closeError();
  4787. unsigned retcode = pipe->wait();
  4788. cleanupJavaErrors(errors, javaErrors, lineNumberOffset);
  4789. if (retcode)
  4790. throw makeStringException(0, "Failed to precompile java code");
  4791. VStringBuffer mainfile("%s" PATHSEPSTR "%s.class", tmpDirName.str(), classname.str());
  4792. JavaClassReader reader(mainfile);
  4793. DBGLOG("Analysing generated class %s", reader.queryClassName());
  4794. // Not sure how useful this warning is.
  4795. //if (persistMode > persistThread && (reader.getFlags(funcName) & JavaClassReader::ACC_SYNCHRONIZED)==0)
  4796. // errors.appendf("Warning: persist mode set but function is not synchronized\n");
  4797. reader.getEmbedData(result, funcName, true);
  4798. removeFileTraceIfFail(mainfile);
  4799. // Now read nested classes
  4800. Owned<IDirectoryIterator> classFiles = tempDir->directoryFiles("*$*.class",false,false);
  4801. ForEach(*classFiles)
  4802. {
  4803. const char *thisFile = classFiles->query().queryFilename();
  4804. JavaClassReader reader(thisFile);
  4805. reader.getEmbedData(result, nullptr, false);
  4806. removeFileTraceIfFail(thisFile);
  4807. }
  4808. // We could give a warning if persist is set to anything over "thread" and the function we are calling is not synchronized.
  4809. }
  4810. removeFileTraceIfFail(javafile);
  4811. tempDir->remove();
  4812. __lenResult = result.length();
  4813. __result = result.detachOwn();
  4814. }
  4815. extern DECL_EXPORT void precompile(size32_t & __lenResult, void * & __result, const char *funcName, size32_t charsBody, const char * body, const char *argNames, const char *compilerOptions, const char *persistOptions)
  4816. {
  4817. StringBuffer errors;
  4818. doPrecompile(__lenResult, __result, funcName, charsBody, body, argNames, compilerOptions, persistOptions, errors, false);
  4819. }
  4820. extern DECL_EXPORT void syntaxCheck(size32_t & __lenResult, char * & __result, const char *funcname, size32_t charsBody, const char * body, const char *argNames, const char *compilerOptions, const char *persistOptions)
  4821. {
  4822. StringBuffer result;
  4823. try
  4824. {
  4825. size32_t ds;
  4826. rtlDataAttr d;
  4827. StringBuffer errors;
  4828. doPrecompile(ds, d.refdata(), funcname, charsBody, body, argNames, compilerOptions, persistOptions, result, true);
  4829. // Reuse result in the precompile that normally immediately follows
  4830. unsigned sizeBody = rtlUtf8Size(charsBody, body); // size in bytes
  4831. prevHash = rtlHash32Data(sizeBody,body,0xcafebabe);
  4832. prevCompile.setBuffer(ds, d.detachdata(), true);
  4833. }
  4834. catch (IException *E)
  4835. {
  4836. StringBuffer msg;
  4837. result.append(E->errorMessage(msg));
  4838. E->Release();
  4839. }
  4840. __lenResult = result.length();
  4841. __result = result.detach();
  4842. }
  4843. extern DECL_EXPORT void checkImport(size32_t & __lenResult, char * & __result, const char *funcname, size32_t charsImport, const char * import, const char *argNames, const char *compilerOptions, const char *persistOptions)
  4844. {
  4845. StringBuffer result;
  4846. try
  4847. {
  4848. StringAttr globalScope;
  4849. PersistMode persistMode = getPersistMode(persistOptions, globalScope);
  4850. if (persistMode > persistThread && !globalScope)
  4851. {
  4852. StringBuffer b(rtlUtf8Size(charsImport, import), import);
  4853. const char *dotpos = strrchr(b, '.');
  4854. if (!dotpos)
  4855. throw MakeStringException(0, "javaembed: cannot determine key for persist in function %s", b.str());
  4856. }
  4857. }
  4858. catch (IException *E)
  4859. {
  4860. StringBuffer msg;
  4861. result.appendf("%s\n", E->errorMessage(msg).str());
  4862. E->Release();
  4863. }
  4864. __lenResult = result.length();
  4865. __result = result.detach();
  4866. }
  4867. } // namespace
  4868. // Callbacks from java
  4869. extern "C" {
  4870. JNIEXPORT jboolean JNICALL Java_com_HPCCSystems_HpccUtils__1hasNext (JNIEnv *, jclass, jlong);
  4871. JNIEXPORT jobject JNICALL Java_com_HPCCSystems_HpccUtils__1next (JNIEnv *, jclass, jlong);
  4872. JNIEXPORT jclass JNICALL Java_com_HPCCSystems_HpccClassLoader_defineClassForEmbed(JNIEnv *env, jobject loader, jint bytecodeLen, jlong bytecode, jstring name);
  4873. JNIEXPORT void JNICALL Java_com_HPCCSystems_HpccUtils_log(JNIEnv *JNIenv, jclass, jstring msg);
  4874. JNIEXPORT jboolean JNICALL Java_com_HPCCSystems_HpccUtils__1isLocal (JNIEnv *, jclass, jlong);
  4875. JNIEXPORT jint JNICALL Java_com_HPCCSystems_HpccUtils__1numSlaves (JNIEnv *, jclass, jlong);
  4876. JNIEXPORT jint JNICALL Java_com_HPCCSystems_HpccUtils__1numStrands (JNIEnv *, jclass, jlong);
  4877. JNIEXPORT jint JNICALL Java_com_HPCCSystems_HpccUtils__1querySlave (JNIEnv *, jclass, jlong);
  4878. JNIEXPORT jint JNICALL Java_com_HPCCSystems_HpccUtils__1queryStrand (JNIEnv *, jclass, jlong);
  4879. }
  4880. JNIEXPORT jboolean JNICALL Java_com_HPCCSystems_HpccUtils__1hasNext (JNIEnv *JNIenv, jclass, jlong proxy)
  4881. {
  4882. try
  4883. {
  4884. javaembed::ECLDatasetIterator *e = (javaembed::ECLDatasetIterator *) proxy;
  4885. return e->hasNext();
  4886. }
  4887. catch (IException *E)
  4888. {
  4889. StringBuffer msg;
  4890. E->errorMessage(msg);
  4891. E->Release();
  4892. JNIenv->ThrowNew(javaembed::langIllegalArgumentExceptionClass, msg.str());
  4893. return false;
  4894. }
  4895. }
  4896. JNIEXPORT jobject JNICALL Java_com_HPCCSystems_HpccUtils__1next (JNIEnv *JNIenv, jclass, jlong proxy)
  4897. {
  4898. try
  4899. {
  4900. javaembed::ECLDatasetIterator *e = (javaembed::ECLDatasetIterator *) proxy;
  4901. return e->next();
  4902. }
  4903. catch (IException *E)
  4904. {
  4905. StringBuffer msg;
  4906. E->errorMessage(msg);
  4907. E->Release();
  4908. JNIenv->ThrowNew(javaembed::langIllegalArgumentExceptionClass, msg.str());
  4909. return NULL;
  4910. }
  4911. }
  4912. JNIEXPORT jclass JNICALL Java_com_HPCCSystems_HpccClassLoader_defineClassForEmbed(JNIEnv *env, jobject loader, jint datalen, jlong data, jstring name)
  4913. {
  4914. const char *nameChars = env->GetStringUTFChars(name, nullptr);
  4915. size32_t namelen = strlen(nameChars);
  4916. MemoryBuffer b;
  4917. b.setBuffer(datalen, (void *) data, false);
  4918. b.setEndian(__BIG_ENDIAN);
  4919. jclass ret = nullptr;
  4920. while (b.remaining())
  4921. {
  4922. uint32_t siglen; b.read(siglen);
  4923. const char *sig = (const char *) b.readDirect(siglen);
  4924. uint32_t bytecodeLen; b.read(bytecodeLen);
  4925. const jbyte * bytecode = (const jbyte *) b.readDirect(bytecodeLen);
  4926. if (siglen >= namelen && memcmp(sig, nameChars, namelen)==0 && (namelen == siglen || sig[namelen] == '.'))
  4927. {
  4928. #ifdef TRACE_CLASSFILE
  4929. DBGLOG("javaembed: loading class %s (%.*s)", nameChars, siglen, sig);
  4930. #endif
  4931. ret = env->DefineClass(nameChars, loader, bytecode, bytecodeLen);
  4932. // NOTE - if there is an exception it will get thrown next time we get into java code
  4933. break;
  4934. }
  4935. }
  4936. if (!ret)
  4937. DBGLOG("javaembed: Failed to load class %s", nameChars);
  4938. env->ReleaseStringUTFChars(name, nameChars);
  4939. return ret;
  4940. }
  4941. JNIEXPORT void JNICALL Java_com_HPCCSystems_HpccUtils_log(JNIEnv *JNIenv, jclass, jstring msg)
  4942. {
  4943. if (msg)
  4944. {
  4945. const char *text = JNIenv->GetStringUTFChars(msg, 0);
  4946. DBGLOG("javaembed: user: %s", text);
  4947. JNIenv->ReleaseStringUTFChars(msg, text);
  4948. }
  4949. }
  4950. JNIEXPORT jboolean JNICALL Java_com_HPCCSystems_HpccUtils__1isLocal(JNIEnv *JNIenv, jclass, jlong proxy)
  4951. {
  4952. const IThorActivityContext *a = (IThorActivityContext *) proxy;
  4953. return a->isLocal();
  4954. }
  4955. JNIEXPORT jint JNICALL Java_com_HPCCSystems_HpccUtils__1numSlaves(JNIEnv *JNIenv, jclass, jlong proxy)
  4956. {
  4957. const IThorActivityContext *a = (IThorActivityContext *) proxy;
  4958. return a->numSlaves();
  4959. }
  4960. JNIEXPORT jint JNICALL Java_com_HPCCSystems_HpccUtils__1numStrands(JNIEnv *JNIenv, jclass, jlong proxy)
  4961. {
  4962. const IThorActivityContext *a = (IThorActivityContext *) proxy;
  4963. return a->numStrands();
  4964. }
  4965. JNIEXPORT jint JNICALL Java_com_HPCCSystems_HpccUtils__1querySlave(JNIEnv *JNIenv, jclass, jlong proxy)
  4966. {
  4967. const IThorActivityContext *a = (IThorActivityContext *) proxy;
  4968. return a->querySlave();
  4969. }
  4970. JNIEXPORT jint JNICALL Java_com_HPCCSystems_HpccUtils__1queryStrand(JNIEnv *JNIenv, jclass, jlong proxy)
  4971. {
  4972. const IThorActivityContext *a = (IThorActivityContext *) proxy;
  4973. return a->queryStrand();
  4974. }
  4975. // Used for dynamically loading in ESDL
  4976. extern "C" DECL_EXPORT IEmbedContext *getEmbedContextDynamic()
  4977. {
  4978. return javaembed::getEmbedContext();
  4979. }