hqlcpp.cpp 434 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712171317141715171617171718171917201721172217231724172517261727172817291730173117321733173417351736173717381739174017411742174317441745174617471748174917501751175217531754175517561757175817591760176117621763176417651766176717681769177017711772177317741775177617771778177917801781178217831784178517861787178817891790179117921793179417951796179717981799180018011802180318041805180618071808180918101811181218131814181518161817181818191820182118221823182418251826182718281829183018311832183318341835183618371838183918401841184218431844184518461847184818491850185118521853185418551856185718581859186018611862186318641865186618671868186918701871187218731874187518761877187818791880188118821883188418851886188718881889189018911892189318941895189618971898189919001901190219031904190519061907190819091910191119121913191419151916191719181919192019211922192319241925192619271928192919301931193219331934193519361937193819391940194119421943194419451946194719481949195019511952195319541955195619571958195919601961196219631964196519661967196819691970197119721973197419751976197719781979198019811982198319841985198619871988198919901991199219931994199519961997199819992000200120022003200420052006200720082009201020112012201320142015201620172018201920202021202220232024202520262027202820292030203120322033203420352036203720382039204020412042204320442045204620472048204920502051205220532054205520562057205820592060206120622063206420652066206720682069207020712072207320742075207620772078207920802081208220832084208520862087208820892090209120922093209420952096209720982099210021012102210321042105210621072108210921102111211221132114211521162117211821192120212121222123212421252126212721282129213021312132213321342135213621372138213921402141214221432144214521462147214821492150215121522153215421552156215721582159216021612162216321642165216621672168216921702171217221732174217521762177217821792180218121822183218421852186218721882189219021912192219321942195219621972198219922002201220222032204220522062207220822092210221122122213221422152216221722182219222022212222222322242225222622272228222922302231223222332234223522362237223822392240224122422243224422452246224722482249225022512252225322542255225622572258225922602261226222632264226522662267226822692270227122722273227422752276227722782279228022812282228322842285228622872288228922902291229222932294229522962297229822992300230123022303230423052306230723082309231023112312231323142315231623172318231923202321232223232324232523262327232823292330233123322333233423352336233723382339234023412342234323442345234623472348234923502351235223532354235523562357235823592360236123622363236423652366236723682369237023712372237323742375237623772378237923802381238223832384238523862387238823892390239123922393239423952396239723982399240024012402240324042405240624072408240924102411241224132414241524162417241824192420242124222423242424252426242724282429243024312432243324342435243624372438243924402441244224432444244524462447244824492450245124522453245424552456245724582459246024612462246324642465246624672468246924702471247224732474247524762477247824792480248124822483248424852486248724882489249024912492249324942495249624972498249925002501250225032504250525062507250825092510251125122513251425152516251725182519252025212522252325242525252625272528252925302531253225332534253525362537253825392540254125422543254425452546254725482549255025512552255325542555255625572558255925602561256225632564256525662567256825692570257125722573257425752576257725782579258025812582258325842585258625872588258925902591259225932594259525962597259825992600260126022603260426052606260726082609261026112612261326142615261626172618261926202621262226232624262526262627262826292630263126322633263426352636263726382639264026412642264326442645264626472648264926502651265226532654265526562657265826592660266126622663266426652666266726682669267026712672267326742675267626772678267926802681268226832684268526862687268826892690269126922693269426952696269726982699270027012702270327042705270627072708270927102711271227132714271527162717271827192720272127222723272427252726272727282729273027312732273327342735273627372738273927402741274227432744274527462747274827492750275127522753275427552756275727582759276027612762276327642765276627672768276927702771277227732774277527762777277827792780278127822783278427852786278727882789279027912792279327942795279627972798279928002801280228032804280528062807280828092810281128122813281428152816281728182819282028212822282328242825282628272828282928302831283228332834283528362837283828392840284128422843284428452846284728482849285028512852285328542855285628572858285928602861286228632864286528662867286828692870287128722873287428752876287728782879288028812882288328842885288628872888288928902891289228932894289528962897289828992900290129022903290429052906290729082909291029112912291329142915291629172918291929202921292229232924292529262927292829292930293129322933293429352936293729382939294029412942294329442945294629472948294929502951295229532954295529562957295829592960296129622963296429652966296729682969297029712972297329742975297629772978297929802981298229832984298529862987298829892990299129922993299429952996299729982999300030013002300330043005300630073008300930103011301230133014301530163017301830193020302130223023302430253026302730283029303030313032303330343035303630373038303930403041304230433044304530463047304830493050305130523053305430553056305730583059306030613062306330643065306630673068306930703071307230733074307530763077307830793080308130823083308430853086308730883089309030913092309330943095309630973098309931003101310231033104310531063107310831093110311131123113311431153116311731183119312031213122312331243125312631273128312931303131313231333134313531363137313831393140314131423143314431453146314731483149315031513152315331543155315631573158315931603161316231633164316531663167316831693170317131723173317431753176317731783179318031813182318331843185318631873188318931903191319231933194319531963197319831993200320132023203320432053206320732083209321032113212321332143215321632173218321932203221322232233224322532263227322832293230323132323233323432353236323732383239324032413242324332443245324632473248324932503251325232533254325532563257325832593260326132623263326432653266326732683269327032713272327332743275327632773278327932803281328232833284328532863287328832893290329132923293329432953296329732983299330033013302330333043305330633073308330933103311331233133314331533163317331833193320332133223323332433253326332733283329333033313332333333343335333633373338333933403341334233433344334533463347334833493350335133523353335433553356335733583359336033613362336333643365336633673368336933703371337233733374337533763377337833793380338133823383338433853386338733883389339033913392339333943395339633973398339934003401340234033404340534063407340834093410341134123413341434153416341734183419342034213422342334243425342634273428342934303431343234333434343534363437343834393440344134423443344434453446344734483449345034513452345334543455345634573458345934603461346234633464346534663467346834693470347134723473347434753476347734783479348034813482348334843485348634873488348934903491349234933494349534963497349834993500350135023503350435053506350735083509351035113512351335143515351635173518351935203521352235233524352535263527352835293530353135323533353435353536353735383539354035413542354335443545354635473548354935503551355235533554355535563557355835593560356135623563356435653566356735683569357035713572357335743575357635773578357935803581358235833584358535863587358835893590359135923593359435953596359735983599360036013602360336043605360636073608360936103611361236133614361536163617361836193620362136223623362436253626362736283629363036313632363336343635363636373638363936403641364236433644364536463647364836493650365136523653365436553656365736583659366036613662366336643665366636673668366936703671367236733674367536763677367836793680368136823683368436853686368736883689369036913692369336943695369636973698369937003701370237033704370537063707370837093710371137123713371437153716371737183719372037213722372337243725372637273728372937303731373237333734373537363737373837393740374137423743374437453746374737483749375037513752375337543755375637573758375937603761376237633764376537663767376837693770377137723773377437753776377737783779378037813782378337843785378637873788378937903791379237933794379537963797379837993800380138023803380438053806380738083809381038113812381338143815381638173818381938203821382238233824382538263827382838293830383138323833383438353836383738383839384038413842384338443845384638473848384938503851385238533854385538563857385838593860386138623863386438653866386738683869387038713872387338743875387638773878387938803881388238833884388538863887388838893890389138923893389438953896389738983899390039013902390339043905390639073908390939103911391239133914391539163917391839193920392139223923392439253926392739283929393039313932393339343935393639373938393939403941394239433944394539463947394839493950395139523953395439553956395739583959396039613962396339643965396639673968396939703971397239733974397539763977397839793980398139823983398439853986398739883989399039913992399339943995399639973998399940004001400240034004400540064007400840094010401140124013401440154016401740184019402040214022402340244025402640274028402940304031403240334034403540364037403840394040404140424043404440454046404740484049405040514052405340544055405640574058405940604061406240634064406540664067406840694070407140724073407440754076407740784079408040814082408340844085408640874088408940904091409240934094409540964097409840994100410141024103410441054106410741084109411041114112411341144115411641174118411941204121412241234124412541264127412841294130413141324133413441354136413741384139414041414142414341444145414641474148414941504151415241534154415541564157415841594160416141624163416441654166416741684169417041714172417341744175417641774178417941804181418241834184418541864187418841894190419141924193419441954196419741984199420042014202420342044205420642074208420942104211421242134214421542164217421842194220422142224223422442254226422742284229423042314232423342344235423642374238423942404241424242434244424542464247424842494250425142524253425442554256425742584259426042614262426342644265426642674268426942704271427242734274427542764277427842794280428142824283428442854286428742884289429042914292429342944295429642974298429943004301430243034304430543064307430843094310431143124313431443154316431743184319432043214322432343244325432643274328432943304331433243334334433543364337433843394340434143424343434443454346434743484349435043514352435343544355435643574358435943604361436243634364436543664367436843694370437143724373437443754376437743784379438043814382438343844385438643874388438943904391439243934394439543964397439843994400440144024403440444054406440744084409441044114412441344144415441644174418441944204421442244234424442544264427442844294430443144324433443444354436443744384439444044414442444344444445444644474448444944504451445244534454445544564457445844594460446144624463446444654466446744684469447044714472447344744475447644774478447944804481448244834484448544864487448844894490449144924493449444954496449744984499450045014502450345044505450645074508450945104511451245134514451545164517451845194520452145224523452445254526452745284529453045314532453345344535453645374538453945404541454245434544454545464547454845494550455145524553455445554556455745584559456045614562456345644565456645674568456945704571457245734574457545764577457845794580458145824583458445854586458745884589459045914592459345944595459645974598459946004601460246034604460546064607460846094610461146124613461446154616461746184619462046214622462346244625462646274628462946304631463246334634463546364637463846394640464146424643464446454646464746484649465046514652465346544655465646574658465946604661466246634664466546664667466846694670467146724673467446754676467746784679468046814682468346844685468646874688468946904691469246934694469546964697469846994700470147024703470447054706470747084709471047114712471347144715471647174718471947204721472247234724472547264727472847294730473147324733473447354736473747384739474047414742474347444745474647474748474947504751475247534754475547564757475847594760476147624763476447654766476747684769477047714772477347744775477647774778477947804781478247834784478547864787478847894790479147924793479447954796479747984799480048014802480348044805480648074808480948104811481248134814481548164817481848194820482148224823482448254826482748284829483048314832483348344835483648374838483948404841484248434844484548464847484848494850485148524853485448554856485748584859486048614862486348644865486648674868486948704871487248734874487548764877487848794880488148824883488448854886488748884889489048914892489348944895489648974898489949004901490249034904490549064907490849094910491149124913491449154916491749184919492049214922492349244925492649274928492949304931493249334934493549364937493849394940494149424943494449454946494749484949495049514952495349544955495649574958495949604961496249634964496549664967496849694970497149724973497449754976497749784979498049814982498349844985498649874988498949904991499249934994499549964997499849995000500150025003500450055006500750085009501050115012501350145015501650175018501950205021502250235024502550265027502850295030503150325033503450355036503750385039504050415042504350445045504650475048504950505051505250535054505550565057505850595060506150625063506450655066506750685069507050715072507350745075507650775078507950805081508250835084508550865087508850895090509150925093509450955096509750985099510051015102510351045105510651075108510951105111511251135114511551165117511851195120512151225123512451255126512751285129513051315132513351345135513651375138513951405141514251435144514551465147514851495150515151525153515451555156515751585159516051615162516351645165516651675168516951705171517251735174517551765177517851795180518151825183518451855186518751885189519051915192519351945195519651975198519952005201520252035204520552065207520852095210521152125213521452155216521752185219522052215222522352245225522652275228522952305231523252335234523552365237523852395240524152425243524452455246524752485249525052515252525352545255525652575258525952605261526252635264526552665267526852695270527152725273527452755276527752785279528052815282528352845285528652875288528952905291529252935294529552965297529852995300530153025303530453055306530753085309531053115312531353145315531653175318531953205321532253235324532553265327532853295330533153325333533453355336533753385339534053415342534353445345534653475348534953505351535253535354535553565357535853595360536153625363536453655366536753685369537053715372537353745375537653775378537953805381538253835384538553865387538853895390539153925393539453955396539753985399540054015402540354045405540654075408540954105411541254135414541554165417541854195420542154225423542454255426542754285429543054315432543354345435543654375438543954405441544254435444544554465447544854495450545154525453545454555456545754585459546054615462546354645465546654675468546954705471547254735474547554765477547854795480548154825483548454855486548754885489549054915492549354945495549654975498549955005501550255035504550555065507550855095510551155125513551455155516551755185519552055215522552355245525552655275528552955305531553255335534553555365537553855395540554155425543554455455546554755485549555055515552555355545555555655575558555955605561556255635564556555665567556855695570557155725573557455755576557755785579558055815582558355845585558655875588558955905591559255935594559555965597559855995600560156025603560456055606560756085609561056115612561356145615561656175618561956205621562256235624562556265627562856295630563156325633563456355636563756385639564056415642564356445645564656475648564956505651565256535654565556565657565856595660566156625663566456655666566756685669567056715672567356745675567656775678567956805681568256835684568556865687568856895690569156925693569456955696569756985699570057015702570357045705570657075708570957105711571257135714571557165717571857195720572157225723572457255726572757285729573057315732573357345735573657375738573957405741574257435744574557465747574857495750575157525753575457555756575757585759576057615762576357645765576657675768576957705771577257735774577557765777577857795780578157825783578457855786578757885789579057915792579357945795579657975798579958005801580258035804580558065807580858095810581158125813581458155816581758185819582058215822582358245825582658275828582958305831583258335834583558365837583858395840584158425843584458455846584758485849585058515852585358545855585658575858585958605861586258635864586558665867586858695870587158725873587458755876587758785879588058815882588358845885588658875888588958905891589258935894589558965897589858995900590159025903590459055906590759085909591059115912591359145915591659175918591959205921592259235924592559265927592859295930593159325933593459355936593759385939594059415942594359445945594659475948594959505951595259535954595559565957595859595960596159625963596459655966596759685969597059715972597359745975597659775978597959805981598259835984598559865987598859895990599159925993599459955996599759985999600060016002600360046005600660076008600960106011601260136014601560166017601860196020602160226023602460256026602760286029603060316032603360346035603660376038603960406041604260436044604560466047604860496050605160526053605460556056605760586059606060616062606360646065606660676068606960706071607260736074607560766077607860796080608160826083608460856086608760886089609060916092609360946095609660976098609961006101610261036104610561066107610861096110611161126113611461156116611761186119612061216122612361246125612661276128612961306131613261336134613561366137613861396140614161426143614461456146614761486149615061516152615361546155615661576158615961606161616261636164616561666167616861696170617161726173617461756176617761786179618061816182618361846185618661876188618961906191619261936194619561966197619861996200620162026203620462056206620762086209621062116212621362146215621662176218621962206221622262236224622562266227622862296230623162326233623462356236623762386239624062416242624362446245624662476248624962506251625262536254625562566257625862596260626162626263626462656266626762686269627062716272627362746275627662776278627962806281628262836284628562866287628862896290629162926293629462956296629762986299630063016302630363046305630663076308630963106311631263136314631563166317631863196320632163226323632463256326632763286329633063316332633363346335633663376338633963406341634263436344634563466347634863496350635163526353635463556356635763586359636063616362636363646365636663676368636963706371637263736374637563766377637863796380638163826383638463856386638763886389639063916392639363946395639663976398639964006401640264036404640564066407640864096410641164126413641464156416641764186419642064216422642364246425642664276428642964306431643264336434643564366437643864396440644164426443644464456446644764486449645064516452645364546455645664576458645964606461646264636464646564666467646864696470647164726473647464756476647764786479648064816482648364846485648664876488648964906491649264936494649564966497649864996500650165026503650465056506650765086509651065116512651365146515651665176518651965206521652265236524652565266527652865296530653165326533653465356536653765386539654065416542654365446545654665476548654965506551655265536554655565566557655865596560656165626563656465656566656765686569657065716572657365746575657665776578657965806581658265836584658565866587658865896590659165926593659465956596659765986599660066016602660366046605660666076608660966106611661266136614661566166617661866196620662166226623662466256626662766286629663066316632663366346635663666376638663966406641664266436644664566466647664866496650665166526653665466556656665766586659666066616662666366646665666666676668666966706671667266736674667566766677667866796680668166826683668466856686668766886689669066916692669366946695669666976698669967006701670267036704670567066707670867096710671167126713671467156716671767186719672067216722672367246725672667276728672967306731673267336734673567366737673867396740674167426743674467456746674767486749675067516752675367546755675667576758675967606761676267636764676567666767676867696770677167726773677467756776677767786779678067816782678367846785678667876788678967906791679267936794679567966797679867996800680168026803680468056806680768086809681068116812681368146815681668176818681968206821682268236824682568266827682868296830683168326833683468356836683768386839684068416842684368446845684668476848684968506851685268536854685568566857685868596860686168626863686468656866686768686869687068716872687368746875687668776878687968806881688268836884688568866887688868896890689168926893689468956896689768986899690069016902690369046905690669076908690969106911691269136914691569166917691869196920692169226923692469256926692769286929693069316932693369346935693669376938693969406941694269436944694569466947694869496950695169526953695469556956695769586959696069616962696369646965696669676968696969706971697269736974697569766977697869796980698169826983698469856986698769886989699069916992699369946995699669976998699970007001700270037004700570067007700870097010701170127013701470157016701770187019702070217022702370247025702670277028702970307031703270337034703570367037703870397040704170427043704470457046704770487049705070517052705370547055705670577058705970607061706270637064706570667067706870697070707170727073707470757076707770787079708070817082708370847085708670877088708970907091709270937094709570967097709870997100710171027103710471057106710771087109711071117112711371147115711671177118711971207121712271237124712571267127712871297130713171327133713471357136713771387139714071417142714371447145714671477148714971507151715271537154715571567157715871597160716171627163716471657166716771687169717071717172717371747175717671777178717971807181718271837184718571867187718871897190719171927193719471957196719771987199720072017202720372047205720672077208720972107211721272137214721572167217721872197220722172227223722472257226722772287229723072317232723372347235723672377238723972407241724272437244724572467247724872497250725172527253725472557256725772587259726072617262726372647265726672677268726972707271727272737274727572767277727872797280728172827283728472857286728772887289729072917292729372947295729672977298729973007301730273037304730573067307730873097310731173127313731473157316731773187319732073217322732373247325732673277328732973307331733273337334733573367337733873397340734173427343734473457346734773487349735073517352735373547355735673577358735973607361736273637364736573667367736873697370737173727373737473757376737773787379738073817382738373847385738673877388738973907391739273937394739573967397739873997400740174027403740474057406740774087409741074117412741374147415741674177418741974207421742274237424742574267427742874297430743174327433743474357436743774387439744074417442744374447445744674477448744974507451745274537454745574567457745874597460746174627463746474657466746774687469747074717472747374747475747674777478747974807481748274837484748574867487748874897490749174927493749474957496749774987499750075017502750375047505750675077508750975107511751275137514751575167517751875197520752175227523752475257526752775287529753075317532753375347535753675377538753975407541754275437544754575467547754875497550755175527553755475557556755775587559756075617562756375647565756675677568756975707571757275737574757575767577757875797580758175827583758475857586758775887589759075917592759375947595759675977598759976007601760276037604760576067607760876097610761176127613761476157616761776187619762076217622762376247625762676277628762976307631763276337634763576367637763876397640764176427643764476457646764776487649765076517652765376547655765676577658765976607661766276637664766576667667766876697670767176727673767476757676767776787679768076817682768376847685768676877688768976907691769276937694769576967697769876997700770177027703770477057706770777087709771077117712771377147715771677177718771977207721772277237724772577267727772877297730773177327733773477357736773777387739774077417742774377447745774677477748774977507751775277537754775577567757775877597760776177627763776477657766776777687769777077717772777377747775777677777778777977807781778277837784778577867787778877897790779177927793779477957796779777987799780078017802780378047805780678077808780978107811781278137814781578167817781878197820782178227823782478257826782778287829783078317832783378347835783678377838783978407841784278437844784578467847784878497850785178527853785478557856785778587859786078617862786378647865786678677868786978707871787278737874787578767877787878797880788178827883788478857886788778887889789078917892789378947895789678977898789979007901790279037904790579067907790879097910791179127913791479157916791779187919792079217922792379247925792679277928792979307931793279337934793579367937793879397940794179427943794479457946794779487949795079517952795379547955795679577958795979607961796279637964796579667967796879697970797179727973797479757976797779787979798079817982798379847985798679877988798979907991799279937994799579967997799879998000800180028003800480058006800780088009801080118012801380148015801680178018801980208021802280238024802580268027802880298030803180328033803480358036803780388039804080418042804380448045804680478048804980508051805280538054805580568057805880598060806180628063806480658066806780688069807080718072807380748075807680778078807980808081808280838084808580868087808880898090809180928093809480958096809780988099810081018102810381048105810681078108810981108111811281138114811581168117811881198120812181228123812481258126812781288129813081318132813381348135813681378138813981408141814281438144814581468147814881498150815181528153815481558156815781588159816081618162816381648165816681678168816981708171817281738174817581768177817881798180818181828183818481858186818781888189819081918192819381948195819681978198819982008201820282038204820582068207820882098210821182128213821482158216821782188219822082218222822382248225822682278228822982308231823282338234823582368237823882398240824182428243824482458246824782488249825082518252825382548255825682578258825982608261826282638264826582668267826882698270827182728273827482758276827782788279828082818282828382848285828682878288828982908291829282938294829582968297829882998300830183028303830483058306830783088309831083118312831383148315831683178318831983208321832283238324832583268327832883298330833183328333833483358336833783388339834083418342834383448345834683478348834983508351835283538354835583568357835883598360836183628363836483658366836783688369837083718372837383748375837683778378837983808381838283838384838583868387838883898390839183928393839483958396839783988399840084018402840384048405840684078408840984108411841284138414841584168417841884198420842184228423842484258426842784288429843084318432843384348435843684378438843984408441844284438444844584468447844884498450845184528453845484558456845784588459846084618462846384648465846684678468846984708471847284738474847584768477847884798480848184828483848484858486848784888489849084918492849384948495849684978498849985008501850285038504850585068507850885098510851185128513851485158516851785188519852085218522852385248525852685278528852985308531853285338534853585368537853885398540854185428543854485458546854785488549855085518552855385548555855685578558855985608561856285638564856585668567856885698570857185728573857485758576857785788579858085818582858385848585858685878588858985908591859285938594859585968597859885998600860186028603860486058606860786088609861086118612861386148615861686178618861986208621862286238624862586268627862886298630863186328633863486358636863786388639864086418642864386448645864686478648864986508651865286538654865586568657865886598660866186628663866486658666866786688669867086718672867386748675867686778678867986808681868286838684868586868687868886898690869186928693869486958696869786988699870087018702870387048705870687078708870987108711871287138714871587168717871887198720872187228723872487258726872787288729873087318732873387348735873687378738873987408741874287438744874587468747874887498750875187528753875487558756875787588759876087618762876387648765876687678768876987708771877287738774877587768777877887798780878187828783878487858786878787888789879087918792879387948795879687978798879988008801880288038804880588068807880888098810881188128813881488158816881788188819882088218822882388248825882688278828882988308831883288338834883588368837883888398840884188428843884488458846884788488849885088518852885388548855885688578858885988608861886288638864886588668867886888698870887188728873887488758876887788788879888088818882888388848885888688878888888988908891889288938894889588968897889888998900890189028903890489058906890789088909891089118912891389148915891689178918891989208921892289238924892589268927892889298930893189328933893489358936893789388939894089418942894389448945894689478948894989508951895289538954895589568957895889598960896189628963896489658966896789688969897089718972897389748975897689778978897989808981898289838984898589868987898889898990899189928993899489958996899789988999900090019002900390049005900690079008900990109011901290139014901590169017901890199020902190229023902490259026902790289029903090319032903390349035903690379038903990409041904290439044904590469047904890499050905190529053905490559056905790589059906090619062906390649065906690679068906990709071907290739074907590769077907890799080908190829083908490859086908790889089909090919092909390949095909690979098909991009101910291039104910591069107910891099110911191129113911491159116911791189119912091219122912391249125912691279128912991309131913291339134913591369137913891399140914191429143914491459146914791489149915091519152915391549155915691579158915991609161916291639164916591669167916891699170917191729173917491759176917791789179918091819182918391849185918691879188918991909191919291939194919591969197919891999200920192029203920492059206920792089209921092119212921392149215921692179218921992209221922292239224922592269227922892299230923192329233923492359236923792389239924092419242924392449245924692479248924992509251925292539254925592569257925892599260926192629263926492659266926792689269927092719272927392749275927692779278927992809281928292839284928592869287928892899290929192929293929492959296929792989299930093019302930393049305930693079308930993109311931293139314931593169317931893199320932193229323932493259326932793289329933093319332933393349335933693379338933993409341934293439344934593469347934893499350935193529353935493559356935793589359936093619362936393649365936693679368936993709371937293739374937593769377937893799380938193829383938493859386938793889389939093919392939393949395939693979398939994009401940294039404940594069407940894099410941194129413941494159416941794189419942094219422942394249425942694279428942994309431943294339434943594369437943894399440944194429443944494459446944794489449945094519452945394549455945694579458945994609461946294639464946594669467946894699470947194729473947494759476947794789479948094819482948394849485948694879488948994909491949294939494949594969497949894999500950195029503950495059506950795089509951095119512951395149515951695179518951995209521952295239524952595269527952895299530953195329533953495359536953795389539954095419542954395449545954695479548954995509551955295539554955595569557955895599560956195629563956495659566956795689569957095719572957395749575957695779578957995809581958295839584958595869587958895899590959195929593959495959596959795989599960096019602960396049605960696079608960996109611961296139614961596169617961896199620962196229623962496259626962796289629963096319632963396349635963696379638963996409641964296439644964596469647964896499650965196529653965496559656965796589659966096619662966396649665966696679668966996709671967296739674967596769677967896799680968196829683968496859686968796889689969096919692969396949695969696979698969997009701970297039704970597069707970897099710971197129713971497159716971797189719972097219722972397249725972697279728972997309731973297339734973597369737973897399740974197429743974497459746974797489749975097519752975397549755975697579758975997609761976297639764976597669767976897699770977197729773977497759776977797789779978097819782978397849785978697879788978997909791979297939794979597969797979897999800980198029803980498059806980798089809981098119812981398149815981698179818981998209821982298239824982598269827982898299830983198329833983498359836983798389839984098419842984398449845984698479848984998509851985298539854985598569857985898599860986198629863986498659866986798689869987098719872987398749875987698779878987998809881988298839884988598869887988898899890989198929893989498959896989798989899990099019902990399049905990699079908990999109911991299139914991599169917991899199920992199229923992499259926992799289929993099319932993399349935993699379938993999409941994299439944994599469947994899499950995199529953995499559956995799589959996099619962996399649965996699679968996999709971997299739974997599769977997899799980998199829983998499859986998799889989999099919992999399949995999699979998999910000100011000210003100041000510006100071000810009100101001110012100131001410015100161001710018100191002010021100221002310024100251002610027100281002910030100311003210033100341003510036100371003810039100401004110042100431004410045100461004710048100491005010051100521005310054100551005610057100581005910060100611006210063100641006510066100671006810069100701007110072100731007410075100761007710078100791008010081100821008310084100851008610087100881008910090100911009210093100941009510096100971009810099101001010110102101031010410105101061010710108101091011010111101121011310114101151011610117101181011910120101211012210123101241012510126101271012810129101301013110132101331013410135101361013710138101391014010141101421014310144101451014610147101481014910150101511015210153101541015510156101571015810159101601016110162101631016410165101661016710168101691017010171101721017310174101751017610177101781017910180101811018210183101841018510186101871018810189101901019110192101931019410195101961019710198101991020010201102021020310204102051020610207102081020910210102111021210213102141021510216102171021810219102201022110222102231022410225102261022710228102291023010231102321023310234102351023610237102381023910240102411024210243102441024510246102471024810249102501025110252102531025410255102561025710258102591026010261102621026310264102651026610267102681026910270102711027210273102741027510276102771027810279102801028110282102831028410285102861028710288102891029010291102921029310294102951029610297102981029910300103011030210303103041030510306103071030810309103101031110312103131031410315103161031710318103191032010321103221032310324103251032610327103281032910330103311033210333103341033510336103371033810339103401034110342103431034410345103461034710348103491035010351103521035310354103551035610357103581035910360103611036210363103641036510366103671036810369103701037110372103731037410375103761037710378103791038010381103821038310384103851038610387103881038910390103911039210393103941039510396103971039810399104001040110402104031040410405104061040710408104091041010411104121041310414104151041610417104181041910420104211042210423104241042510426104271042810429104301043110432104331043410435104361043710438104391044010441104421044310444104451044610447104481044910450104511045210453104541045510456104571045810459104601046110462104631046410465104661046710468104691047010471104721047310474104751047610477104781047910480104811048210483104841048510486104871048810489104901049110492104931049410495104961049710498104991050010501105021050310504105051050610507105081050910510105111051210513105141051510516105171051810519105201052110522105231052410525105261052710528105291053010531105321053310534105351053610537105381053910540105411054210543105441054510546105471054810549105501055110552105531055410555105561055710558105591056010561105621056310564105651056610567105681056910570105711057210573105741057510576105771057810579105801058110582105831058410585105861058710588105891059010591105921059310594105951059610597105981059910600106011060210603106041060510606106071060810609106101061110612106131061410615106161061710618106191062010621106221062310624106251062610627106281062910630106311063210633106341063510636106371063810639106401064110642106431064410645106461064710648106491065010651106521065310654106551065610657106581065910660106611066210663106641066510666106671066810669106701067110672106731067410675106761067710678106791068010681106821068310684106851068610687106881068910690106911069210693106941069510696106971069810699107001070110702107031070410705107061070710708107091071010711107121071310714107151071610717107181071910720107211072210723107241072510726107271072810729107301073110732107331073410735107361073710738107391074010741107421074310744107451074610747107481074910750107511075210753107541075510756107571075810759107601076110762107631076410765107661076710768107691077010771107721077310774107751077610777107781077910780107811078210783107841078510786107871078810789107901079110792107931079410795107961079710798107991080010801108021080310804108051080610807108081080910810108111081210813108141081510816108171081810819108201082110822108231082410825108261082710828108291083010831108321083310834108351083610837108381083910840108411084210843108441084510846108471084810849108501085110852108531085410855108561085710858108591086010861108621086310864108651086610867108681086910870108711087210873108741087510876108771087810879108801088110882108831088410885108861088710888108891089010891108921089310894108951089610897108981089910900109011090210903109041090510906109071090810909109101091110912109131091410915109161091710918109191092010921109221092310924109251092610927109281092910930109311093210933109341093510936109371093810939109401094110942109431094410945109461094710948109491095010951109521095310954109551095610957109581095910960109611096210963109641096510966109671096810969109701097110972109731097410975109761097710978109791098010981109821098310984109851098610987109881098910990109911099210993109941099510996109971099810999110001100111002110031100411005110061100711008110091101011011110121101311014110151101611017110181101911020110211102211023110241102511026110271102811029110301103111032110331103411035110361103711038110391104011041110421104311044110451104611047110481104911050110511105211053110541105511056110571105811059110601106111062110631106411065110661106711068110691107011071110721107311074110751107611077110781107911080110811108211083110841108511086110871108811089110901109111092110931109411095110961109711098110991110011101111021110311104111051110611107111081110911110111111111211113111141111511116111171111811119111201112111122111231112411125111261112711128111291113011131111321113311134111351113611137111381113911140111411114211143111441114511146111471114811149111501115111152111531115411155111561115711158111591116011161111621116311164111651116611167111681116911170111711117211173111741117511176111771117811179111801118111182111831118411185111861118711188111891119011191111921119311194111951119611197111981119911200112011120211203112041120511206112071120811209112101121111212112131121411215112161121711218112191122011221112221122311224112251122611227112281122911230112311123211233112341123511236112371123811239112401124111242112431124411245112461124711248112491125011251112521125311254112551125611257112581125911260112611126211263112641126511266112671126811269112701127111272112731127411275112761127711278112791128011281112821128311284112851128611287112881128911290112911129211293112941129511296112971129811299113001130111302113031130411305113061130711308113091131011311113121131311314113151131611317113181131911320113211132211323113241132511326113271132811329113301133111332113331133411335113361133711338113391134011341113421134311344113451134611347113481134911350113511135211353113541135511356113571135811359113601136111362113631136411365113661136711368113691137011371113721137311374113751137611377113781137911380113811138211383113841138511386113871138811389113901139111392113931139411395113961139711398113991140011401114021140311404114051140611407114081140911410114111141211413114141141511416114171141811419114201142111422114231142411425114261142711428114291143011431114321143311434114351143611437114381143911440114411144211443114441144511446114471144811449114501145111452114531145411455114561145711458114591146011461114621146311464114651146611467114681146911470114711147211473114741147511476114771147811479114801148111482114831148411485114861148711488114891149011491114921149311494114951149611497114981149911500115011150211503115041150511506115071150811509115101151111512115131151411515115161151711518115191152011521115221152311524115251152611527115281152911530115311153211533115341153511536115371153811539115401154111542115431154411545115461154711548115491155011551115521155311554115551155611557115581155911560115611156211563115641156511566115671156811569115701157111572115731157411575115761157711578115791158011581115821158311584115851158611587115881158911590115911159211593115941159511596115971159811599116001160111602116031160411605116061160711608116091161011611116121161311614116151161611617116181161911620116211162211623116241162511626116271162811629116301163111632116331163411635116361163711638116391164011641116421164311644116451164611647116481164911650116511165211653116541165511656116571165811659116601166111662116631166411665116661166711668116691167011671116721167311674116751167611677116781167911680116811168211683116841168511686116871168811689116901169111692116931169411695116961169711698116991170011701117021170311704117051170611707117081170911710117111171211713117141171511716117171171811719117201172111722117231172411725117261172711728117291173011731117321173311734117351173611737117381173911740117411174211743117441174511746117471174811749117501175111752117531175411755117561175711758117591176011761117621176311764117651176611767117681176911770117711177211773117741177511776117771177811779117801178111782117831178411785117861178711788117891179011791117921179311794117951179611797117981179911800118011180211803118041180511806118071180811809118101181111812118131181411815118161181711818118191182011821118221182311824118251182611827118281182911830118311183211833118341183511836118371183811839118401184111842118431184411845118461184711848118491185011851118521185311854118551185611857118581185911860118611186211863118641186511866118671186811869118701187111872118731187411875118761187711878118791188011881118821188311884118851188611887118881188911890118911189211893118941189511896118971189811899119001190111902119031190411905119061190711908119091191011911119121191311914119151191611917119181191911920119211192211923119241192511926119271192811929119301193111932119331193411935119361193711938119391194011941119421194311944119451194611947119481194911950119511195211953119541195511956119571195811959119601196111962119631196411965119661196711968119691197011971119721197311974119751197611977119781197911980119811198211983119841198511986119871198811989119901199111992119931199411995119961199711998119991200012001120021200312004120051200612007120081200912010120111201212013120141201512016120171201812019120201202112022120231202412025120261202712028120291203012031120321203312034120351203612037120381203912040120411204212043120441204512046120471204812049120501205112052120531205412055120561205712058120591206012061120621206312064120651206612067120681206912070120711207212073120741207512076120771207812079120801208112082120831208412085120861208712088120891209012091120921209312094120951209612097120981209912100121011210212103121041210512106121071210812109121101211112112121131211412115121161211712118121191212012121121221212312124121251212612127121281212912130121311213212133121341213512136121371213812139121401214112142121431214412145121461214712148121491215012151121521215312154121551215612157121581215912160121611216212163121641216512166121671216812169121701217112172121731217412175121761217712178121791218012181121821218312184121851218612187121881218912190121911219212193121941219512196121971219812199122001220112202122031220412205122061220712208122091221012211122121221312214122151221612217122181221912220122211222212223122241222512226122271222812229122301223112232122331223412235122361223712238122391224012241122421224312244122451224612247122481224912250122511225212253122541225512256122571225812259122601226112262122631226412265122661226712268122691227012271122721227312274122751227612277122781227912280122811228212283122841228512286122871228812289122901229112292122931229412295122961229712298122991230012301123021230312304123051230612307123081230912310123111231212313123141231512316123171231812319123201232112322123231232412325123261232712328123291233012331123321233312334123351233612337123381233912340123411234212343123441234512346123471234812349123501235112352123531235412355123561235712358123591236012361123621236312364123651236612367123681236912370123711237212373123741237512376123771237812379123801238112382123831238412385123861238712388123891239012391123921239312394123951239612397123981239912400124011240212403124041240512406124071240812409124101241112412124131241412415124161241712418124191242012421124221242312424124251242612427124281242912430124311243212433124341243512436124371243812439124401244112442124431244412445124461244712448124491245012451124521245312454124551245612457124581245912460124611246212463124641246512466124671246812469124701247112472124731247412475124761247712478124791248012481124821248312484124851248612487124881248912490124911249212493124941249512496124971249812499125001250112502125031250412505125061250712508125091251012511125121251312514125151251612517125181251912520125211252212523125241252512526125271252812529125301253112532125331253412535125361253712538125391254012541125421254312544125451254612547125481254912550125511255212553125541255512556125571255812559125601256112562125631256412565125661256712568125691257012571125721257312574125751257612577125781257912580125811258212583125841258512586125871258812589125901259112592125931259412595125961259712598125991260012601126021260312604126051260612607126081260912610126111261212613126141261512616126171261812619126201262112622126231262412625126261262712628126291263012631126321263312634126351263612637126381263912640126411264212643126441264512646126471264812649126501265112652126531265412655126561265712658126591266012661126621266312664126651266612667126681266912670126711267212673126741267512676126771267812679126801268112682126831268412685126861268712688126891269012691126921269312694126951269612697126981269912700127011270212703127041270512706127071270812709127101271112712127131271412715127161271712718127191272012721127221272312724127251272612727127281272912730127311273212733127341273512736127371273812739127401274112742127431274412745127461274712748127491275012751127521275312754127551275612757127581275912760127611276212763127641276512766127671276812769127701277112772127731277412775127761277712778127791278012781127821278312784127851278612787127881278912790127911279212793127941279512796127971279812799
  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 <algorithm>
  15. #include "jliball.hpp"
  16. #include "jlib.hpp"
  17. #include "jexcept.hpp"
  18. #include "jmisc.hpp"
  19. #include "javahash.hpp"
  20. #include "jmd5.hpp"
  21. #include "jfile.hpp"
  22. #include "eclhelper.hpp"
  23. #include "hql.hpp"
  24. #include "hqlfunc.hpp"
  25. #include "hqlattr.hpp"
  26. #include "hqlcpp.ipp"
  27. #include "hqlwcpp.hpp"
  28. #include "hqlcpputil.hpp"
  29. #include "hqlres.hpp"
  30. #include "hqlerror.hpp"
  31. #include "hqlcerrors.hpp"
  32. #include "hqlcatom.hpp"
  33. #include "hqlpmap.hpp"
  34. #include "hqlthql.hpp"
  35. #include "hqlfold.hpp"
  36. #include "eclrtl.hpp"
  37. #include "hqllib.ipp"
  38. #include "hqlnlp.ipp"
  39. #include "hqlutil.hpp"
  40. #include "hqltcppc.ipp"
  41. #include "hqlttcpp.ipp"
  42. #include "hqlccommon.hpp"
  43. #include "hqlopt.hpp"
  44. #include "hqlpopt.hpp"
  45. #include "hqlcse.ipp"
  46. #include "thorplugin.hpp"
  47. #include "hqlinline.hpp"
  48. #include "hqlusage.hpp"
  49. #ifdef _DEBUG
  50. //#define ADD_ASSIGNMENT_COMMENTS
  51. //#define ADD_RESOURCE_AS_CPP_COMMENT
  52. #endif
  53. //Defaults for various options.
  54. #define COMPLEXITY_TO_HOIST 2
  55. #define INLINE_COMPARE_THRESHOLD 2 //7 // above this, a loop is generated
  56. #define MAX_NESTED_CASES 8
  57. #define MAX_SIMPLE_VAR_SIZE 99999
  58. #define MAX_STATIC_ROW_SIZE 10000
  59. #define MAX_LOCAL_ROW_SIZE 32
  60. #define DEFAULT_NLP_DETAIL 0
  61. #define DEFAULT_EXPIRY_PERIOD 7
  62. #define DEFAULT_PERSIST_EXPIRY_PERIOD 0 // Persists that expire by default have a period defined in sasha
  63. #ifdef _WIN32
  64. #define DEFAULT_ACTIVITIES_PER_CPP 800 // windows compiler is fast, linker is slow, but compiler also has quite a small compile limit
  65. #else
  66. #define DEFAULT_ACTIVITIES_PER_CPP 500 // gcc assembler is v.slow
  67. #endif
  68. //MORE: Simple vars don't work if they are made class members...
  69. //#define SEARCH_VARIABLE "v211"
  70. //===========================================================================
  71. static CriticalSection * systemCS;
  72. static IHqlScope * cppSystemScope;
  73. //===========================================================================
  74. MODULE_INIT(INIT_PRIORITY_STANDARD)
  75. {
  76. systemCS = new CriticalSection;
  77. return true;
  78. }
  79. MODULE_EXIT()
  80. {
  81. ::Release(cppSystemScope);
  82. cppSystemScope = NULL;
  83. delete systemCS;
  84. }
  85. //---------------------------------------------------------------------------
  86. class SubStringInfo : public SubStringHelper
  87. {
  88. public:
  89. SubStringInfo(IHqlExpression * _expr) : SubStringHelper(_expr) { expr = _expr; }
  90. void bindToFrom(HqlCppTranslator & translator, BuildCtx & ctx);
  91. public:
  92. IHqlExpression * expr;
  93. CHqlBoundExpr boundFrom;
  94. CHqlBoundExpr boundTo;
  95. };
  96. void SubStringInfo::bindToFrom(HqlCppTranslator & translator, BuildCtx & ctx)
  97. {
  98. if (to && to->isAttribute())
  99. throwError(HQLERR_StarRangeOnlyInJoinCondition);
  100. if (from == to)
  101. {
  102. if (from)
  103. {
  104. translator.buildSimpleExpr(ctx, from, boundFrom);
  105. boundTo.expr.set(boundFrom.expr);
  106. }
  107. }
  108. else
  109. {
  110. if (from)
  111. translator.buildCachedExpr(ctx, from, boundFrom);
  112. if (to)
  113. translator.buildCachedExpr(ctx, to, boundTo);
  114. }
  115. }
  116. //---------------------------------------------------------------------------
  117. WorkflowItem::WorkflowItem(IHqlExpression * _function) : wfid(0), function(_function), workflowOp(no_funcdef)
  118. {
  119. IHqlExpression * body = function->queryChild(0);
  120. assertex(body->getOperator() == no_outofline);
  121. IHqlExpression * ecl = body->queryChild(0);
  122. exprs.append(*createValue(no_return_stmt, makeVoidType(), LINK(ecl)));
  123. label.append(function->queryId());
  124. }
  125. IHqlExpression * WorkflowItem::getFunction() const
  126. {
  127. IHqlExpression * body = function->queryChild(0);
  128. unsigned max = exprs.ordinality();
  129. assertex(max);
  130. LinkedHqlExpr newecl = exprs.item(max-1).queryChild(0);
  131. for (unsigned i=max-1; i--; )
  132. {
  133. IHqlExpression * cur = &exprs.item(i);
  134. newecl.setown(createCompound(LINK(cur), newecl.getClear()));
  135. }
  136. OwnedHqlExpr newBody = replaceChild(body, 0, newecl);
  137. return replaceChild(function, 0, newBody);
  138. }
  139. //---------------------------------------------------------------------------
  140. IHqlExpression * DatasetReference::querySelector() const
  141. {
  142. if (side == no_none)
  143. return ds->queryNormalizedSelector();
  144. return selector;
  145. }
  146. IHqlExpression * DatasetReference::querySelSeq() const
  147. {
  148. if (side == no_none)
  149. return NULL;
  150. return selector->queryChild(1);
  151. }
  152. IHqlExpression * DatasetReference::mapCompound(IHqlExpression * expr, IHqlExpression * to) const
  153. {
  154. return replaceSelector(expr, querySelector(), to);
  155. }
  156. IHqlExpression * DatasetReference::mapScalar(IHqlExpression * expr, IHqlExpression * to) const
  157. {
  158. return replaceSelector(expr, querySelector(), to);
  159. }
  160. //---------------------------------------------------------------------------
  161. IHqlExpression * createVariable(ITypeInfo * type)
  162. {
  163. StringBuffer tempName;
  164. getUniqueId(tempName.append('v'));
  165. #ifdef _DEBUG
  166. #ifdef SEARCH_VARIABLE
  167. if (stricmp(tempName.str(), SEARCH_VARIABLE)==0)
  168. type = type;
  169. #endif
  170. #endif
  171. return ::createVariable(tempName.str(), type);
  172. }
  173. IHqlExpression * convertWrapperToPointer(IHqlExpression * expr)
  174. {
  175. ITypeInfo * type = expr->queryType();
  176. if (hasWrapperModifier(type))
  177. return createValue(no_implicitcast, makeReferenceModifier(removeModifier(type, typemod_wrapper)), LINK(expr));
  178. return LINK(expr);
  179. }
  180. IHqlExpression * ensureIndexable(IHqlExpression * expr)
  181. {
  182. ITypeInfo * type = expr->queryType();
  183. if (type->getTypeCode() == type_data)
  184. {
  185. IHqlExpression * base = queryStripCasts(expr);
  186. return createValue(no_implicitcast, makeReferenceModifier(makeStringType(type->getSize(), NULL, NULL)), LINK(base));
  187. }
  188. return convertWrapperToPointer(expr);
  189. }
  190. void extendConjunctionOwn(HqlExprAttr & cond, IHqlExpression * next)
  191. {
  192. if (cond)
  193. next = createBoolExpr(no_and, cond.getClear(), next);
  194. cond.setown(next);
  195. }
  196. inline bool isPushed(const IHqlExpression * expr)
  197. {
  198. return (expr->getOperator() == no_decimalstack);
  199. }
  200. inline bool isPushed(const CHqlBoundExpr & bound)
  201. {
  202. return isPushed(bound.expr);
  203. }
  204. bool isSimpleTranslatedStringExpr(IHqlExpression * expr)
  205. {
  206. for (;;)
  207. {
  208. node_operator op = expr->getOperator();
  209. switch (op)
  210. {
  211. case no_constant:
  212. case no_variable:
  213. case no_callback:
  214. case no_select:
  215. return true;
  216. case no_cast:
  217. case no_implicitcast:
  218. case no_typetransfer:
  219. case no_deref:
  220. case no_address:
  221. expr = expr->queryChild(0);
  222. break;
  223. case no_add:
  224. case no_sub:
  225. if (!isSimpleTranslatedStringExpr(expr->queryChild(1)))
  226. return false;
  227. expr = expr->queryChild(0);
  228. break;
  229. default:
  230. return false;
  231. }
  232. }
  233. }
  234. bool isSimpleTranslatedExpr(IHqlExpression * expr)
  235. {
  236. switch (expr->queryType()->getTypeCode())
  237. {
  238. case type_data:
  239. case type_string:
  240. case type_qstring:
  241. case type_varstring:
  242. case type_decimal:
  243. case type_unicode:
  244. case type_varunicode:
  245. //Less strict rules for strings (and decimal), because string temporaries are more expensive.
  246. return isSimpleTranslatedStringExpr(expr);
  247. case type_set:
  248. //for the moment assume set expressions are always simple once translated.
  249. return true;
  250. }
  251. for (;;)
  252. {
  253. node_operator op = expr->getOperator();
  254. switch (op)
  255. {
  256. case no_constant:
  257. case no_variable:
  258. case no_callback:
  259. case no_nullptr:
  260. return true;
  261. case no_typetransfer:
  262. expr = expr->queryChild(0);
  263. break;
  264. default:
  265. return false;
  266. }
  267. }
  268. }
  269. bool isFixedLengthList(IHqlExpression * expr)
  270. {
  271. switch (expr->getOperator())
  272. {
  273. case no_list:
  274. case no_datasetlist:
  275. case no_sortlist:
  276. return true;
  277. }
  278. return false;
  279. }
  280. bool needVarStringCompare(ITypeInfo * leftType, ITypeInfo * rightType)
  281. {
  282. unsigned lSize = leftType->getSize();
  283. unsigned rSize = rightType->getSize();
  284. return (lSize != rSize) || (lSize == UNKNOWN_LENGTH);
  285. }
  286. IIdAtom * queryStrCompareFunc(ITypeInfo * realType)
  287. {
  288. switch (realType->getTypeCode())
  289. {
  290. case type_data:
  291. return compareDataDataId;
  292. case type_qstring:
  293. return compareQStrQStrId;
  294. }
  295. ICharsetInfo * charset = realType->queryCharset();
  296. IAtom * charsetName = charset->queryName();
  297. if (charsetName == dataAtom)
  298. return compareDataDataId;
  299. if (charsetName == asciiAtom)
  300. return compareStrStrId;
  301. if (charsetName == ebcdicAtom)
  302. return compareEStrEStrId;
  303. assertex(!"Unknown string comparison");
  304. return compareStrStrId;
  305. }
  306. IHqlExpression * getAddress(IHqlExpression * expr)
  307. {
  308. if (expr->getOperator() == no_deref)
  309. {
  310. IHqlExpression * address = expr->queryChild(0);
  311. return LINK(address);
  312. }
  313. return createValue(no_address, makePointerType(expr->getType()), LINK(expr));
  314. }
  315. IHqlExpression * getRawAddress(IHqlExpression * expr)
  316. {
  317. OwnedHqlExpr raw = getAddress(expr);
  318. for (;;)
  319. {
  320. switch (raw->getOperator())
  321. {
  322. case no_cast:
  323. case no_implicitcast:
  324. break;
  325. default:
  326. return raw.getClear();
  327. }
  328. raw.set(raw->queryChild(0));
  329. }
  330. }
  331. IHqlExpression * getPointer(IHqlExpression * source)
  332. {
  333. if (source->getOperator() == no_constant)
  334. return LINK(source);
  335. ITypeInfo * type = source->queryType();
  336. Owned<ITypeInfo> newType;
  337. switch (type->getTypeCode())
  338. {
  339. case type_set:
  340. if (type->isReference())
  341. return LINK(source);
  342. newType.setown(makeReferenceModifier(LINK(queryUnqualifiedType(type))));
  343. if (hasWrapperModifier(type))
  344. {
  345. OwnedHqlExpr cast = createValue(no_implicitcast, LINK(newType), LINK(source));
  346. return createValue(no_typetransfer, LINK(newType), LINK(cast));
  347. }
  348. break;
  349. case type_dictionary:
  350. case type_table:
  351. case type_groupedtable:
  352. case type_row:
  353. case type_decimal:
  354. case type_string:
  355. case type_data:
  356. case type_qstring:
  357. case type_varstring:
  358. case type_unicode:
  359. case type_varunicode:
  360. case type_utf8:
  361. if (isTypePassedByAddress(type))
  362. {
  363. if (type->isReference())
  364. return LINK(source);
  365. newType.setown(makeReferenceModifier(LINK(queryUnqualifiedType(type))));
  366. if (hasLinkCountedModifier(type))
  367. newType.setown(makeAttributeModifier(newType.getClear(), getLinkCountedAttr()));
  368. if (hasWrapperModifier(type))
  369. return createValue(no_implicitcast, LINK(newType), LINK(source));
  370. //An array of X is implicitly converted to pointer to X so no need to do &a[0]
  371. return createValue(no_typetransfer, LINK(newType), LINK(source));
  372. }
  373. else
  374. {
  375. newType.setown(removeModifier(type, typemod_wrapper));
  376. if (hasWrapperModifier(type))
  377. return createValue(no_implicitcast, LINK(newType), LINK(source));
  378. return LINK(source);
  379. }
  380. break;
  381. case type_pointer:
  382. return LINK(source);
  383. default:
  384. newType.setown(makePointerType(LINK(type)));
  385. break;
  386. }
  387. IHqlExpression * cur = source;
  388. while (cur->getOperator() == no_typetransfer)
  389. cur = cur->queryChild(0);
  390. if (cur->getOperator() == no_deref)
  391. {
  392. IHqlExpression * address = cur->queryChild(0);
  393. if (address->queryType() == newType)
  394. return LINK(address);
  395. return createValue(no_implicitcast, newType.getClear(), LINK(address));
  396. }
  397. else
  398. return createValue(no_address, newType.getClear(), LINK(source));
  399. }
  400. bool isChildOf(IHqlExpression * parent, IHqlExpression * child)
  401. {
  402. unsigned max = parent->numChildren();
  403. unsigned idx;
  404. for (idx = 0; idx < max; idx++)
  405. {
  406. IHqlExpression * cur = parent->queryChild(idx);
  407. if (cur == child)
  408. return true;
  409. }
  410. return false;
  411. }
  412. bool canRemoveStringCast(ITypeInfo * to, ITypeInfo * from)
  413. {
  414. unsigned fromSize = from->getSize();
  415. unsigned toSize = to->getSize();
  416. //Special case string conversions that don't require us to copy any data.
  417. if ((toSize == UNKNOWN_LENGTH) || ((fromSize != UNKNOWN_LENGTH) && (toSize <= fromSize)))
  418. {
  419. switch (from->getTypeCode())
  420. {
  421. case type_varstring:
  422. if (toSize != UNKNOWN_LENGTH)
  423. break;
  424. //fall through
  425. case type_data:
  426. case type_string:
  427. {
  428. ICharsetInfo * srcset = from->queryCharset();
  429. ICharsetInfo * tgtset = to->queryCharset();
  430. //Data never calls a conversion function...
  431. if ((srcset == tgtset) || (to->getTypeCode() == type_data) || (from->getTypeCode() == type_data))
  432. return true;
  433. }
  434. case type_qstring:
  435. return false;
  436. }
  437. }
  438. return false;
  439. }
  440. bool isProjectedInRecord(IHqlExpression * record, IHqlExpression * expr)
  441. {
  442. unsigned max = record->numChildren();
  443. unsigned idx;
  444. for (idx = 0; idx < max; idx++)
  445. {
  446. IHqlExpression * cur = record->queryChild(idx);
  447. if (cur->queryChild(0) == expr)
  448. return true;
  449. }
  450. return false;
  451. }
  452. // Format the list is stored in doesn't matter, so allow constant strings to be stored by reference
  453. IHqlExpression * getOptimialListFormat(IHqlExpression * table)
  454. {
  455. if (table->isConstant() && table->getOperator() == no_list)
  456. {
  457. ITypeInfo * elemType = table->queryType()->queryChildType();
  458. if (!elemType->isReference())
  459. {
  460. switch (elemType->getTypeCode())
  461. {
  462. case type_string:
  463. case type_data:
  464. {
  465. HqlExprArray args;
  466. table->unwindList(args, no_list);
  467. return createValue(no_list, makeSetType(makeReferenceModifier(LINK(elemType))), args);
  468. }
  469. }
  470. }
  471. }
  472. return LINK(table);
  473. }
  474. bool canOptimizeAdjust(IHqlExpression * value)
  475. {
  476. switch (value->getOperator())
  477. {
  478. case no_constant:
  479. return true;
  480. case no_add:
  481. case no_sub:
  482. return value->queryChild(1)->queryValue() != NULL;
  483. }
  484. return false;
  485. }
  486. IHqlExpression * adjustValue(IHqlExpression * value, __int64 delta)
  487. {
  488. if (delta == 0)
  489. return LINK(value);
  490. ITypeInfo * type = value->queryType();
  491. switch (value->getOperator())
  492. {
  493. case no_constant:
  494. {
  495. __int64 newValue = value->queryValue()->getIntValue()+delta;
  496. if (type == sizetType)
  497. return getSizetConstant((size32_t)newValue);
  498. return createConstant(type->castFrom(true, newValue));
  499. }
  500. case no_add:
  501. {
  502. IHqlExpression * lhs = value->queryChild(0);
  503. IHqlExpression * rhs = value->queryChild(1);
  504. IValue * rhsValue = rhs->queryValue();
  505. IValue * lhsValue = lhs->queryValue();
  506. if (rhsValue)
  507. {
  508. delta += rhsValue->getIntValue();
  509. if (delta == 0)
  510. return LINK(lhs);
  511. value = lhs;
  512. }
  513. else if (lhsValue)
  514. {
  515. delta += lhsValue->getIntValue();
  516. if (delta == 0)
  517. return LINK(rhs);
  518. value = rhs;
  519. }
  520. else if (canOptimizeAdjust(rhs))
  521. return createValue(no_add, value->getType(), LINK(lhs), adjustValue(rhs, delta));
  522. break;
  523. }
  524. case no_sub:
  525. {
  526. IValue * rhsValue = value->queryChild(1)->queryValue();
  527. if (rhsValue)
  528. {
  529. IHqlExpression * lhs = value->queryChild(0);
  530. delta -= rhsValue->getIntValue();
  531. if (delta == 0)
  532. return LINK(lhs);
  533. value = lhs;
  534. }
  535. break;
  536. }
  537. case no_translated:
  538. {
  539. IHqlExpression * arg = value->queryChild(0);
  540. if (arg->queryValue())
  541. {
  542. OwnedHqlExpr newValue = adjustValue(arg, delta);
  543. return createTranslated(newValue);
  544. }
  545. break;
  546. }
  547. //optimize no_case because it is generated by a transformation of the dataset no_case
  548. case no_case:
  549. case no_mapto:
  550. {
  551. HqlExprArray args;
  552. args.append(*LINK(value->queryChild(0)));
  553. ForEachChildFrom(i, value, 1)
  554. {
  555. IHqlExpression * cur = value->queryChild(i);
  556. args.append(*adjustValue(cur, delta));
  557. }
  558. return value->clone(args);
  559. }
  560. }
  561. IHqlExpression * deltaExpr;
  562. node_operator op = no_add;
  563. if (delta < 0)
  564. {
  565. op = no_sub;
  566. delta = -delta;
  567. }
  568. if (type == sizetType || !type->isInteger())
  569. deltaExpr = getSizetConstant((size32_t)delta);
  570. else
  571. deltaExpr = createConstant(type->castFrom(true, delta));
  572. return createValue(op, LINK(type), LINK(value), deltaExpr);
  573. }
  574. IHqlExpression * adjustIndexBaseToZero(IHqlExpression * index)
  575. {
  576. return adjustValue(index, -1);
  577. }
  578. IHqlExpression * adjustIndexBaseToOne(IHqlExpression * index)
  579. {
  580. return adjustValue(index, +1);
  581. }
  582. IHqlExpression * adjustBoundIntegerValues(IHqlExpression * left, IHqlExpression * right, bool subtract)
  583. {
  584. if (!left)
  585. {
  586. if (!subtract)
  587. return LINK(right);
  588. else
  589. return getNegative(right);
  590. }
  591. assertex(queryUnqualifiedType(left->queryType()) == queryUnqualifiedType(right->queryType()));
  592. if (canOptimizeAdjust(left))
  593. {
  594. node_operator op = right->getOperator();
  595. switch (op)
  596. {
  597. case no_constant:
  598. {
  599. __int64 rhsValue = right->queryValue()->getIntValue();
  600. if (subtract)
  601. rhsValue = -rhsValue;
  602. return adjustValue(left, rhsValue);
  603. }
  604. case no_add:
  605. case no_sub:
  606. {
  607. IHqlExpression * rl = right->queryChild(0);
  608. IHqlExpression * rr = right->queryChild(1);
  609. if (rr->getOperator() == no_constant)
  610. {
  611. if (queryUnqualifiedType(rl->queryType()) == queryUnqualifiedType(rr->queryType()))
  612. {
  613. __int64 delta = rr->queryValue()->getIntValue();
  614. if (op == no_sub)
  615. delta = -delta;
  616. if (subtract)
  617. delta = -delta;
  618. OwnedHqlExpr newLeft = adjustValue(left, delta);
  619. return adjustBoundIntegerValues(newLeft, rl, subtract);
  620. }
  621. }
  622. break;
  623. }
  624. }
  625. }
  626. switch (left->getOperator())
  627. {
  628. case no_constant:
  629. if (!subtract)
  630. return adjustBoundIntegerValues(right, left, false);
  631. break;
  632. case no_add:
  633. {
  634. IHqlExpression * lr = left->queryChild(1);
  635. if (lr->getOperator() == no_constant)
  636. {
  637. OwnedHqlExpr newLeft = adjustBoundIntegerValues(left->queryChild(0), right, subtract);
  638. return adjustBoundIntegerValues(newLeft, lr, false);
  639. }
  640. break;
  641. }
  642. case no_variable:
  643. if (!subtract && (right->getOperator() == no_add) && (right->queryChild(1)->getOperator() == no_constant))
  644. {
  645. OwnedHqlExpr temp = adjustBoundIntegerValues(left, right->queryChild(0), false);
  646. return adjustBoundIntegerValues(temp, right->queryChild(1), false);
  647. }
  648. break;
  649. }
  650. return createValue(subtract ? no_sub : no_add, left->getType(), LINK(left), LINK(right));
  651. }
  652. IHqlExpression * multiplyValue(IHqlExpression * expr, unsigned __int64 value)
  653. {
  654. if (isZero(expr))
  655. return LINK(expr);
  656. ITypeInfo * type = expr->queryType();
  657. IValue * exprValue = expr->queryValue();
  658. if (exprValue && type->isInteger())
  659. return createConstant(type->castFrom(false, exprValue->getIntValue() * value));
  660. if (expr->getOperator() == no_translated)
  661. {
  662. IHqlExpression * translated = expr->queryChild(0);
  663. if (translated->queryValue())
  664. {
  665. OwnedHqlExpr newValue = multiplyValue(translated, value);
  666. return createTranslated(newValue);
  667. }
  668. }
  669. return createValue(no_mul, LINK(type), LINK(expr), createConstant(type->castFrom(false, value)));
  670. }
  671. bool matchesConstValue(IHqlExpression * expr, __int64 matchValue)
  672. {
  673. IValue * value = expr->queryValue();
  674. if (value)
  675. return value->getIntValue() == matchValue;
  676. if (expr->getOperator() == no_translated)
  677. return matchesConstValue(expr->queryChild(0), matchValue);
  678. return false;
  679. }
  680. IHqlExpression * createTranslated(IHqlExpression * expr)
  681. {
  682. return createValue(no_translated, expr->getType(), LINK(expr));
  683. }
  684. IHqlExpression * createTranslatedOwned(IHqlExpression * expr)
  685. {
  686. return createValue(no_translated, expr->getType(), expr);
  687. }
  688. IHqlExpression * createTranslated(IHqlExpression * expr, IHqlExpression * length)
  689. {
  690. ITypeInfo * type = expr->queryType();
  691. switch (type->getTypeCode())
  692. {
  693. case type_dictionary:
  694. return createDictionary(no_translated, LINK(expr), LINK(length));
  695. case type_table:
  696. case type_groupedtable:
  697. return createDataset(no_translated, LINK(expr), LINK(length));
  698. }
  699. return createValue(no_translated, expr->getType(), LINK(expr), LINK(length));
  700. }
  701. static IHqlExpression * querySimplifyCompareArgCast(IHqlExpression * expr)
  702. {
  703. if (expr->isConstant())
  704. return expr;
  705. while ((expr->getOperator() == no_implicitcast) || (expr->getOperator() == no_cast))
  706. {
  707. ITypeInfo * type = expr->queryType()->queryPromotedType();
  708. switch (type->getTypeCode())
  709. {
  710. case type_string:
  711. case type_data:
  712. case type_unicode:
  713. case type_qstring:
  714. case type_utf8:
  715. break;
  716. default:
  717. return expr;
  718. }
  719. IHqlExpression * child = expr->queryChild(0);
  720. ITypeInfo * childType = child->queryType()->queryPromotedType();
  721. if (type->getStringLen() < childType->getStringLen())
  722. break;
  723. type_t tc = type->getTypeCode();
  724. if (tc != childType->getTypeCode())
  725. {
  726. if (tc == type_string)
  727. {
  728. if (childType->getTypeCode() != type_varstring)
  729. break;
  730. if (type->queryCharset() != childType->queryCharset())
  731. break;
  732. }
  733. else if (tc == type_unicode)
  734. {
  735. if (childType->getTypeCode() != type_varunicode)
  736. break;
  737. if (type->queryLocale() != childType->queryLocale())
  738. break;
  739. }
  740. else
  741. break;
  742. }
  743. else
  744. {
  745. Owned<ITypeInfo> stretched = getStretchedType(type->getStringLen(), childType);
  746. if (stretched != type)
  747. break;
  748. }
  749. expr = child;
  750. }
  751. return expr;
  752. }
  753. IHqlExpression * getSimplifyCompareArg(IHqlExpression * expr)
  754. {
  755. IHqlExpression * cast = querySimplifyCompareArgCast(expr);
  756. if (cast->getOperator() != no_substring)
  757. return LINK(cast);
  758. if (cast->queryChild(0)->queryType()->getTypeCode() == type_qstring)
  759. return LINK(cast);
  760. HqlExprArray args;
  761. unwindChildren(args, cast);
  762. args.append(*createAttribute(quickAtom));
  763. return cast->clone(args);
  764. }
  765. bool isNullAssign(const CHqlBoundTarget & target, IHqlExpression * expr)
  766. {
  767. ITypeInfo * targetType = target.expr->queryType();
  768. //if an assignment to a local variable size temporary object, then it is ok to omit an assignment of null
  769. //since it won't change its value, and it isn't going to be assigned more than once.
  770. if ((targetType->getSize() == UNKNOWN_LENGTH) && target.length && hasWrapperModifier(targetType) && !hasModifier(targetType, typemod_member))
  771. {
  772. ITypeInfo * exprType = expr->queryType();
  773. switch (exprType->getTypeCode())
  774. {
  775. case type_data:
  776. case type_string:
  777. case type_qstring:
  778. return exprType->getSize() == 0;
  779. case type_dictionary:
  780. case type_table:
  781. return expr->getOperator() == no_null;
  782. }
  783. }
  784. return false;
  785. }
  786. ExpressionFormat queryNaturalFormat(ITypeInfo * type)
  787. {
  788. if (hasStreamedModifier(type))
  789. return FormatStreamedDataset;
  790. if (hasOutOfLineModifier(type))
  791. return FormatArrayDataset;
  792. if (hasLinkCountedModifier(type))
  793. return FormatLinkedDataset;
  794. return FormatBlockedDataset;
  795. }
  796. //===========================================================================
  797. SubGraphInfo::SubGraphInfo(IPropertyTree * _tree, unsigned _id, unsigned _graphId, IHqlExpression * _graphTag, SubGraphType _type)
  798. : HqlExprAssociation(subGraphMarker), tree(_tree)
  799. {
  800. id = _id;
  801. graphId = _graphId;
  802. type = _type;
  803. graphTag.set(_graphTag);
  804. }
  805. //===========================================================================
  806. IHqlExpression * CHqlBoundExpr::getTranslatedExpr() const
  807. {
  808. HqlExprArray args;
  809. args.append(*LINK(expr));
  810. if (length) args.append(*LINK(length));
  811. if (count) args.append(*createAttribute(countAtom, LINK(count)));
  812. if (isAll) args.append(*createAttribute(allAtom, LINK(isAll)));
  813. ITypeInfo * type = expr->queryType();
  814. switch (type->getTypeCode())
  815. {
  816. case type_dictionary:
  817. return createDictionary(no_translated, args);
  818. case type_table:
  819. case type_groupedtable:
  820. return createDataset(no_translated, args);
  821. }
  822. return createValue(no_translated, LINK(type), args);
  823. }
  824. IHqlExpression * CHqlBoundExpr::getComplexExpr() const
  825. {
  826. assertex(count && expr);
  827. return createValue(no_complex, expr->getType(), LINK(count), LINK(expr));
  828. }
  829. IHqlExpression * CHqlBoundExpr::getIsAll() const
  830. {
  831. if (isAll)
  832. return LINK(isAll);
  833. return LINK(queryBoolExpr(false));
  834. }
  835. void CHqlBoundExpr::setFromTarget(const CHqlBoundTarget & target)
  836. {
  837. isAll.set(target.isAll);
  838. count.set(target.count);
  839. length.set(target.length);
  840. expr.setown(convertWrapperToPointer(target.expr));
  841. }
  842. void CHqlBoundExpr::setFromTranslated(IHqlExpression * translatedExpr)
  843. {
  844. expr.set(translatedExpr->queryChild(0));
  845. IHqlExpression * arg = translatedExpr->queryChild(1);
  846. if (arg)
  847. {
  848. unsigned i = 2;
  849. if (arg->getOperator() != no_attr)
  850. {
  851. length.set(arg);
  852. arg = translatedExpr->queryChild(i++);
  853. }
  854. while (arg)
  855. {
  856. IAtom * name = arg->queryName();
  857. if (name == countAtom)
  858. count.set(arg->queryChild(0));
  859. else if (name == allAtom)
  860. isAll.set(arg->queryChild(0));
  861. else
  862. UNIMPLEMENTED;
  863. arg = translatedExpr->queryChild(i++);
  864. }
  865. }
  866. }
  867. bool CHqlBoundTarget::extractFrom(const CHqlBoundExpr & bound)
  868. {
  869. ITypeInfo * boundType = bound.queryType();
  870. if (bound.count)
  871. {
  872. if (bound.count->getOperator() != no_variable)
  873. return false;
  874. if (!hasLinkCountedModifier(boundType))
  875. return false;
  876. }
  877. if (bound.isAll)
  878. {
  879. assertex(bound.isAll->getOperator() == no_variable);
  880. }
  881. else if (boundType->getTypeCode() == type_set)
  882. return false;
  883. if (bound.length)
  884. {
  885. if (bound.length->getOperator() != no_variable)
  886. return false;
  887. }
  888. else if (boundType->getSize() == UNKNOWN_LENGTH)
  889. {
  890. type_t btc = boundType->getTypeCode();
  891. if ((btc != type_varstring) && (btc != type_varunicode) && !hasLinkCountedModifier(boundType))
  892. return false;
  893. }
  894. IHqlExpression * boundExpr = bound.expr;
  895. if (boundExpr->getOperator() == no_implicitcast)
  896. {
  897. IHqlExpression * uncast = boundExpr->queryChild(0);
  898. if (hasModifier(uncast->queryType(), typemod_member) &&
  899. (queryUnqualifiedType(boundExpr->queryType()) == queryUnqualifiedType(uncast->queryType())))
  900. boundExpr = uncast;
  901. }
  902. expr.set(boundExpr);
  903. isAll.set(bound.isAll);
  904. length.set(bound.length);
  905. count.set(bound.count);
  906. return true;
  907. }
  908. bool CHqlBoundTarget::isFixedSize() const
  909. {
  910. validate();
  911. return queryType()->getSize() != UNKNOWN_LENGTH;
  912. }
  913. void CHqlBoundTarget::validate() const
  914. {
  915. if (expr)
  916. {
  917. ITypeInfo * type = queryType();
  918. type_t tc = type->getTypeCode();
  919. if (tc == type_row || type->isReference())
  920. {
  921. //No checks to apply in these cases.
  922. }
  923. else if (type->getSize() != UNKNOWN_LENGTH)
  924. {
  925. assertex(!length);
  926. }
  927. else if (isArrayRowset(type))
  928. {
  929. if (!hasStreamedModifier(type))
  930. assertex(count != NULL);
  931. }
  932. else
  933. {
  934. assertex(length || tc == type_varstring || tc == type_varunicode || hasStreamedModifier(queryType()));
  935. }
  936. }
  937. }
  938. ITypeInfo * CHqlBoundTarget::queryType() const
  939. {
  940. return expr->queryType();
  941. }
  942. IHqlExpression * CHqlBoundTarget::getTranslatedExpr() const
  943. {
  944. CHqlBoundExpr temp;
  945. temp.setFromTarget(*this);
  946. return temp.getTranslatedExpr();
  947. }
  948. //===========================================================================
  949. CompoundBuilder::CompoundBuilder(node_operator _op)
  950. {
  951. op = _op;
  952. }
  953. void CompoundBuilder::addOperand(IHqlExpression * arg)
  954. {
  955. if (first.get())
  956. {
  957. compound.setown(createOpenValue(op, makeBoolType()));
  958. compound->addOperand(first.getClear());
  959. }
  960. if (compound)
  961. compound->addOperand(arg);
  962. else
  963. first.setown(arg);
  964. }
  965. IHqlExpression * CompoundBuilder::getCompound()
  966. {
  967. if (compound)
  968. return compound.getClear()->closeExpr();
  969. return first.getClear();
  970. }
  971. //===========================================================================
  972. void buildClearPointer(BuildCtx & ctx, IHqlExpression * expr, CompilerType compiler)
  973. {
  974. StringBuffer s;
  975. generateExprCpp(s, expr, compiler).append("=NULL;");
  976. ctx.addQuoted(s);
  977. }
  978. void insertUniqueString(StringAttrArray & array, const char * text)
  979. {
  980. ForEachItemIn(idx, array)
  981. {
  982. StringAttrItem & cur = array.item(idx);
  983. if (stricmp(cur.text, text) == 0)
  984. return;
  985. }
  986. array.append(* new StringAttrItem(text));
  987. }
  988. HqlCppInstance::HqlCppInstance(IWorkUnit *_wu, const char * _wupathname)
  989. {
  990. workunit.set(_wu);
  991. wupathname.set(_wupathname);
  992. }
  993. HqlStmts * HqlCppInstance::ensureSection(IAtom * section)
  994. {
  995. HqlStmts * match = querySection(section);
  996. if (match)
  997. return match;
  998. HqlCppSection * cur = new HqlCppSection;
  999. cur->section = section;
  1000. sections.append(*cur);
  1001. return &cur->stmts;
  1002. }
  1003. void HqlCppInstance::processIncludes()
  1004. {
  1005. BuildCtx ctx(*this, includeAtom);
  1006. StringBuffer s;
  1007. ForEachItemIn(idx, includes)
  1008. {
  1009. s.clear().append("#include \"").append(includes.item(idx).text).append("\"");
  1010. ctx.addQuoted(s);
  1011. }
  1012. }
  1013. const char * HqlCppInstance::queryLibrary(unsigned idx)
  1014. {
  1015. if (modules.isItem(idx))
  1016. return modules.item(idx).text;
  1017. return NULL;
  1018. }
  1019. const char * HqlCppInstance::queryObjectFile(unsigned idx)
  1020. {
  1021. if (objectFiles.isItem(idx))
  1022. return objectFiles.item(idx).text;
  1023. return NULL;
  1024. }
  1025. const char * HqlCppInstance::querySourceFile(unsigned idx)
  1026. {
  1027. if (sourceFiles.isItem(idx))
  1028. return sourceFiles.item(idx).text;
  1029. return NULL;
  1030. }
  1031. HqlStmts * HqlCppInstance::querySection(IAtom * section)
  1032. {
  1033. ForEachItemIn(idx, sections)
  1034. {
  1035. HqlCppSection & cur = (HqlCppSection &)sections.item(idx);
  1036. if (cur.section == section)
  1037. return &cur.stmts;
  1038. }
  1039. return NULL;
  1040. }
  1041. void HqlCppInstance::addPlugin(const char *plugin, const char *version)
  1042. {
  1043. if (!plugin || !*plugin)
  1044. return;
  1045. StringBuffer dllname(plugin);
  1046. getFileNameOnly(dllname, false); // MORE - shouldn't really need to do this here....
  1047. if (workunit)
  1048. {
  1049. Owned<IWUPlugin> p = workunit->updatePluginByName(dllname.str());
  1050. if (version && *version)
  1051. p->setPluginVersion(version);
  1052. }
  1053. if (!plugins)
  1054. plugins.setown(createPTree("Plugins"));
  1055. StringBuffer xpath;
  1056. xpath.append("Plugin[@dll='").append(dllname).append("']");
  1057. if (!plugins->hasProp(xpath.str()))
  1058. {
  1059. IPropertyTree * pluginNode = createPTree("Plugin");
  1060. pluginNode->setProp("@dll", dllname.str());
  1061. if (version && *version)
  1062. pluginNode->setProp("@version", version);
  1063. plugins->addPropTree("Plugin", pluginNode);
  1064. }
  1065. }
  1066. void HqlCppInstance::addPluginsAsResource()
  1067. {
  1068. if (!plugins)
  1069. return;
  1070. StringBuffer pluginXML;
  1071. toXML(plugins, pluginXML);
  1072. addResource("PLUGINS", pluginXML.length(), pluginXML.str(), NULL, 1);
  1073. }
  1074. void HqlCppInstance::getActivityRange(unsigned cppIndex, unsigned & minActivityId, unsigned & maxActivityId)
  1075. {
  1076. if (cppInfo.isItem(cppIndex))
  1077. {
  1078. minActivityId = cppInfo.item(cppIndex).minActivityId;
  1079. maxActivityId = cppInfo.item(cppIndex).maxActivityId;
  1080. }
  1081. else
  1082. {
  1083. minActivityId = 0;
  1084. maxActivityId = 0;
  1085. }
  1086. }
  1087. bool HqlCppInstance::useFunction(IHqlExpression * func)
  1088. {
  1089. assertex(func);
  1090. func = func->queryBody();
  1091. if (helpers.contains(*func))
  1092. return false;
  1093. helpers.append(*LINK(func));
  1094. IHqlExpression * funcDef = func->queryChild(0);
  1095. StringBuffer libname, init, include, source;
  1096. getAttribute(funcDef, libraryAtom, libname);
  1097. getAttribute(funcDef, initfunctionAtom, init);
  1098. getAttribute(funcDef, includeAtom, include);
  1099. getAttribute(funcDef, sourceAtom, source);
  1100. if (init.length())
  1101. {
  1102. BuildCtx ctx(*this, initAtom);
  1103. ctx.addQuoted(init.append("(wuid);"));
  1104. }
  1105. IHqlExpression *pluginAttr = funcDef->queryAttribute(pluginAtom);
  1106. if (pluginAttr)
  1107. {
  1108. StringBuffer plugin, version;
  1109. getStringValue(plugin, pluginAttr->queryChild(0));
  1110. getStringValue(version, pluginAttr->queryChild(1));
  1111. addPlugin(plugin.str(), version.str());
  1112. if (!libname.length())
  1113. {
  1114. getStringValue(libname, pluginAttr->queryChild(0));
  1115. getFullFileName(libname, true);
  1116. }
  1117. }
  1118. if (!funcDef->hasAttribute(ctxmethodAtom) && !funcDef->hasAttribute(gctxmethodAtom) && !funcDef->hasAttribute(methodAtom))
  1119. {
  1120. if (libname.length())
  1121. useLibrary(libname.str());
  1122. }
  1123. if (include.length())
  1124. useInclude(include.str());
  1125. if (source.length())
  1126. useSourceFile(source);
  1127. return true;
  1128. }
  1129. void HqlCppInstance::useInclude(const char * include)
  1130. {
  1131. insertUniqueString(includes, include);
  1132. }
  1133. void HqlCppInstance::useLibrary(const char * libname)
  1134. {
  1135. insertUniqueString(modules, libname);
  1136. }
  1137. void HqlCppInstance::useObjectFile(const char * objname)
  1138. {
  1139. insertUniqueString(objectFiles, objname);
  1140. }
  1141. void HqlCppInstance::useSourceFile(const char * srcname)
  1142. {
  1143. insertUniqueString(sourceFiles, srcname);
  1144. }
  1145. void HqlCppInstance::addHint(const char * hintXml, ICodegenContextCallback * ctxCallback)
  1146. {
  1147. if (!hintFile)
  1148. {
  1149. StringBuffer hintFilename;
  1150. if (wupathname)
  1151. hintFilename.append(wupathname);
  1152. else
  1153. hintFilename.append("wu");
  1154. hintFilename.append("_hints.xml");
  1155. Owned<IFile> file = createIFile(hintFilename);
  1156. Owned<IFileIO> io = file->open(IFOcreate);
  1157. if (!io)
  1158. return;
  1159. hintFile.setown(createIOStream(io));
  1160. appendHintText("<Hints>\n");
  1161. Owned<IWUQuery> query = workunit->updateQuery();
  1162. associateLocalFile(query, FileTypeCpp, hintFilename.str(), "Hints", 0);
  1163. }
  1164. appendHintText(hintXml);
  1165. }
  1166. void HqlCppInstance::appendHintText(const char * xml)
  1167. {
  1168. hintFile->write(strlen(xml), xml);
  1169. }
  1170. unsigned HqlCppInstance::addStringResource(unsigned len, const char * body)
  1171. {
  1172. return resources.addString(len, body);
  1173. }
  1174. void HqlCppInstance::addResource(const char * type, unsigned len, const void * body, IPropertyTree *manifestEntry, unsigned id)
  1175. {
  1176. resources.addNamed(type, len, body, manifestEntry, id);
  1177. }
  1178. void HqlCppInstance::addCompressResource(const char * type, unsigned len, const void * body, IPropertyTree *manifestEntry, unsigned id)
  1179. {
  1180. #ifdef ADD_RESOURCE_AS_CPP_COMMENT
  1181. BuildCtx ctx(*this, includeAtom);
  1182. StringBuffer s;
  1183. s.append("/* ").append(type).append(".").append(id).append(":\n").append(len,(const char *)data).newline().append("*/");
  1184. ctx.addQuoted(s);
  1185. #endif
  1186. resources.addCompress(type, len, body, manifestEntry, id);
  1187. }
  1188. void HqlCppInstance::flushHints()
  1189. {
  1190. if (hintFile)
  1191. {
  1192. appendHintText("</Hints>\n");
  1193. hintFile.clear();
  1194. }
  1195. }
  1196. void HqlCppInstance::flushResources(const char *filename, ICodegenContextCallback * ctxCallback)
  1197. {
  1198. addPluginsAsResource();
  1199. if (resources.count())
  1200. {
  1201. bool flushText = workunit->getDebugValueBool("flushResourceAsText", false);
  1202. #ifdef __64BIT__
  1203. bool target64bit = workunit->getDebugValueBool("target64bit", true);
  1204. #else
  1205. bool target64bit = workunit->getDebugValueBool("target64bit", false);
  1206. #endif
  1207. StringBuffer resname;
  1208. bool isObjectFile = resources.flush(resname, filename, flushText, target64bit);
  1209. StringBuffer resTextName;
  1210. if (flushText && resources.queryWriteText(resTextName, resname))
  1211. {
  1212. Owned<IWUQuery> query = workunit->updateQuery();
  1213. associateLocalFile(query, FileTypeHintXml, resTextName, "Workunit resource text", 0);
  1214. }
  1215. if (isObjectFile)
  1216. useObjectFile(resname);
  1217. else
  1218. useSourceFile(resname);
  1219. }
  1220. }
  1221. IHqlCppInstance * createCppInstance(IWorkUnit *wu, const char * wupathname)
  1222. {
  1223. return new HqlCppInstance(wu, wupathname);
  1224. }
  1225. //===========================================================================
  1226. #include "hqlcppsys.ecl"
  1227. HqlCppTranslator::HqlCppTranslator(IErrorReceiver * _errors, const char * _soName, IHqlCppInstance * _code, ClusterType _targetClusterType, ICodegenContextCallback *_ctxCallback) : ctxCallback(_ctxCallback)
  1228. {
  1229. //Insert a couple of warning mapping layers - one for global #onwarnings, and another for local : onwarning
  1230. globalOnWarnings.setown(new ErrorSeverityMapper(*_errors));
  1231. localOnWarnings.setown(new ErrorSeverityMapper((IErrorReceiver &)*globalOnWarnings)); // horrible: cast required, otherwise copy constructor is called!
  1232. //Ensure that any errors reported within the code generator automatically abort compiling immediately
  1233. errorProcessor.setown(createAbortingErrorReceiver(*localOnWarnings));
  1234. targetClusterType = _targetClusterType;
  1235. {
  1236. CriticalBlock block(*systemCS);
  1237. if (!cppSystemScope)
  1238. {
  1239. StringBuffer systemText;
  1240. unsigned size = 0;
  1241. for (unsigned i1=0; cppSystemText[i1]; i1++)
  1242. size += strlen(cppSystemText[i1]) + 2;
  1243. systemText.ensureCapacity(size);
  1244. for (unsigned i2=0; cppSystemText[i2]; i2++)
  1245. systemText.append(cppSystemText[i2]).newline();
  1246. MultiErrorReceiver errs;
  1247. HqlDummyLookupContext ctx(&errs);
  1248. cppSystemScope = createScope();
  1249. Owned<ISourcePath> sysPath = createSourcePath("<system-definitions>");
  1250. Owned<IFileContents> systemContents = createFileContentsFromText(systemText.str(), sysPath, true, NULL, 0);
  1251. OwnedHqlExpr query = parseQuery(cppSystemScope, systemContents, ctx, NULL, NULL, false, false);
  1252. if (errs.errCount())
  1253. {
  1254. StringBuffer errtext;
  1255. IError *first = errs.firstError();
  1256. first->toString(errtext);
  1257. throw MakeStringException(HQLERR_FailedToLoadSystemModule, "%s @ %d:%d", errtext.str(), first->getColumn(), first->getLine());
  1258. }
  1259. #if 0
  1260. else if (errs.warnCount())
  1261. {
  1262. StringBuffer s;
  1263. errs.toString(s);
  1264. PrintLog("Parsing system scope: ");
  1265. PrintLog(s.str());
  1266. }
  1267. #endif
  1268. }
  1269. }
  1270. litno = 0;
  1271. soName.set(_soName);
  1272. HqlDummyLookupContext dummyctx(NULL);
  1273. OwnedHqlExpr internalScopeLookup = cppSystemScope->lookupSymbol(createIdAtom("InternalCppService"), LSFsharedOK, dummyctx);
  1274. internalScope = internalScopeLookup->queryScope();
  1275. _clear(options); // init options is called later, but depends on the workunit.
  1276. startCursorSet = 0;
  1277. requireTable = true;
  1278. activeGraphCtx = NULL;
  1279. maxSequence = 0;
  1280. contextAvailable = true;
  1281. graphSeqNumber = 0;
  1282. nlpParse = NULL;
  1283. outputLibrary = NULL;
  1284. activitiesThisCpp = 0;
  1285. curCppFile = 0;
  1286. timeReporter.setown(createStdTimeReporter());
  1287. curActivityId = 0;
  1288. holeUniqueSequence = 0;
  1289. nextUid = 0;
  1290. nextTypeId = 0;
  1291. nextFieldId = 0;
  1292. curWfid = 0;
  1293. code = (HqlCppInstance*)_code;
  1294. xmlUsesContents = false;
  1295. }
  1296. HqlCppTranslator::~HqlCppTranslator()
  1297. {
  1298. ::Release(nlpParse);
  1299. ::Release(outputLibrary);
  1300. }
  1301. void HqlCppTranslator::setTargetClusterType(ClusterType clusterType)
  1302. {
  1303. targetClusterType = clusterType;
  1304. }
  1305. void HqlCppTranslator::ensureDiskAccessAllowed(IHqlExpression * expr)
  1306. {
  1307. bool isSigned = expr->hasAttribute(_signed_Atom);
  1308. if (!queryCallback()->allowAccess("datafile", isSigned))
  1309. reportErrorNoAbort(expr, HQLERR_DatafileRequiresSigned, "You do not have permission to directly access datafiles");
  1310. }
  1311. void HqlCppTranslator::checkAbort()
  1312. {
  1313. if (wu() && wu()->aborting())
  1314. throw MakeStringException(HQLERR_ErrorAlreadyReported, "Aborting");
  1315. }
  1316. // Option: (Name, value, ?overridden, default())
  1317. // problems:
  1318. // default value can depend on another option (e.g., cluster type/supports lcr).
  1319. // don't want code in multiple places - e.g., the values initialized and defaulted, and dependencies calculations duplicated separately.
  1320. // don't want lots of start up costs each time the translator is created -> lightweight classes if any.
  1321. // don't really want two structures, one for the definitions, and another for the values.
  1322. //RESOLVED? want to walk the debug options provided, instead of checking for each possibility in turn.
  1323. // Without this restriction it becomes much easier.
  1324. void HqlCppTranslator::cacheOptions()
  1325. {
  1326. SCMStringBuffer targetText;
  1327. wu()->getDebugValue("targetClusterType", targetText);
  1328. ClusterType clusterType = getClusterType(targetText.s.str());
  1329. if (clusterType != NoCluster)
  1330. setTargetClusterType(clusterType);
  1331. //Some compound flags, which provide defaults for various other options.
  1332. bool paranoid = getDebugFlag("paranoid", false);
  1333. bool releaseMode = getDebugFlag("release", true);
  1334. struct DebugOption
  1335. {
  1336. typedef enum { typeByte, typeUnsigned, typeSigned, typeBool } OptionType;
  1337. DebugOption (bool & _option, const char * name, bool defaultValue) : option(&_option), optName(name)
  1338. {
  1339. _option = defaultValue;
  1340. type = typeBool;
  1341. }
  1342. DebugOption (byte & _option, const char * name, byte defaultValue) : option(&_option), optName(name)
  1343. {
  1344. _option = defaultValue;
  1345. type = typeByte;
  1346. }
  1347. DebugOption (unsigned & _option, const char * name, unsigned defaultValue) : option(&_option), optName(name)
  1348. {
  1349. _option = defaultValue;
  1350. type = typeUnsigned;
  1351. }
  1352. DebugOption (int & _option, const char * name, unsigned defaultValue) : option(&_option), optName(name)
  1353. {
  1354. _option = defaultValue;
  1355. type = typeSigned;
  1356. }
  1357. void setValue(const char * val)
  1358. {
  1359. switch (type)
  1360. {
  1361. case typeBool:
  1362. {
  1363. bool * b = (bool*)option;
  1364. *b = strToBool(val);
  1365. break;
  1366. }
  1367. case typeUnsigned:
  1368. {
  1369. unsigned * u = (unsigned*)option;
  1370. *u = (unsigned)atoi(val);
  1371. break;
  1372. }
  1373. case typeSigned:
  1374. {
  1375. signed * u = (signed*)option;
  1376. *u = (signed)atoi(val);
  1377. break;
  1378. }
  1379. case typeByte:
  1380. {
  1381. byte * b = (byte*)option;
  1382. *b = (byte)atoi(val);
  1383. break;
  1384. }
  1385. }
  1386. }
  1387. void * option;
  1388. const char * optName;
  1389. OptionType type;
  1390. };
  1391. //Note this list cannot have any initial values which are dependent on other options.
  1392. DebugOption debugOptions[] =
  1393. {
  1394. DebugOption(options.peephole,"peephole", true),
  1395. DebugOption(options.foldConstantCast,"foldConstantCast", true),
  1396. DebugOption(options.optimizeBoolReturn,"optimizeBoolReturn", true),
  1397. DebugOption(options.freezePersists,"freezePersists", false),
  1398. DebugOption(options.maxRecordSize, "defaultMaxLengthRecord", MAX_RECORD_SIZE),
  1399. DebugOption(options.subgraphToRegenerate, "subgraphToRegenerate", 0),
  1400. DebugOption(options.checkRoxieRestrictions,"checkRoxieRestrictions", true), // a debug aid for running regression suite
  1401. DebugOption(options.checkThorRestrictions,"checkThorRestrictions", true), // a debug aid for running regression suite
  1402. DebugOption(options.allowCsvWorkunitRead,"allowStoredCsvFormat", false),
  1403. DebugOption(options.evaluateCoLocalRowInvariantInExtract,"evaluateCoLocalRowInvariantInExtract", false),
  1404. DebugOption(options.spanMultipleCpp,"spanMultipleCpp", true),
  1405. DebugOption(options.activitiesPerCpp, "<exception>", 0x7fffffff),
  1406. DebugOption(options.allowInlineSpill,"allowInlineSpill", true),
  1407. DebugOption(options.optimizeGlobalProjects,"optimizeGlobalProjects", false),
  1408. DebugOption(options.optimizeResourcedProjects,"optimizeResourcedProjects", false),
  1409. DebugOption(options.reduceNetworkTraffic,"aggressiveOptimizeProjects", false),
  1410. DebugOption(options.notifyOptimizedProjects, "notifyOptimizedProjects", 0),
  1411. DebugOption(options.optimizeProjectsPreservePersists,"optimizeProjectsPreservePersists", false),
  1412. DebugOption(options.expirePersists, "expirePersists", true),
  1413. DebugOption(options.defaultPersistExpiry, "defaultPersistExpiry", DEFAULT_PERSIST_EXPIRY_PERIOD),
  1414. DebugOption(options.defaultExpiry, "defaultExpiry", DEFAULT_EXPIRY_PERIOD),
  1415. DebugOption(options.searchDistanceThreshold, "searchDistanceThreshold", 1000000),
  1416. DebugOption(options.generateActivityThreshold, "generateActivityThreshold", 0), // most users are not interested so disable by default
  1417. DebugOption(options.checkAsserts,"checkAsserts", true),
  1418. DebugOption(options.assertSortedDistributed,"assertSortedDistributed", false),
  1419. DebugOption(options.optimizeLoopInvariant,"optimizeLoopInvariant", false), // doesn't fully work yet! and has little effect, and messes up the alias dependencies
  1420. DebugOption(options.defaultImplicitKeyedJoinLimit, "defaultImplicitKeyedJoinLimit", 10000),
  1421. DebugOption(options.defaultImplicitIndexReadLimit, "defaultImplicitIndexReadLimit", 0),
  1422. DebugOption(options.commonUpChildGraphs,"commonUpChildGraphs", true),
  1423. DebugOption(options.detectAmbiguousSelector,"detectAmbiguousSelector", false),
  1424. DebugOption(options.allowAmbiguousSelector,"allowAmbiguousSelector", false),
  1425. #ifdef _DEBUG
  1426. DebugOption(options.regressionTest,"regressionTest", true),
  1427. #else
  1428. DebugOption(options.regressionTest,"regressionTest", false),
  1429. #endif
  1430. //recreating case can cause duplicate branches in weird situations.
  1431. DebugOption(options.recreateMapFromIf,"recreateMapFromIf", !targetThor()),
  1432. DebugOption(options.showMetaText,"debugShowMetaText", false),
  1433. DebugOption(options.resourceSequential,"resourceSequential", false),
  1434. DebugOption(options.workunitTemporaries,"workunitTemporaries", true),
  1435. DebugOption(options.resourceConditionalActions,"resourceConditionalActions", false), //targetRoxie() ??
  1436. DebugOption(options.actionLinkInNewGraph,"actionLinkInNewGraph", false),
  1437. DebugOption(options.minimizeWorkunitTemporaries, "<exception>", false),
  1438. DebugOption(options.pickBestEngine,"pickBestEngine", true),
  1439. DebugOption(options.groupedChildIterators,"groupedChildIterators", false),
  1440. DebugOption(options.noAllToLookupConversion,"noAllToLookupConversion", false),
  1441. DebugOption(options.notifyWorkflowCse,"notifyWorkflowCse", true),
  1442. DebugOption(options.performWorkflowCse,"performWorkflowCse", false),
  1443. DebugOption(options.warnOnImplicitJoinLimit,"warnOnImplicitJoinLimit", targetRoxie()),
  1444. DebugOption(options.warnOnImplicitReadLimit,"warnOnImplicitReadLimit", targetRoxie()),
  1445. DebugOption(options.convertJoinToLookup,"convertJoinToLookup", true),
  1446. DebugOption(options.convertJoinToLookupIfSorted,"convertJoinToLookupIfSorted", false),
  1447. DebugOption(options.spotCSE,"spotCSE", true),
  1448. DebugOption(options.spotCseInIfDatasetConditions,"spotCseInIfDatasetConditions", true),
  1449. DebugOption(options.optimizeNonEmpty,"optimizeNonEmpty", !targetThor()), // not sure that it will be conditional resourced correctly for thor
  1450. DebugOption(options.allowVariableRoxieFilenames,"allowVariableRoxieFilenames", false),
  1451. DebugOption(options.foldConstantDatasets,"foldConstantDatasets", true),
  1452. DebugOption(options.hoistSimpleGlobal,"hoistSimpleGlobal", true),
  1453. DebugOption(options.percolateConstants,"percolateConstants", true),
  1454. DebugOption(options.percolateFilters,"percolateFilters", false),
  1455. DebugOption(options.usePrefetchForAllProjects,"usePrefetchForAllProjects", false),
  1456. DebugOption(options.allFilenamesDynamic,"allFilenamesDynamic", false),
  1457. DebugOption(options.optimizeSteppingPostfilter,"optimizeSteppingPostfilter", true),
  1458. DebugOption(options.moveUnconditionalActions,"moveUnconditionalActions", false),
  1459. DebugOption(options.paranoidCheckNormalized, "paranoidCheckNormalized", paranoid),
  1460. DebugOption(options.paranoidCheckDependencies, "paranoidCheckDependencies", paranoid),
  1461. DebugOption(options.paranoidCheckSelects, "paranoidCheckSelects", paranoid),
  1462. DebugOption(options.preventKeyedSplit,"preventKeyedSplit", true),
  1463. DebugOption(options.preventSteppedSplit,"preventSteppedSplit", true),
  1464. DebugOption(options.canGenerateSimpleAction,"canGenerateSimpleAction", true),
  1465. DebugOption(options.minimizeActivityClasses,"minimizeActivityClasses", true),
  1466. DebugOption(options.maxRootMaybeThorActions, "maxRootMaybeThorActions", 0),
  1467. DebugOption(options.minimizeSkewBeforeSpill,"minimizeSkewBeforeSpill", false),
  1468. DebugOption(options.createSerializeForUnknownSize,"createSerializeForUnknownSize", false),
  1469. DebugOption(options.implicitLinkedChildRows,"implicitLinkedChildRows", false),
  1470. DebugOption(options.mainRowsAreLinkCounted,"mainRowsAreLinkCounted", true),
  1471. DebugOption(options.allowSections,"allowSections", true),
  1472. DebugOption(options.autoPackRecords,"autoPackRecords", false),
  1473. DebugOption(options.commonUniqueNameAttributes,"commonUniqueNameAttributes", true),
  1474. DebugOption(options.sortIndexPayload,"sortIndexPayload", true),
  1475. DebugOption(options.foldFilter,"foldFilter", true),
  1476. DebugOption(options.finalizeAllRows, "finalizeAllRows", false),
  1477. DebugOption(options.maxLocalRowSize , "maxLocalRowSize", MAX_LOCAL_ROW_SIZE),
  1478. DebugOption(options.optimizeGraph,"optimizeGraph", true),
  1479. DebugOption(options.orderDiskFunnel,"orderDiskFunnel", false),
  1480. DebugOption(options.alwaysAllowAllNodes,"alwaysAllowAllNodes", false),
  1481. DebugOption(options.slidingJoins,"slidingJoins", false),
  1482. DebugOption(options.foldOptimized,"foldOptimized", false),
  1483. DebugOption(options.globalOptimize,"globalOptimize", false),
  1484. DebugOption(options.applyInstantEclTransformations,"applyInstantEclTransformations", false), // testing option
  1485. DebugOption(options.calculateComplexity,"calculateComplexity", false),
  1486. DebugOption(options.generateLogicalGraph,"generateLogicalGraph", false),
  1487. DebugOption(options.generateLogicalGraphOnly,"generateLogicalGraphOnly", false),
  1488. DebugOption(options.globalAutoHoist,"globalAutoHoist", true),
  1489. DebugOption(options.applyInstantEclTransformationsLimit, "applyInstantEclTransformationsLimit", 100),
  1490. DebugOption(options.insertProjectCostLevel, "insertProjectCostLevel", (unsigned)-1),
  1491. DebugOption(options.dfaRepeatMax, "dfaRepeatMax", 10),
  1492. DebugOption(options.dfaRepeatMaxScore, "dfaRepeatMaxScore", 100),
  1493. DebugOption(options.debugNlp, "debugNlp", DEFAULT_NLP_DETAIL),
  1494. DebugOption(options.regexVersion, "regexVersion",0),
  1495. DebugOption(options.parseDfaComplexity, "parseDfaComplexity", (unsigned)-1),
  1496. DebugOption(options.expandRepeatAnyAsDfa,"expandRepeatAnyAsDfa", true),
  1497. DebugOption(options.resourceMaxMemory, "resourceMaxMemory", 0),
  1498. DebugOption(options.resourceMaxSockets, "resourceMaxSockets", 0),
  1499. DebugOption(options.resourceMaxActivities, "resourceMaxActivities", 0),
  1500. DebugOption(options.resourceMaxHeavy, "resourceMaxHeavy", 1),
  1501. DebugOption(options.resourceMaxDistribute, "resourceMaxDistribute", 8),
  1502. DebugOption(options.unlimitedResources,"unlimitedResources", false),
  1503. DebugOption(options.filteredReadSpillThreshold, "filteredReadSpillThreshold", 999),
  1504. DebugOption(options.allowThroughSpill,"allowThroughSpill", true),
  1505. DebugOption(options.minimiseSpills,"minimiseSpills", false),
  1506. DebugOption(options.spillMultiCondition,"spillMultiCondition", false),
  1507. DebugOption(options.spotThroughAggregate,"spotThroughAggregate", true),
  1508. DebugOption(options.hoistResourced,"hoistResourced", true),
  1509. DebugOption(options.minimizeSpillSize, "minimizeSpillSize", 0),
  1510. DebugOption(options.maximizeLexer,"maximizeLexer", false),
  1511. DebugOption(options.foldStored,"foldStored", false),
  1512. DebugOption(options.spotTopN,"spotTopN", true),
  1513. DebugOption(options.topnLimit, "topnLimit", 10000),
  1514. DebugOption(options.groupAllDistribute,"groupAllDistribute", true),
  1515. DebugOption(options.spotLocalMerge,"spotLocalMerge", true),
  1516. DebugOption(options.spotPotentialKeyedJoins,"spotPotentialKeyedJoins", false),
  1517. DebugOption(options.combineTrivialStored,"combineTrivialStored", true),
  1518. DebugOption(options.combineAllStored,"combineAllStored", false),
  1519. DebugOption(options.allowStoredDuplicate,"allowStoredDuplicate", false), // only here as a temporary workaround
  1520. DebugOption(options.specifiedClusterSize, "clusterSize", 0),
  1521. DebugOption(options.globalFoldOptions, "globalFoldOptions", (unsigned)-1),
  1522. DebugOption(options.allowScopeMigrate,"allowScopeMigrate", true),
  1523. DebugOption(options.supportFilterProject,"supportFilterProject", true),
  1524. DebugOption(options.normalizeExplicitCasts,"normalizeExplicitCasts", true),
  1525. DebugOption(options.optimizeInlineSource,"optimizeInlineSource", false),
  1526. DebugOption(options.optimizeDiskSource,"optimizeDiskSource", true),
  1527. DebugOption(options.optimizeIndexSource,"optimizeIndexSource", true),
  1528. DebugOption(options.optimizeChildSource,"optimizeChildSource", false),
  1529. DebugOption(options.reportLocations,"reportLocations", true),
  1530. DebugOption(options.debugGeneratedCpp,"debugGeneratedCpp", false),
  1531. DebugOption(options.addFilesnamesToGraph,"addFilesnamesToGraph", true),
  1532. DebugOption(options.normalizeLocations,"normalizeLocations", true),
  1533. DebugOption(options.ensureRecordsHaveSymbols,"ensureRecordsHaveSymbols", true),
  1534. DebugOption(options.constantFoldNormalize,"constantFoldNormalize", true),
  1535. DebugOption(options.constantFoldPostNormalize,"constantFoldPostNormalize", false),
  1536. DebugOption(options.optimizeGrouping,"optimizeGrouping", true),
  1537. DebugOption(options.showMetaInGraph,"showMetaInGraph", false),
  1538. DebugOption(options.spotComplexClasses,"spotComplexClasses", true),
  1539. DebugOption(options.complexClassesThreshold,"complexClassesThreshold", 5000),
  1540. DebugOption(options.complexClassesActivityFilter,"complexClassesActivityFilter", 0),
  1541. DebugOption(options.optimizeString1Compare,"optimizeString1Compare", true),
  1542. DebugOption(options.optimizeSpillProject,"optimizeSpillProject", true),
  1543. DebugOption(options.expressionPeephole,"expressionPeephole", false),
  1544. DebugOption(options.optimizeIncrement,"optimizeIncrement", true),
  1545. DebugOption(options.supportsMergeDistribute,"supportsMergeDistribute", true),
  1546. DebugOption(options.debugNlpAsHint,"debugNlpAsHint", false),
  1547. DebugOption(options.forceVariableWuid,"forceVariableWuid", false),
  1548. DebugOption(options.okToDeclareAndAssign,"okToDeclareAndAssign", false),
  1549. DebugOption(options.noteRecordSizeInGraph,"noteRecordSizeInGraph", true),
  1550. DebugOption(options.convertRealAssignToMemcpy,"convertRealAssignToMemcpy", false),
  1551. DebugOption(options.allowActivityForKeyedJoin,"allowActivityForKeyedJoin", false),
  1552. DebugOption(options.forceActivityForKeyedJoin,"forceActivityForKeyedJoin", false),
  1553. DebugOption(options.addLibraryInputsToGraph,"addLibraryInputsToGraph", false),
  1554. DebugOption(options.showRecordCountInGraph,"showRecordCountInGraph", true),
  1555. DebugOption(options.serializeRowsetInExtract,"serializeRowsetInExtract", false),
  1556. DebugOption(options.testIgnoreMaxLength,"testIgnoreMaxLength", false),
  1557. DebugOption(options.trackDuplicateActivities,"trackDuplicateActivities", false),
  1558. DebugOption(options.showActivitySizeInGraph,"showActivitySizeInGraph", false),
  1559. DebugOption(options.addLocationToCpp,"addLocationToCpp", false),
  1560. DebugOption(options.alwaysCreateRowBuilder,"alwaysCreateRowBuilder", false),
  1561. DebugOption(options.precalculateFieldOffsets,"precalculateFieldOffsets", false),
  1562. DebugOption(options.generateStaticInlineTables,"generateStaticInlineTables", true),
  1563. DebugOption(options.staticRowsUseStringInitializer,"staticRowsUseStringInitializer", true),
  1564. DebugOption(options.convertWhenExecutedToCompound,"convertWhenExecutedToCompound", queryLegacyWhenSemantics()),
  1565. DebugOption(options.standAloneExe,"standAloneExe", false),
  1566. DebugOption(options.enableCompoundCsvRead,"enableCompoundCsvRead", true),
  1567. // The following works 99% of the time, but disabled due to potential problems with the ambiguity of LEFT
  1568. //possibly causing filters on nested records to be incorrectly removed.
  1569. DebugOption(options.optimizeNestedConditional,"optimizeNestedConditional", false),
  1570. DebugOption(options.createImplicitAliases,"createImplicitAliases", false),
  1571. DebugOption(options.combineSiblingGraphs,"combineSiblingGraphs", true),
  1572. DebugOption(options.optimizeSharedGraphInputs,"optimizeSharedGraphInputs", true),
  1573. DebugOption(options.supportsSubSortActivity,"supportsSubSortActivity",false),
  1574. DebugOption(options.implicitSubSort,"implicitSubSort",true),
  1575. DebugOption(options.implicitBuildIndexSubSort,"implicitBuildIndexSubSort",true),
  1576. DebugOption(options.implicitJoinSubSort,"implicitJoinSubSort",true),
  1577. DebugOption(options.implicitGroupSubSort,"implicitGroupSubSort",true),
  1578. DebugOption(options.implicitGroupHashAggregate,"implicitGroupHashAggregate",false),
  1579. DebugOption(options.implicitGroupHashDedup,"implicitGroupHashDedup",false),
  1580. DebugOption(options.reportFieldUsage,"reportFieldUsage",false),
  1581. DebugOption(options.reportFileUsage,"reportFileUsage",false),
  1582. DebugOption(options.recordFieldUsage,"recordFieldUsage",false),
  1583. DebugOption(options.subsortLocalJoinConditions,"subsortLocalJoinConditions",false),
  1584. DebugOption(options.projectNestedTables,"projectNestedTables",true),
  1585. DebugOption(options.showSeqInGraph,"showSeqInGraph",false), // For tracking down why projects are not commoned up
  1586. DebugOption(options.normalizeSelectorSequence,"normalizeSelectorSequence",false), // For tracking down why projects are not commoned up
  1587. DebugOption(options.removeXpathFromOutput,"removeXpathFromOutput",false),
  1588. DebugOption(options.canLinkConstantRows,"canLinkConstantRows",true),
  1589. DebugOption(options.checkAmbiguousRollupCondition,"checkAmbiguousRollupCondition",true),
  1590. DebugOption(options.matchExistingDistributionForJoin,"matchExistingDistributionForJoin",true),
  1591. DebugOption(options.createImplicitKeyedDistributeForJoin,"createImplicitKeyedDistributeForJoin",false),
  1592. DebugOption(options.expandHashJoin,"expandHashJoin",true),
  1593. DebugOption(options.traceIR,"traceIR",false),
  1594. DebugOption(options.preserveCaseExternalParameter,"preserveCaseExternalParameter",true),
  1595. DebugOption(options.optimizeParentAccess,"optimizeParentAccess",false),
  1596. DebugOption(options.expandPersistInputDependencies,"expandPersistInputDependencies",true),
  1597. DebugOption(options.multiplePersistInstances,"multiplePersistInstances",true),
  1598. DebugOption(options.defaultNumPersistInstances,"defaultNumPersistInstances",-1),
  1599. DebugOption(options.optimizeMax,"optimizeMax",false),
  1600. DebugOption(options.useResultsForChildSpills,"useResultsForChildSpills",false),
  1601. DebugOption(options.alwaysUseGraphResults,"alwaysUseGraphResults",false),
  1602. DebugOption(options.noConditionalLinks,"noConditionalLinks",false),
  1603. DebugOption(options.reportAssertFilenameTail,"reportAssertFilenameTail",false),
  1604. DebugOption(options.newBalancedSpotter,"newBalancedSpotter",true),
  1605. DebugOption(options.keyedJoinPreservesOrder,"keyedJoinPreservesOrder",true),
  1606. DebugOption(options.expandSelectCreateRow,"expandSelectCreateRow",false),
  1607. DebugOption(options.obfuscateOutput,"obfuscateOutput",false),
  1608. DebugOption(options.showEclInGraph,"showEclInGraph",true),
  1609. DebugOption(options.showChildCountInGraph,"showChildCountInGraph",false),
  1610. DebugOption(options.optimizeSortAllFields,"optimizeSortAllFields",true),
  1611. DebugOption(options.optimizeSortAllFieldsStrict,"optimizeSortAllFieldsStrict",false),
  1612. DebugOption(options.alwaysReuseGlobalSpills,"alwaysReuseGlobalSpills",true),
  1613. DebugOption(options.forceAllDatasetsParallel,"forceAllDatasetsParallel",false), // Purely for regression testing.
  1614. DebugOption(options.embeddedWarningsAsErrors,"embeddedWarningsAreFatal",true),
  1615. DebugOption(options.optimizeCriticalFunctions,"optimizeCriticalFunctions",true),
  1616. DebugOption(options.addLikelihoodToGraph,"addLikelihoodToGraph", true),
  1617. DebugOption(options.varFieldAccessorThreshold,"varFieldAccessorThreshold",3), // Generate accessor classes for rows with #variable width fields >= threshold
  1618. DebugOption(options.translateDFSlayouts,"translateDFSlayouts", false),
  1619. DebugOption(options.timeTransforms,"timeTransforms", false),
  1620. DebugOption(options.reportDFSinfo,"reportDFSinfo", 0),
  1621. DebugOption(options.useGlobalCompareClass,"useGlobalCompareClass", false),
  1622. DebugOption(options.createValueSets,"createValueSets", true),
  1623. DebugOption(options.implicitKeyedDiskFilter,"implicitKeyedDiskFilter", false),
  1624. DebugOption(options.addDefaultBloom,"addDefaultBloom", true),
  1625. DebugOption(options.newDiskReadMapping, "newDiskReadMapping", true),
  1626. DebugOption(options.transformNestedSequential, "transformNestedSequential", true),
  1627. DebugOption(options.forceAllProjectedDiskSerialized, "internalForceAllProjectedDiskSerialized", false), // Delete in 8.0 once new code has been proved in anger
  1628. };
  1629. //get options values from workunit
  1630. const unsigned numDebugOptions = _elements_in(debugOptions);
  1631. Owned<IStringIterator> debugs(&wu()->getDebugValues());
  1632. SCMStringBuffer name, val;
  1633. ForEach(*debugs)
  1634. {
  1635. debugs->str(name);
  1636. wu()->getDebugValue(name.str(),val);
  1637. unsigned x = 0;
  1638. for (; x < numDebugOptions; x++)
  1639. {
  1640. if (0 == stricmp(name.str(), debugOptions[x].optName))
  1641. {
  1642. debugOptions[x].setValue(val.str());
  1643. break;
  1644. }
  1645. }
  1646. }
  1647. //Configure the divide by zero action
  1648. options.divideByZeroAction = DBZzero;
  1649. const char * dbz = wu()->getDebugValue("divideByZero",val).str();
  1650. if (strieq(dbz, "0") || strieq(dbz, "zero"))
  1651. options.divideByZeroAction = DBZzero;
  1652. else if (strieq(dbz, "nan"))
  1653. options.divideByZeroAction = DBZnan;
  1654. else if (strieq(dbz, "fail") || strieq(dbz, "throw"))
  1655. options.divideByZeroAction = DBZfail;
  1656. else if (val.length())
  1657. throwError2(HQLERR_UnexpectedOptionValue_XY, "divideByZero", dbz);
  1658. //The following cases handle options whose default values are dependent on other options.
  1659. //Or where one debug options sets more than one option
  1660. if (options.spanMultipleCpp)
  1661. {
  1662. code->cppInfo.append(* new CppFileInfo(0)); // Add an entry for the main file which contains no activities
  1663. options.activitiesPerCpp = wu()->getDebugValueInt("activitiesPerCpp", DEFAULT_ACTIVITIES_PER_CPP);
  1664. curCppFile = 1;
  1665. }
  1666. code->cppInfo.append(* new CppFileInfo(0));
  1667. options.targetCompiler = DEFAULT_COMPILER;
  1668. if (wu()->hasDebugValue("targetGcc"))
  1669. options.targetCompiler = wu()->getDebugValueBool("targetGcc", false) ? GccCppCompiler : Vs6CppCompiler;
  1670. SCMStringBuffer compilerText;
  1671. wu()->getDebugValue("targetCompiler", compilerText);
  1672. for (CompilerType iComp = (CompilerType)0; iComp < MaxCompiler; iComp = (CompilerType)(iComp+1))
  1673. {
  1674. if (stricmp(compilerText.s.str(), compilerTypeText[iComp]) == 0)
  1675. options.targetCompiler = iComp;
  1676. }
  1677. if (getDebugFlag("optimizeProjects", true))
  1678. {
  1679. options.optimizeGlobalProjects = true;
  1680. options.optimizeResourcedProjects = true;
  1681. }
  1682. options.minimizeWorkunitTemporaries = !options.workunitTemporaries || getDebugFlag("minimizeWorkunitTemporaries", false);//options.resourceConditionalActions);
  1683. options.inlineStringThreshold = wu()->getDebugValueInt("inlineStringThreshold", (options.targetCompiler != Vs6CppCompiler) ? 0 : 10000);
  1684. options.implicitLinkedChildRows = true;
  1685. options.finalizeAllRows = true; // inline temporary rows should actually be ok.
  1686. postProcessOptions();
  1687. }
  1688. void HqlCppTranslator::postProcessOptions()
  1689. {
  1690. if (options.optimizeMax)
  1691. {
  1692. //Enable any extra potentially expensive optimizations options here...
  1693. options.foldConstantDatasets = true;
  1694. options.percolateConstants = true;
  1695. options.percolateFilters = true;
  1696. }
  1697. //Any post processing - e.g., dependent flags goes here...
  1698. options.optimizeDiskFlag = 0;
  1699. if (options.optimizeInlineSource)
  1700. options.optimizeDiskFlag |= CSFnewinline;
  1701. if (options.optimizeDiskSource)
  1702. options.optimizeDiskFlag |= CSFnewdisk;
  1703. if (options.optimizeIndexSource)
  1704. options.optimizeDiskFlag |= CSFnewindex;
  1705. if (options.optimizeChildSource)
  1706. options.optimizeDiskFlag |= CSFnewchild;
  1707. if (!targetThor())
  1708. {
  1709. //Roxie doesn't gain from additional projects, hthor doesn't support split
  1710. options.optimizeSpillProject = false;
  1711. }
  1712. if (options.resourceSequential)
  1713. options.resourceConditionalActions = true;
  1714. options.generateActivityThresholdCycles = nanosec_to_cycle(options.generateActivityThreshold * I64C(1000000));
  1715. //Probably best to ignore this warning. - possibly configure it based on some other option
  1716. globalOnWarnings->addOnWarning(HQLWRN_FoldRemoveKeyed, ignoreAtom);
  1717. //Ensure the settings for the following options are always present in the workunit
  1718. wu()->setDebugValueInt("expandPersistInputDependencies",options.expandPersistInputDependencies,true);
  1719. if (options.forceVariableWuid)
  1720. wu()->setCloneable(true);
  1721. }
  1722. unsigned HqlCppTranslator::getOptimizeFlags(bool insideChildQuery) const
  1723. {
  1724. unsigned optFlags = HOOfold;
  1725. switch (targetClusterType)
  1726. {
  1727. case RoxieCluster:
  1728. optFlags |= HOOnoclonelimit|HOOalwayslocal;
  1729. break;
  1730. case HThorCluster:
  1731. optFlags |= HOOnocloneindexlimit|HOOalwayslocal;
  1732. break;
  1733. case ThorLCRCluster:
  1734. break;
  1735. default:
  1736. UNIMPLEMENTED;
  1737. }
  1738. if ((options.optimizeDiskFlag & CSFnewdisk) && (options.optimizeDiskFlag & CSFnewindex))
  1739. optFlags |= HOOhascompoundaggregate;
  1740. if (options.foldConstantDatasets)
  1741. optFlags |= HOOfoldconstantdatasets;
  1742. if (options.optimizeMax)
  1743. optFlags |= HOOexpensive;
  1744. if (options.expandSelectCreateRow)
  1745. optFlags |= HOOexpandselectcreaterow;
  1746. //The following flag should possibly be dependent on this condition:
  1747. //if (targetThor() && !insideChildQuery)
  1748. //However it may be better to err on the side of caution - see HPCC-10144 since moving a project over a barrier may
  1749. //cause it in some circumstances to be evaluated many more times. See HPCC-17439 to revisit later.
  1750. optFlags |= HOOminimizeNetworkAndMemory;
  1751. return optFlags;
  1752. }
  1753. void HqlCppTranslator::exportWarningMappings()
  1754. {
  1755. globalOnWarnings->exportMappings(wu());
  1756. }
  1757. void HqlCppTranslator::overrideOptionsForLibrary()
  1758. {
  1759. options.workunitTemporaries = false;
  1760. options.pickBestEngine = false;
  1761. options.minimizeWorkunitTemporaries = true;
  1762. }
  1763. void HqlCppTranslator::overrideOptionsForQuery()
  1764. {
  1765. options.workunitTemporaries = getDebugFlag("workunitTemporaries", true);
  1766. options.pickBestEngine = getDebugFlag("pickBestEngine", true);
  1767. options.minimizeWorkunitTemporaries = !options.workunitTemporaries || getDebugFlag("minimizeWorkunitTemporaries", false);//options.resourceConditionalActions);
  1768. }
  1769. bool HqlCppTranslator::needToSerializeToSlave(IHqlExpression * expr) const
  1770. {
  1771. if (targetThor())
  1772. return true;
  1773. switch (expr->getOperator())
  1774. {
  1775. case no_getresult:
  1776. case no_workunit_dataset:
  1777. return !matchesConstantValue(queryAttributeChild(expr, sequenceAtom, 0), ResultSequenceOnce);
  1778. default:
  1779. return true;
  1780. }
  1781. }
  1782. IHqlExpression *HqlCppTranslator::addBigLiteral(const char *lit, unsigned litLen)
  1783. {
  1784. unsigned resid = code->addStringResource(litLen, lit);
  1785. HqlExprArray args;
  1786. args.append(*getSizetConstant(resid));
  1787. return bindTranslatedFunctionCall(loadResourceId, args);
  1788. }
  1789. IHqlExpression * HqlCppTranslator::addLiteral(const char * text)
  1790. {
  1791. return createConstant(text);
  1792. }
  1793. IHqlExpression *HqlCppTranslator::addDataLiteral(const char *lit, unsigned litLen)
  1794. {
  1795. if (!canGenerateStringInline(litLen))
  1796. return addBigLiteral(lit, litLen);
  1797. else
  1798. return createConstant(createStringValue(lit, litLen));
  1799. }
  1800. IHqlExpression *HqlCppTranslator::addStringLiteral(const char *lit)
  1801. {
  1802. unsigned litLen = strlen(lit);
  1803. if (!canGenerateStringInline(litLen))
  1804. return addBigLiteral(lit, litLen+1);
  1805. else
  1806. return createConstant(createStringValue(lit, litLen));
  1807. }
  1808. IHqlExpression * HqlCppTranslator::bindFunctionCall(IIdAtom * name, HqlExprArray & args)
  1809. {
  1810. OwnedHqlExpr function = needFunction(name);
  1811. assertex(function != NULL);
  1812. return bindFunctionCall(function, args);
  1813. }
  1814. IHqlExpression * HqlCppTranslator::bindFunctionCall(IIdAtom * name, IHqlExpression * arg1)
  1815. {
  1816. HqlExprArray args;
  1817. args.append(*arg1);
  1818. return bindFunctionCall(name, args);
  1819. }
  1820. IHqlExpression * HqlCppTranslator::bindFunctionCall(IHqlExpression * function, HqlExprArray & args)
  1821. {
  1822. useFunction(function);
  1823. IHqlExpression * ret = createBoundFunction(NULL, function, args, NULL, false);
  1824. assertex(ret->queryExternalDefinition());
  1825. args.kill();
  1826. return ret;
  1827. }
  1828. IHqlExpression * HqlCppTranslator::bindFunctionCall(IIdAtom * name, HqlExprArray & args, ITypeInfo * newType)
  1829. {
  1830. if (!newType)
  1831. return bindFunctionCall(name, args);
  1832. OwnedHqlExpr function = needFunction(name);
  1833. useFunction(function);
  1834. assertex(function->getOperator() == no_funcdef);
  1835. IFunctionTypeExtra * funcTypeExtra = queryFunctionTypeExtra(function->queryType());
  1836. assertex(funcTypeExtra);
  1837. IHqlExpression * body = function->queryChild(0);
  1838. HqlExprArray bodyArgs;
  1839. unwindChildren(bodyArgs, body);
  1840. HqlExprArray funcArgs;
  1841. funcArgs.append(*createValue(body->getOperator(), LINK(newType), bodyArgs));
  1842. unwindChildren(funcArgs, function, 1);
  1843. ITypeInfo * funcType = makeFunctionType(LINK(newType), LINK(function->queryChild(1)), LINK(function->queryChild(2)), LINK(funcTypeExtra->queryAttributes()));
  1844. OwnedHqlExpr newFunction = createValue(function->getOperator(), funcType, funcArgs);
  1845. return bindFunctionCall(newFunction, args);
  1846. }
  1847. IHqlExpression * HqlCppTranslator::bindTranslatedFunctionCall(IHqlExpression * function, HqlExprArray & args)
  1848. {
  1849. useFunction(function);
  1850. IHqlExpression * ret = createTranslatedExternalCall(NULL, function, args);
  1851. assertex(ret->queryExternalDefinition());
  1852. args.kill();
  1853. return ret;
  1854. }
  1855. IHqlExpression * HqlCppTranslator::bindTranslatedFunctionCall(IIdAtom * name, HqlExprArray & args)
  1856. {
  1857. OwnedHqlExpr function = needFunction(name);
  1858. return bindTranslatedFunctionCall(function, args);
  1859. }
  1860. void HqlCppTranslator::buildTranslatedFunctionCall(BuildCtx & ctx, IIdAtom * name, HqlExprArray & args)
  1861. {
  1862. OwnedHqlExpr call = bindTranslatedFunctionCall(name, args);
  1863. ctx.addExpr(call);
  1864. }
  1865. void HqlCppTranslator::buildFunctionCall(BuildCtx & ctx, IIdAtom * name, HqlExprArray & args)
  1866. {
  1867. OwnedHqlExpr call = bindFunctionCall(name, args);
  1868. buildStmt(ctx, call);
  1869. }
  1870. /* Args: all elements in it are LINKED */
  1871. void HqlCppTranslator::callProcedure(BuildCtx & ctx, IIdAtom * name, HqlExprArray & args)
  1872. {
  1873. OwnedHqlExpr call = bindTranslatedFunctionCall(name, args);
  1874. assertex(call->queryExternalDefinition());
  1875. ctx.addExpr(call);
  1876. }
  1877. bool HqlCppTranslator::getDebugFlag(const char * name, bool defValue)
  1878. {
  1879. return wu()->getDebugValueBool(name, defValue);
  1880. }
  1881. void HqlCppTranslator::doReportWarning(WarnErrorCategory category, ErrorSeverity explicitSeverity, IHqlExpression * location, unsigned id, const char * msg)
  1882. {
  1883. Owned<IError> warnError;
  1884. if (!location)
  1885. location = queryActiveActivityLocation();
  1886. unsigned activity = 0;
  1887. const char * scope = nullptr;
  1888. StringBuffer scopeText;
  1889. if (activeActivities.ordinality())
  1890. {
  1891. ABoundActivity & top = activeActivities.tos();
  1892. activity = top.queryActivityId();
  1893. ActivityInstance * active = top.queryActive();
  1894. dbgassertex(active);
  1895. if (active)
  1896. {
  1897. active->getScope(scopeText);
  1898. scope = scopeText;
  1899. }
  1900. }
  1901. ErrorSeverity severity = (explicitSeverity == SeverityUnknown) ? queryDefaultSeverity(category) : explicitSeverity;
  1902. if (location)
  1903. warnError.setown(createError(category, severity, id, msg, str(location->querySourcePath()), location->getStartLine(), location->getStartColumn(), 0, activity, scope));
  1904. else
  1905. warnError.setown(createError(category, severity, id, msg, activity, scope));
  1906. errorProcessor->report(warnError);
  1907. }
  1908. void HqlCppTranslator::reportWarning(WarnErrorCategory category, ErrorSeverity explicitSeverity, IHqlExpression * location, unsigned id, const char * msg, ...)
  1909. {
  1910. StringBuffer s;
  1911. va_list args;
  1912. va_start(args, msg);
  1913. s.valist_appendf(msg, args);
  1914. va_end(args);
  1915. doReportWarning(category, explicitSeverity, location, id, s.str());
  1916. }
  1917. void HqlCppTranslator::reportWarning(WarnErrorCategory category, unsigned id, const char * msg, ...)
  1918. {
  1919. StringBuffer s;
  1920. va_list args;
  1921. va_start(args, msg);
  1922. s.valist_appendf(msg, args);
  1923. va_end(args);
  1924. doReportWarning(category, SeverityUnknown, NULL, id, s.str());
  1925. }
  1926. void HqlCppTranslator::addWorkunitException(ErrorSeverity severity, unsigned code, const char * text, IHqlExpression * location)
  1927. {
  1928. Owned<IWUException> msg = wu()->createException();
  1929. msg->setExceptionSource("eclcc");
  1930. if (code)
  1931. msg->setExceptionCode(code);
  1932. msg->setExceptionMessage(text);
  1933. msg->setSeverity(severity);
  1934. msg->setTimeStamp(NULL);
  1935. if (!location)
  1936. location = queryActiveActivityLocation();
  1937. if (location)
  1938. {
  1939. msg->setExceptionFileName(str(location->querySourcePath()));
  1940. msg->setExceptionLineNo(location->getStartLine());
  1941. msg->setExceptionColumn(location->getStartColumn());
  1942. }
  1943. }
  1944. IHqlExpression * HqlCppTranslator::queryActiveNamedActivity()
  1945. {
  1946. ForEachItemInRev(i, activityExprStack)
  1947. {
  1948. IHqlExpression & cur = activityExprStack.item(i);
  1949. IHqlExpression * symbol = queryNamedSymbol(&cur);
  1950. if (symbol && symbol->querySourcePath())
  1951. return symbol;
  1952. if (isCompoundSource(&cur))
  1953. {
  1954. IHqlExpression * child = cur.queryChild(0);
  1955. if (hasNamedSymbol(child))
  1956. return child;
  1957. }
  1958. }
  1959. return NULL;
  1960. }
  1961. IHqlExpression * HqlCppTranslator::queryActiveActivityLocation() const
  1962. {
  1963. ForEachItemInRev(i, activityExprStack)
  1964. {
  1965. IHqlExpression & cur = activityExprStack.item(i);
  1966. IHqlExpression * location = queryLocation(&cur);
  1967. if (location)
  1968. return location;
  1969. if (isCompoundSource(&cur))
  1970. {
  1971. location = queryLocation(cur.queryChild(0));
  1972. if (location)
  1973. return location;
  1974. }
  1975. }
  1976. return NULL;
  1977. }
  1978. void HqlCppTranslator::report(IError* error)
  1979. {
  1980. return errorProcessor->report(error);
  1981. }
  1982. IError * HqlCppTranslator::mapError(IError * error)
  1983. {
  1984. return errorProcessor->mapError(error);
  1985. }
  1986. size32_t HqlCppTranslator::errCount()
  1987. {
  1988. return errorProcessor->errCount();
  1989. }
  1990. size32_t HqlCppTranslator::warnCount()
  1991. {
  1992. return errorProcessor->warnCount();
  1993. }
  1994. void HqlCppTranslator::exportMappings(IWorkUnit * wu) const
  1995. {
  1996. errorProcessor->exportMappings(wu);
  1997. }
  1998. void HqlCppTranslator::ThrowStringException(int code,const char *format, ...) const
  1999. {
  2000. IHqlExpression * location = queryActiveActivityLocation();
  2001. if (errorProcessor && location)
  2002. {
  2003. StringBuffer errorMsg;
  2004. va_list args;
  2005. va_start(args, format);
  2006. errorMsg.valist_appendf(format, args);
  2007. va_end(args);
  2008. throw createError(code, errorMsg.str(), str(location->querySourcePath()), location->getStartLine(), location->getStartColumn(), 0);
  2009. }
  2010. va_list args;
  2011. va_start(args, format);
  2012. IException *ret = MakeStringExceptionVA(code, format, args);
  2013. va_end(args);
  2014. throw ret;
  2015. }
  2016. void HqlCppTranslator::reportErrorDirect(IHqlExpression * exprOrLocation, int code,const char *msg, bool alwaysAbort)
  2017. {
  2018. ECLlocation loc;
  2019. if (!loc.extractLocationAttr(exprOrLocation))
  2020. loc.extractLocationAttr(queryActiveActivityLocation());
  2021. const char * sourcePath = str(loc.sourcePath);
  2022. if (alwaysAbort)
  2023. throw createError(code, msg, sourcePath, loc.lineno, loc.column, loc.position);
  2024. errorProcessor->reportError(code, msg, sourcePath, loc.lineno, loc.column, loc.position);
  2025. }
  2026. void HqlCppTranslator::reportError(IHqlExpression * location, int code,const char *format, ...)
  2027. {
  2028. StringBuffer errorMsg;
  2029. va_list args;
  2030. va_start(args, format);
  2031. errorMsg.valist_appendf(format, args);
  2032. va_end(args);
  2033. reportErrorDirect(location, code, errorMsg.str(), true);
  2034. }
  2035. void HqlCppTranslator::reportErrorNoAbort(IHqlExpression * location, int code,const char *format, ...)
  2036. {
  2037. StringBuffer errorMsg;
  2038. va_list args;
  2039. va_start(args, format);
  2040. errorMsg.valist_appendf(format, args);
  2041. va_end(args);
  2042. reportErrorDirect(location, code, errorMsg.str(), false);
  2043. }
  2044. //---------------------------------------------------------------------------
  2045. void HqlCppTranslator::doBuildStmtAssign(BuildCtx & ctx, IHqlExpression * target, IHqlExpression * expr)
  2046. {
  2047. buildAssign(ctx, target, expr);
  2048. }
  2049. void HqlCppTranslator::buildAddress(BuildCtx & ctx, IHqlExpression * expr, CHqlBoundExpr & tgt)
  2050. {
  2051. node_operator op = expr->getOperator();
  2052. switch (op)
  2053. {
  2054. case no_address:
  2055. buildExpr(ctx, expr->queryChild(0), tgt);
  2056. break;
  2057. case no_typetransfer:
  2058. buildAddress(ctx, expr->queryChild(0), tgt);
  2059. break;
  2060. default:
  2061. {
  2062. Owned<IReferenceSelector> selector = buildReference(ctx, expr);
  2063. selector->buildAddress(ctx, tgt);
  2064. break;
  2065. }
  2066. }
  2067. }
  2068. bool HqlCppTranslator::hasAddress(BuildCtx & ctx, IHqlExpression * expr)
  2069. {
  2070. switch (expr->getOperator())
  2071. {
  2072. case no_deref:
  2073. case no_variable:
  2074. return true;
  2075. case no_field:
  2076. case no_select:
  2077. {
  2078. Owned<IReferenceSelector> selector = buildReference(ctx, expr);
  2079. return !selector->isConditional();
  2080. }
  2081. case no_typetransfer:
  2082. return hasAddress(ctx, expr->queryChild(0));
  2083. default:
  2084. return false;
  2085. }
  2086. }
  2087. void HqlCppTranslator::buildAssign(BuildCtx & ctx, IHqlExpression * target, IHqlExpression * expr)
  2088. {
  2089. #ifdef ADD_ASSIGNMENT_COMMENTS
  2090. if (target->getOperator() == no_select)
  2091. {
  2092. StringBuffer s;
  2093. ctx.addQuoted(s.append("//Assign to field ").append(target->queryChild(1)->queryName()));
  2094. }
  2095. #endif
  2096. Owned<IReferenceSelector> selector = buildReference(ctx, target);
  2097. if (expr->getOperator() == no_null)
  2098. selector->buildClear(ctx, 0);
  2099. else if (target->isDatarow() && (!hasReferenceModifier(target->queryType()) || !recordTypesMatch(target->queryType(), expr->queryType())))
  2100. buildRowAssign(ctx, selector, expr);
  2101. else
  2102. selector->set(ctx, expr);
  2103. }
  2104. void HqlCppTranslator::doBuildStmtAssignModify(BuildCtx & ctx, IHqlExpression * target, IHqlExpression * expr, node_operator assignOp)
  2105. {
  2106. Owned<IReferenceSelector> selector = buildReference(ctx, target);
  2107. selector->modifyOp(ctx, expr, assignOp);
  2108. }
  2109. void HqlCppTranslator::buildExprAssign(BuildCtx & ctx, const CHqlBoundTarget & target, IHqlExpression * expr)
  2110. {
  2111. /*
  2112. switch (target.queryType().getTypeCode())
  2113. {
  2114. case type_table:
  2115. case type_groupedtable:
  2116. buildDatasetAssign(ctx, target, expr);
  2117. break;
  2118. case type_row:
  2119. buildAssignRow(ctx, target, expr);
  2120. break;
  2121. }
  2122. */
  2123. node_operator op = expr->getOperator();
  2124. switch (op)
  2125. {
  2126. case no_constant:
  2127. if (!isNullAssign(target, expr))
  2128. doBuildExprAssign(ctx, target, expr);
  2129. else
  2130. ctx.addAssign(target.length, queryZero());
  2131. break;
  2132. case no_regex_find:
  2133. case no_regex_replace:
  2134. doBuildAssignRegexFindReplace(ctx, target, expr);
  2135. break;
  2136. case no_matched:
  2137. case no_matchtext:
  2138. case no_matchlength:
  2139. case no_matchposition:
  2140. case no_matchunicode:
  2141. case no_matchutf8:
  2142. doBuildMatched(ctx, &target, expr, NULL);
  2143. break;
  2144. case no_matchattr:
  2145. doBuildMatchAttr(ctx, &target, expr, NULL);
  2146. break;
  2147. case no_loopcounter:
  2148. doBuildAssignLoopCounter(ctx, target, expr);
  2149. break;
  2150. case no_evalonce:
  2151. doBuildEvalOnce(ctx, &target, expr, NULL);
  2152. break;
  2153. case no_alias_scope:
  2154. {
  2155. expandAliasScope(ctx, expr);
  2156. buildExprAssign(ctx, target, expr->queryChild(0));
  2157. break;
  2158. }
  2159. case no_case:
  2160. case no_map:
  2161. {
  2162. HqlCppCaseInfo info(*this);
  2163. doBuildCaseInfo(expr, info);
  2164. info.buildAssign(ctx, target);
  2165. break;
  2166. }
  2167. case no_which:
  2168. case no_rejected:
  2169. doBuildAssignWhich(ctx, target, expr);
  2170. break;
  2171. case no_call:
  2172. case no_externalcall:
  2173. doBuildAssignCall(ctx, target, expr);
  2174. break;
  2175. case no_cast:
  2176. case no_implicitcast:
  2177. doBuildAssignCast(ctx, target, expr);
  2178. break;
  2179. case no_choose:
  2180. doBuildAssignChoose(ctx, target, expr);
  2181. break;
  2182. case no_comma:
  2183. case no_compound:
  2184. buildStmt(ctx, expr->queryChild(0));
  2185. buildExprAssign(ctx, target, expr->queryChild(1));
  2186. break;
  2187. case no_executewhen:
  2188. doBuildAssignExecuteWhen(ctx, target, expr);
  2189. break;
  2190. case no_concat:
  2191. doBuildAssignConcat(ctx, target, expr);
  2192. break;
  2193. case no_div:
  2194. case no_modulus:
  2195. doBuildAssignDivide(ctx, target, expr);
  2196. break;
  2197. case no_crc:
  2198. case no_hash:
  2199. case no_hash32:
  2200. case no_hash64:
  2201. doBuildAssignHashCrc(ctx, target, expr);
  2202. break;
  2203. case no_hashmd5:
  2204. doBuildAssignHashMd5(ctx, target, expr);
  2205. break;
  2206. case no_if:
  2207. doBuildAssignIf(ctx, target, expr);
  2208. break;
  2209. case no_index:
  2210. doBuildAssignIndex(ctx, target, expr);
  2211. break;
  2212. case no_in:
  2213. case no_notin:
  2214. {
  2215. OwnedHqlExpr optimized = querySimplifyInExpr(expr);
  2216. if (optimized)
  2217. {
  2218. OwnedHqlExpr folded = foldHqlExpression(optimized);
  2219. buildExprAssign(ctx, target, folded);
  2220. }
  2221. else
  2222. doBuildAssignIn(ctx, target, expr);
  2223. break;
  2224. }
  2225. case no_intformat:
  2226. doBuildAssignFormat(intFormatId, ctx, target, expr);
  2227. break;
  2228. case no_nofold:
  2229. case no_nohoist:
  2230. case no_section:
  2231. case no_sectioninput:
  2232. case no_forcegraph:
  2233. case no_nocombine:
  2234. case no_likely:
  2235. case no_unlikely:
  2236. buildExprAssign(ctx, target, expr->queryChild(0));
  2237. break;
  2238. case no_realformat:
  2239. doBuildAssignFormat(realFormatId, ctx, target, expr);
  2240. break;
  2241. case no_order:
  2242. doBuildAssignOrder(ctx, target, expr);
  2243. break;
  2244. case no_unicodeorder:
  2245. doBuildAssignUnicodeOrder(ctx, target, expr);
  2246. break;
  2247. case no_substring:
  2248. doBuildAssignSubString(ctx, target, expr);
  2249. break;
  2250. case no_trim:
  2251. doBuildAssignTrim(ctx, target, expr);
  2252. break;
  2253. case no_field:
  2254. throwUnexpected();
  2255. case no_select:
  2256. {
  2257. OwnedHqlExpr aggregate = convertToSimpleAggregate(expr);
  2258. if (aggregate && canProcessInline(&ctx, aggregate->queryChild(0)))
  2259. {
  2260. buildExprAssign(ctx, target, aggregate);
  2261. return;
  2262. }
  2263. if (shouldEvaluateSelectAsAlias(ctx, expr) && !insideOnStart(ctx))
  2264. {
  2265. CHqlBoundExpr temp;
  2266. doBuildAliasValue(ctx, expr, temp, NULL);
  2267. assign(ctx, target, temp);
  2268. return;
  2269. }
  2270. Owned<IReferenceSelector> selector = buildReference(ctx, expr);
  2271. selector->assignTo(ctx, target);
  2272. return;
  2273. }
  2274. break;
  2275. case no_not:
  2276. {
  2277. IHqlExpression * child = expr->queryChild(0);
  2278. node_operator childOp = child->getOperator();
  2279. if (((childOp == no_and) || (childOp == no_or)) && requiresTempAfterFirst(ctx, child))
  2280. {
  2281. if (childOp == no_and)
  2282. doBuildAssignAnd(ctx, target, child, true);
  2283. else
  2284. {
  2285. OwnedHqlExpr inverted = convertOrToAnd(expr);
  2286. buildExprAssign(ctx, target, inverted);
  2287. }
  2288. }
  2289. else
  2290. doBuildExprAssign(ctx, target, expr);
  2291. break;
  2292. }
  2293. case no_or:
  2294. {
  2295. IHqlExpression * left = expr->queryChild(0);
  2296. //in always goes via an assign, so do this first, and then filter on result.
  2297. if (left->getOperator() == no_in)
  2298. {
  2299. BuildCtx subctx(ctx);
  2300. buildExprAssign(subctx, target, left);
  2301. OwnedHqlExpr inverse = getInverse(target.expr);
  2302. subctx.addFilter(inverse);
  2303. buildExprAssign(subctx, target, expr->queryChild(1));
  2304. }
  2305. else if (requiresTempAfterFirst(ctx, expr))
  2306. {
  2307. OwnedHqlExpr inverted = convertOrToAnd(expr);
  2308. buildExprAssign(ctx, target, inverted);
  2309. }
  2310. else
  2311. doBuildExprAssign(ctx, target, expr);
  2312. break;
  2313. }
  2314. case no_and:
  2315. if (requiresTempAfterFirst(ctx, expr))
  2316. doBuildAssignAnd(ctx, target, expr, false);
  2317. else
  2318. doBuildExprAssign(ctx, target, expr);
  2319. break;
  2320. case no_fromunicode:
  2321. case no_tounicode:
  2322. doBuildAssignToFromUnicode(ctx, target, expr);
  2323. break;
  2324. case no_toxml:
  2325. case no_tojson:
  2326. doBuildAssignToXmlorJson(ctx, target, expr);
  2327. break;
  2328. case no_wuid:
  2329. doBuildAssignWuid(ctx, target, expr);
  2330. break;
  2331. case no_xmldecode:
  2332. case no_xmlencode:
  2333. doBuildXmlEncode(ctx, &target, expr, NULL);
  2334. break;
  2335. case no_all:
  2336. doBuildAssignAll(ctx, target, expr);
  2337. return;
  2338. case no_list:
  2339. doBuildAssignList(ctx, target, expr);
  2340. return;
  2341. case no_addsets:
  2342. doBuildAssignAddSets(ctx, target, expr);
  2343. return;
  2344. case no_createset:
  2345. buildSetAssignViaBuilder(ctx, target, expr);
  2346. return;
  2347. case no_failmessage:
  2348. doBuildAssignFailMessage(ctx, target, expr);
  2349. return;
  2350. case no_eventname:
  2351. doBuildAssignEventName(ctx, target, expr);
  2352. return;
  2353. case no_eventextra:
  2354. doBuildAssignEventExtra(ctx, target, expr);
  2355. return;
  2356. case no_catch:
  2357. doBuildAssignCatch(ctx, target, expr);
  2358. break;
  2359. case no_id2blob:
  2360. doBuildAssignIdToBlob(ctx, target, expr);
  2361. break;
  2362. case no_getresult:
  2363. case no_workunit_dataset:
  2364. if (isSameFullyUnqualifiedType(expr->queryType(), target.queryType()))
  2365. doBuildAssignGetResult(ctx, target, expr);
  2366. else
  2367. doBuildExprAssign(ctx, target, expr);
  2368. break;
  2369. case no_getgraphresult:
  2370. doBuildAssignGetGraphResult(ctx, target, expr);
  2371. break;
  2372. case no_existslist:
  2373. doBuildAggregateList(ctx, &target, expr, NULL);
  2374. break;
  2375. case no_countlist:
  2376. doBuildAggregateList(ctx, &target, expr, NULL);
  2377. break;
  2378. case no_sumlist:
  2379. doBuildAggregateList(ctx, &target, expr, NULL);
  2380. break;
  2381. case no_minlist:
  2382. doBuildAggregateList(ctx, &target, expr, NULL);
  2383. break;
  2384. case no_maxlist:
  2385. doBuildAggregateList(ctx, &target, expr, NULL);
  2386. break;
  2387. case no_skip:
  2388. {
  2389. bool canReachFollowing = false;
  2390. doBuildStmtSkip(ctx, expr, &canReachFollowing);
  2391. if (canReachFollowing)
  2392. {
  2393. OwnedHqlExpr null = createNullExpr(expr);
  2394. doBuildExprAssign(ctx, target, null);
  2395. }
  2396. break;
  2397. }
  2398. case no_count:
  2399. case no_max:
  2400. case no_min:
  2401. case no_sum:
  2402. case no_exists:
  2403. doBuildAssignAggregate(ctx, target, expr);
  2404. break;
  2405. case no_getenv:
  2406. {
  2407. OwnedHqlExpr mapped = cvtGetEnvToCall(expr);
  2408. buildExprAssign(ctx, target, mapped);
  2409. break;
  2410. }
  2411. default:
  2412. doBuildExprAssign(ctx, target, expr);
  2413. break;
  2414. }
  2415. }
  2416. void HqlCppTranslator::buildExprAssignViaType(BuildCtx & ctx, const CHqlBoundTarget & target, IHqlExpression * expr, ITypeInfo * type)
  2417. {
  2418. OwnedHqlExpr temp = createValue(no_implicitcast, LINK(type), LINK(expr));
  2419. buildExprAssign(ctx, target, temp);
  2420. }
  2421. void HqlCppTranslator::buildExprAssignViaString(BuildCtx & ctx, const CHqlBoundTarget & target, IHqlExpression * expr, ITypeInfo *dest)
  2422. {
  2423. unsigned len;
  2424. switch (dest->getTypeCode())
  2425. {
  2426. case type_varstring:
  2427. case type_varunicode:
  2428. len = UNKNOWN_LENGTH;
  2429. break;
  2430. default:
  2431. len = dest->getStringLen();
  2432. break;
  2433. }
  2434. OwnedITypeInfo type = makeStringType(len, NULL, NULL);
  2435. buildExprAssignViaType(ctx, target, expr, type);
  2436. }
  2437. void HqlCppTranslator::buildAssignToTemp(BuildCtx & ctx, IHqlExpression * variable, IHqlExpression * expr)
  2438. {
  2439. CHqlBoundTarget boundTarget;
  2440. boundTarget.expr.set(variable);
  2441. buildExprAssign(ctx, boundTarget, expr);
  2442. }
  2443. void HqlCppTranslator::buildAssignViaTemp(BuildCtx & ctx, const CHqlBoundTarget & target, IHqlExpression * expr)
  2444. {
  2445. CHqlBoundExpr temp;
  2446. buildTempExpr(ctx, expr, temp);
  2447. buildExprAssign(ctx, target, temp.expr);
  2448. }
  2449. static bool canOptimizeIncrementAssign(ITypeInfo * type)
  2450. {
  2451. switch (type->getTypeCode())
  2452. {
  2453. case type_real:
  2454. return true;
  2455. case type_int:
  2456. switch (type->getSize())
  2457. {
  2458. case 1:
  2459. case 2:
  2460. case 4:
  2461. case 8:
  2462. return true;
  2463. }
  2464. }
  2465. return false;
  2466. }
  2467. IHqlExpression * HqlCppTranslator::optimizeIncrementAssign(BuildCtx & ctx, IHqlExpression * value)
  2468. {
  2469. //MORE: Could spot x += if(cond, y, 0) and convert to if (cond) x+= y; (especially if y is 1)
  2470. if (value->getOperator() == no_if)
  2471. {
  2472. IHqlExpression * left = value->queryChild(1);
  2473. IHqlExpression * right = value->queryChild(2);
  2474. if (isZero(right))
  2475. {
  2476. buildFilter(ctx, value->queryChild(0));
  2477. return optimizeIncrementAssign(ctx, left);
  2478. }
  2479. if (isZero(left))
  2480. {
  2481. OwnedHqlExpr filter = getInverse(value->queryChild(0));
  2482. buildFilter(ctx, filter);
  2483. return optimizeIncrementAssign(ctx, right);
  2484. }
  2485. }
  2486. if (isCast(value))
  2487. {
  2488. IHqlExpression * uncast = value->queryChild(0);
  2489. OwnedHqlExpr optimizedValue = optimizeIncrementAssign(ctx, uncast);
  2490. if (optimizedValue != uncast)
  2491. return ensureExprType(optimizedValue, value->queryType());
  2492. }
  2493. return LINK(value);
  2494. }
  2495. void HqlCppTranslator::buildIncrementAssign(BuildCtx & ctx, IHqlExpression * target, IHqlExpression * value)
  2496. {
  2497. // CHqlBoundExpr bound;
  2498. // buildExpr(ctx, target, bound);
  2499. ITypeInfo * type = target->queryType();
  2500. OwnedHqlExpr castValue = ensureExprType(value, type);
  2501. BuildCtx condctx(ctx);
  2502. if (options.optimizeIncrement)
  2503. {
  2504. castValue.setown(optimizeIncrementAssign(condctx, castValue));
  2505. if (isZero(castValue))
  2506. return;
  2507. if (canOptimizeIncrementAssign(type))
  2508. {
  2509. CHqlBoundExpr boundTarget;
  2510. buildExpr(condctx, target, boundTarget); // Not very clean!
  2511. CHqlBoundExpr boundValue;
  2512. buildExpr(condctx, castValue, boundValue);
  2513. condctx.addAssignIncrement(boundTarget.expr, boundValue.expr);
  2514. return;
  2515. }
  2516. }
  2517. OwnedHqlExpr plus = createValue(no_add, LINK(target), castValue.getClear());
  2518. buildAssign(condctx, target, plus);
  2519. }
  2520. void HqlCppTranslator::buildIncrementAssign(BuildCtx & ctx, IReferenceSelector * target, IHqlExpression * value)
  2521. {
  2522. buildIncrementAssign(ctx, target->queryExpr(), value);
  2523. // OwnedHqlExpr plus = createValue(no_add, LINK(type), ensureExprType(target->queryExpr(), type), ensureExprType(value, type));
  2524. // target->set(condctx, plus);
  2525. }
  2526. void HqlCppTranslator::buildIncrementAssign(BuildCtx & ctx, const CHqlBoundTarget & target, IHqlExpression * value)
  2527. {
  2528. ITypeInfo * type = target.queryType();
  2529. OwnedHqlExpr castValue = ensureExprType(value, type);
  2530. BuildCtx condctx(ctx);
  2531. if (options.optimizeIncrement)
  2532. {
  2533. castValue.setown(optimizeIncrementAssign(condctx, castValue));
  2534. if (isZero(castValue))
  2535. return;
  2536. if (canOptimizeIncrementAssign(type))
  2537. {
  2538. CHqlBoundExpr boundValue;
  2539. buildExpr(ctx, castValue, boundValue);
  2540. ctx.addAssignIncrement(target.expr, boundValue.expr);
  2541. return;
  2542. }
  2543. }
  2544. OwnedHqlExpr plus = createValue(no_add, target.getTranslatedExpr(), castValue.getClear());
  2545. buildExprAssign(ctx, target, plus);
  2546. }
  2547. void HqlCppTranslator::buildClear(BuildCtx & ctx, IHqlExpression * expr)
  2548. {
  2549. OwnedHqlExpr null = createNullExpr(expr);
  2550. buildAssign(ctx, expr, null);
  2551. }
  2552. void HqlCppTranslator::buildClear(BuildCtx & ctx, const CHqlBoundTarget & target)
  2553. {
  2554. if (target.length)
  2555. {
  2556. buildAssignToTemp(ctx, target.length, queryZero());
  2557. //NB: Don't need to clear target.pointer/target.variable if a length is defined......
  2558. return;
  2559. }
  2560. if (target.isFixedSize())
  2561. {
  2562. OwnedHqlExpr null = createNullExpr(target.expr);
  2563. buildExprAssign(ctx, target, null);
  2564. }
  2565. else
  2566. {
  2567. StringBuffer code;
  2568. if (hasWrapperModifier(target.queryType()))
  2569. generateExprCpp(code, target.expr).append(".clear();");
  2570. else
  2571. generateExprCpp(code, target.expr).append(" = 0;");
  2572. ctx.addQuoted(code);
  2573. }
  2574. }
  2575. void HqlCppTranslator::buildFilter(BuildCtx & ctx, IHqlExpression * expr)
  2576. {
  2577. node_operator op = expr->getOperator();
  2578. switch (op)
  2579. {
  2580. case no_attr:
  2581. case no_attr_expr:
  2582. case no_attr_link:
  2583. return;
  2584. case no_and:
  2585. doBuildFilterAnd(ctx, expr);
  2586. return;
  2587. case no_not:
  2588. {
  2589. IHqlExpression * child = expr->queryChild(0);
  2590. if ((child->getOperator() == no_or) && requiresTempAfterFirst(ctx, child))
  2591. {
  2592. OwnedHqlExpr inverted = convertOrToAnd(expr);
  2593. buildFilter(ctx,inverted);
  2594. return;
  2595. }
  2596. }
  2597. break;
  2598. case no_or:
  2599. if (requiresTempAfterFirst(ctx, expr))
  2600. {
  2601. OwnedHqlExpr inverted = convertOrToAnd(expr);
  2602. buildFilter(ctx, inverted);
  2603. return;
  2604. }
  2605. break;
  2606. case no_between:
  2607. case no_notbetween:
  2608. {
  2609. OwnedHqlExpr between = expandBetween(expr);
  2610. buildFilter(ctx, between);
  2611. return;
  2612. }
  2613. case no_alias_scope:
  2614. {
  2615. expandAliasScope(ctx, expr);
  2616. buildFilter(ctx, expr->queryChild(0));
  2617. return;
  2618. }
  2619. case no_compound:
  2620. {
  2621. buildStmt(ctx, expr->queryChild(0));
  2622. buildFilter(ctx, expr->queryChild(1));
  2623. break;
  2624. }
  2625. }
  2626. buildFilterViaExpr(ctx, expr);
  2627. }
  2628. IHqlStmt * HqlCppTranslator::buildFilterViaExpr(BuildCtx & ctx, IHqlExpression * expr)
  2629. {
  2630. //default action...
  2631. CHqlBoundExpr pure;
  2632. buildExpr(ctx, expr, pure);
  2633. if (pure.length) // check length is non zero
  2634. return ctx.addFilter(pure.length);
  2635. else
  2636. {
  2637. IHqlStmt * stmt = ctx.addFilter(pure.expr);
  2638. ctx.associateExpr(expr, queryBoolExpr(true));
  2639. return stmt;
  2640. }
  2641. }
  2642. void HqlCppTranslator::tidyupExpr(BuildCtx & ctx, CHqlBoundExpr & bound)
  2643. {
  2644. if (isPushed(bound))
  2645. {
  2646. HqlExprArray args;
  2647. callProcedure(ctx, DecPopLongId, args);
  2648. bound.expr.set(NULL);
  2649. }
  2650. }
  2651. void HqlCppTranslator::expandTranslated(IHqlExpression * expr, CHqlBoundExpr & tgt)
  2652. {
  2653. tgt.setFromTranslated(expr);
  2654. }
  2655. void HqlCppTranslator::buildCachedExpr(BuildCtx & ctx, IHqlExpression * expr, CHqlBoundExpr & tgt)
  2656. {
  2657. buildExpr(ctx, expr, tgt);
  2658. }
  2659. void HqlCppTranslator::buildAnyExpr(BuildCtx & ctx, IHqlExpression * expr, CHqlBoundExpr & tgt)
  2660. {
  2661. if (expr->isDataset())
  2662. buildDataset(ctx, expr, tgt, FormatNatural);
  2663. else if (expr->isDatarow())
  2664. {
  2665. Owned<IReferenceSelector> selector = buildNewRow(ctx, expr);
  2666. selector->buildAddress(ctx, tgt);
  2667. }
  2668. else
  2669. buildExpr(ctx, expr, tgt);
  2670. }
  2671. bool HqlCppTranslator::shouldEvaluateSelectAsAlias(BuildCtx & ctx, IHqlExpression * expr)
  2672. {
  2673. //If we're inside an activity that serializes onStart data, then the code to serialize fields
  2674. //selected from a row is generally simpler than the code to serialize the row itself.
  2675. if (insideActivityRemoteSerialize(ctx) && !expr->isList())
  2676. {
  2677. bool isNew;
  2678. IHqlExpression * ds = querySelectorDataset(expr, isNew);
  2679. if (isNew && ds->getOperator() == no_getgraphresult)
  2680. {
  2681. IHqlExpression * graphId = ds->queryChild(1);
  2682. if (isCurrentActiveGraph(ctx, graphId))
  2683. return true;
  2684. }
  2685. }
  2686. return false;
  2687. }
  2688. void HqlCppTranslator::buildExpr(BuildCtx & ctx, IHqlExpression * expr, CHqlBoundExpr & tgt)
  2689. {
  2690. node_operator op = expr->getOperator();
  2691. switch (op)
  2692. {
  2693. case no_counter:
  2694. doBuildExprCounter(ctx, expr, tgt);
  2695. return;
  2696. case no_evaluate:
  2697. doBuildExprEvaluate(ctx, expr, tgt);
  2698. return;
  2699. case no_thor:
  2700. throwUnexpectedX("THOR() annotation created in a scalar context");
  2701. // assertex(expr->queryType()->isScalar());
  2702. // buildExpr(ctx, expr->queryChild(0), tgt);
  2703. return;
  2704. case no_count:
  2705. if (!(expr->isPure() && ctx.getMatchExpr(expr, tgt)))
  2706. doBuildExprCount(ctx, expr, tgt);
  2707. return;
  2708. case no_max:
  2709. case no_min:
  2710. case no_sum:
  2711. if (!(expr->isPure() && ctx.getMatchExpr(expr, tgt)))
  2712. doBuildExprAggregate(ctx, expr, tgt);
  2713. return;
  2714. case no_exists:
  2715. if (!(expr->isPure() && ctx.getMatchExpr(expr, tgt)))
  2716. doBuildExprExists(ctx, expr, tgt);
  2717. return;
  2718. case no_countdict:
  2719. if (!(expr->isPure() && ctx.getMatchExpr(expr, tgt)))
  2720. doBuildExprCountDict(ctx, expr, tgt);
  2721. return;
  2722. case no_existsdict:
  2723. if (!(expr->isPure() && ctx.getMatchExpr(expr, tgt)))
  2724. doBuildExprExistsDict(ctx, expr, tgt);
  2725. return;
  2726. case no_existslist:
  2727. doBuildAggregateList(ctx, NULL, expr, &tgt);
  2728. return;
  2729. case no_countlist:
  2730. doBuildAggregateList(ctx, NULL, expr, &tgt);
  2731. return;
  2732. case no_sumlist:
  2733. doBuildAggregateList(ctx, NULL, expr, &tgt);
  2734. return;
  2735. case no_minlist:
  2736. doBuildAggregateList(ctx, NULL, expr, &tgt);
  2737. return;
  2738. case no_maxlist:
  2739. doBuildAggregateList(ctx, NULL, expr, &tgt);
  2740. return;
  2741. case no_sizeof:
  2742. doBuildExprSizeof(ctx, expr, tgt);
  2743. return;
  2744. case no_filepos:
  2745. doBuildExprFilepos(ctx, expr, tgt);
  2746. return;
  2747. case no_file_logicalname:
  2748. doBuildExprFileLogicalName(ctx, expr, tgt);
  2749. return;
  2750. case no_getresult:
  2751. case no_workunit_dataset:
  2752. doBuildExprGetResult(ctx, expr, tgt);
  2753. return;
  2754. case no_getgraphresult:
  2755. doBuildExprGetGraphResult(ctx, expr, tgt, FormatNatural);
  2756. return;
  2757. case no_regex_find:
  2758. case no_regex_replace:
  2759. doBuildExprRegexFindReplace(ctx, expr, tgt);
  2760. return;
  2761. case no_regex_findset:
  2762. doBuildExprRegexFindSet(ctx, expr, tgt);
  2763. return;
  2764. case no_skip:
  2765. case no_assert:
  2766. {
  2767. buildStmt(ctx, expr);
  2768. OwnedHqlExpr null = createNullExpr(expr);
  2769. buildExpr(ctx, null, tgt);
  2770. return;
  2771. }
  2772. case no_matched_injoin:
  2773. doBuildExprMatchedInJoin(ctx, expr, tgt);
  2774. return;
  2775. case no_matched:
  2776. case no_matchtext:
  2777. case no_matchlength:
  2778. case no_matchposition:
  2779. case no_matchunicode:
  2780. case no_matchutf8:
  2781. doBuildMatched(ctx, NULL, expr, &tgt);
  2782. return;
  2783. case no_matchattr:
  2784. doBuildMatchAttr(ctx, NULL, expr, &tgt);
  2785. return;
  2786. case no_rowdiff:
  2787. doBuildExprRowDiff(ctx, expr, tgt);
  2788. return;
  2789. case no_xmltext:
  2790. doBuildExprXmlText(ctx, expr, tgt);
  2791. return;
  2792. case no_xmlunicode:
  2793. doBuildExprXmlUnicode(ctx, expr, tgt);
  2794. return;
  2795. case no_evalonce:
  2796. doBuildEvalOnce(ctx, NULL, expr, &tgt);
  2797. return;
  2798. case no_alias:
  2799. doBuildExprAlias(ctx, expr, &tgt, NULL);
  2800. return;
  2801. case no_alias_scope:
  2802. {
  2803. expandAliasScope(ctx, expr);
  2804. buildExpr(ctx, expr->queryChild(0), tgt);
  2805. return;
  2806. }
  2807. case no_between:
  2808. case no_notbetween:
  2809. {
  2810. OwnedHqlExpr between = expandBetween(expr);
  2811. buildExpr(ctx, between, tgt);
  2812. return;
  2813. }
  2814. case no_libraryinput:
  2815. doBuildAliasValue(ctx, expr, tgt, NULL);
  2816. return;
  2817. case no_externalcall:
  2818. case no_call:
  2819. if (isTimed(expr))
  2820. buildTempExpr(ctx, expr, tgt);
  2821. else
  2822. doBuildExprCall(ctx, expr, tgt);
  2823. return;
  2824. case no_comma:
  2825. case no_compound:
  2826. buildStmt(ctx, expr->queryChild(0));
  2827. buildExpr(ctx, expr->queryChild(1), tgt);
  2828. return;
  2829. case no_cast:
  2830. case no_implicitcast:
  2831. doBuildExprCast(ctx, expr, tgt);
  2832. return;
  2833. case no_charlen:
  2834. tgt.expr.setown(doBuildCharLength(ctx, expr->queryChild(0)));
  2835. return;
  2836. case no_add:
  2837. doBuildExprAdd(ctx, expr, tgt);
  2838. return;
  2839. case no_mul:
  2840. case no_sub:
  2841. doBuildExprArith(ctx, expr, tgt);
  2842. return;
  2843. case no_abs:
  2844. doBuildExprAbs(ctx, expr, tgt);
  2845. return;
  2846. case no_negate:
  2847. doBuildExprNegate(ctx, expr, tgt);
  2848. return;
  2849. case no_div:
  2850. case no_modulus:
  2851. doBuildExprDivide(ctx, expr, tgt);
  2852. return;
  2853. case no_if:
  2854. doBuildExprIf(ctx, expr, tgt);
  2855. return;
  2856. case no_index:
  2857. doBuildExprIndex(ctx, expr, tgt);
  2858. return;
  2859. case no_list:
  2860. doBuildExprList(ctx, expr, tgt);
  2861. return;
  2862. case no_all:
  2863. doBuildExprAll(ctx, expr, tgt);
  2864. return;
  2865. case no_trim:
  2866. doBuildExprTrim(ctx, expr, tgt);
  2867. return;
  2868. case no_intformat:
  2869. doBuildExprFormat(intFormatId, ctx, expr, tgt);
  2870. return;
  2871. case no_realformat:
  2872. doBuildExprFormat(realFormatId, ctx, expr, tgt);
  2873. return;
  2874. case no_exp:
  2875. doBuildExprSysFunc(ctx, expr, tgt, clibExpId);
  2876. return;
  2877. case no_ln:
  2878. doBuildExprSysFunc(ctx, expr, tgt, lnId, options.divideByZeroAction);
  2879. return;
  2880. case no_sin:
  2881. doBuildExprSysFunc(ctx, expr, tgt, sinId);
  2882. return;
  2883. case no_cos:
  2884. doBuildExprSysFunc(ctx, expr, tgt, cosId);
  2885. return;
  2886. case no_tan:
  2887. doBuildExprSysFunc(ctx, expr, tgt, tanId);
  2888. return;
  2889. case no_asin:
  2890. doBuildExprSysFunc(ctx, expr, tgt, asinId, options.divideByZeroAction);
  2891. return;
  2892. case no_acos:
  2893. doBuildExprSysFunc(ctx, expr, tgt, acosId, options.divideByZeroAction);
  2894. return;
  2895. case no_atan:
  2896. doBuildExprSysFunc(ctx, expr, tgt, atanId);
  2897. return;
  2898. case no_atan2:
  2899. doBuildExprSysFunc(ctx, expr, tgt, atan2Id);
  2900. return;
  2901. case no_sinh:
  2902. doBuildExprSysFunc(ctx, expr, tgt, sinhId);
  2903. return;
  2904. case no_cosh:
  2905. doBuildExprSysFunc(ctx, expr, tgt, coshId);
  2906. return;
  2907. case no_tanh:
  2908. doBuildExprSysFunc(ctx, expr, tgt, tanhId);
  2909. return;
  2910. case no_log10:
  2911. doBuildExprSysFunc(ctx, expr, tgt, log10Id, options.divideByZeroAction);
  2912. return;
  2913. case no_power:
  2914. doBuildExprSysFunc(ctx, expr, tgt, powerId);
  2915. return;
  2916. case no_fail:
  2917. doBuildStmtFail(ctx, expr);
  2918. tgt.expr.setown(createConstant(0));
  2919. return;
  2920. case no_failcode:
  2921. doBuildExprFailCode(ctx, expr, tgt);
  2922. return;
  2923. case no_ordered:
  2924. doBuildExprOrdered(ctx, expr, tgt);
  2925. return;
  2926. case no_random:
  2927. doBuildExprSysFunc(ctx, expr, tgt, rtlRandomId);
  2928. return;
  2929. case no_rank:
  2930. doBuildExprRank(ctx, expr, tgt);
  2931. return;
  2932. case no_ranked:
  2933. doBuildExprRanked(ctx, expr, tgt);
  2934. return;
  2935. case no_round:
  2936. case no_roundup:
  2937. doBuildExprRound(ctx, expr, tgt);
  2938. return;
  2939. case no_sqrt:
  2940. doBuildExprSysFunc(ctx, expr, tgt, sqrtId, options.divideByZeroAction);
  2941. return;
  2942. case no_truncate:
  2943. doBuildExprTrunc(ctx, expr, tgt);
  2944. return;
  2945. case no_offsetof:
  2946. doBuildExprOffsetOf(ctx, expr, tgt);
  2947. return;
  2948. case no_substring:
  2949. doBuildExprSubString(ctx, expr, tgt);
  2950. return;
  2951. case no_in:
  2952. case no_notin:
  2953. {
  2954. if (expr->queryChild(1)->getOperator() == no_all)
  2955. tgt.expr.setown(createConstant(op == no_in));
  2956. else
  2957. {
  2958. OwnedHqlExpr optimized = querySimplifyInExpr(expr);
  2959. if (optimized)
  2960. {
  2961. OwnedHqlExpr folded = foldHqlExpression(optimized);
  2962. buildExpr(ctx, folded, tgt);
  2963. }
  2964. else
  2965. buildTempExpr(ctx, expr, tgt);
  2966. }
  2967. return;
  2968. }
  2969. case no_indict:
  2970. doBuildExprInDict(ctx, expr, tgt);
  2971. return;
  2972. case no_case:
  2973. case no_choose:
  2974. case no_concat:
  2975. case no_crc:
  2976. case no_hash:
  2977. case no_hash32:
  2978. case no_hash64:
  2979. case no_hashmd5:
  2980. case no_map:
  2981. case no_order:
  2982. case no_unicodeorder:
  2983. case no_rejected:
  2984. case no_which:
  2985. case no_addsets:
  2986. case no_createset:
  2987. case no_catch:
  2988. case no_failmessage:
  2989. case no_eventname:
  2990. case no_eventextra:
  2991. case no_loopcounter:
  2992. case no_toxml:
  2993. case no_tojson:
  2994. case no_executewhen:
  2995. buildTempExpr(ctx, expr, tgt);
  2996. return;
  2997. case no_asstring:
  2998. case no_typetransfer:
  2999. doBuildExprTransfer(ctx, expr, tgt);
  3000. return;
  3001. case no_translated:
  3002. {
  3003. expandTranslated(expr, tgt);
  3004. return;
  3005. }
  3006. case no_eq:
  3007. case no_ne:
  3008. case no_le:
  3009. case no_lt:
  3010. case no_ge:
  3011. case no_gt:
  3012. if (options.expressionPeephole)
  3013. {
  3014. OwnedHqlExpr optimized = peepholeOptimize(ctx, expr);
  3015. if (optimized != expr)
  3016. {
  3017. buildExpr(ctx, optimized, tgt);
  3018. return;
  3019. }
  3020. }
  3021. doBuildExprCompare(ctx, expr, tgt);
  3022. return;
  3023. case no_wuid:
  3024. doBuildExprWuid(ctx, expr, tgt);
  3025. return;
  3026. case no_getenv:
  3027. {
  3028. OwnedHqlExpr mapped = cvtGetEnvToCall(expr);
  3029. buildExpr(ctx, mapped, tgt);
  3030. return;
  3031. }
  3032. case no_notnot:
  3033. {
  3034. OwnedHqlExpr castChild = ensureExprType(expr->queryChild(0), queryBoolType());
  3035. buildExpr(ctx, castChild, tgt);
  3036. }
  3037. return;
  3038. case no_not:
  3039. {
  3040. IHqlExpression * child = expr->queryChild(0);
  3041. node_operator childOp = child->getOperator();
  3042. if (((childOp == no_and) || (childOp == no_or)) && requiresTempAfterFirst(ctx, child))
  3043. buildTempExpr(ctx, expr, tgt);
  3044. else
  3045. doBuildExprNot(ctx, expr, tgt);
  3046. return;
  3047. }
  3048. case no_constant:
  3049. {
  3050. ITypeInfo * type = expr->queryType();
  3051. if ((options.inlineStringThreshold > 0) && (type->getSize() > options.inlineStringThreshold) && (type->getSize() != UNKNOWN_LENGTH))
  3052. {
  3053. IHqlExpression * literal = addBigLiteral((const char *)expr->queryValue()->queryValue(), type->getSize());
  3054. Owned<ITypeInfo> retType = makeReferenceModifier(LINK(type));
  3055. switch (type->getTypeCode())
  3056. {
  3057. case type_unicode:
  3058. case type_varunicode:
  3059. case type_utf8:
  3060. literal = createValue(no_implicitcast, LINK(retType), literal);
  3061. break;
  3062. }
  3063. if (literal->queryType() != retType)
  3064. literal = createValue(no_typetransfer, LINK(retType), literal);
  3065. tgt.expr.setown(literal);
  3066. }
  3067. else
  3068. tgt.expr.set(expr);
  3069. return;
  3070. }
  3071. case no_quoted:
  3072. case no_variable:
  3073. tgt.expr.set(expr);
  3074. return;
  3075. case no_globalscope:
  3076. if (options.regressionTest && !ctx.queryMatchExpr(globalContextMarkerExpr))
  3077. throwUnexpected();
  3078. buildExpr(ctx, expr->queryChild(0), tgt);
  3079. return;
  3080. case no_nothor:
  3081. case no_nofold:
  3082. case no_nohoist:
  3083. case no_section:
  3084. case no_sectioninput:
  3085. case no_pure:
  3086. case no_forcegraph:
  3087. case no_nocombine:
  3088. case no_likely:
  3089. case no_unlikely:
  3090. buildExpr(ctx, expr->queryChild(0), tgt);
  3091. return;
  3092. case no_band:
  3093. case no_bor:
  3094. case no_bnot:
  3095. case no_bxor:
  3096. case no_lshift:
  3097. case no_rshift:
  3098. doBuildPureSubExpr(ctx, expr, tgt);
  3099. return;
  3100. //MORE: Shouldn't these be special cased?
  3101. case no_xor:
  3102. doBuildPureSubExpr(ctx, expr, tgt);
  3103. return;
  3104. case no_select:
  3105. {
  3106. OwnedHqlExpr aggregate = convertToSimpleAggregate(expr);
  3107. if (aggregate && canProcessInline(&ctx, aggregate->queryChild(0)))
  3108. {
  3109. buildExpr(ctx, aggregate, tgt);
  3110. return;
  3111. }
  3112. if (shouldEvaluateSelectAsAlias(ctx, expr))
  3113. {
  3114. doBuildAliasValue(ctx, expr, tgt, NULL);
  3115. return;
  3116. }
  3117. Owned<IReferenceSelector> selector = buildReference(ctx, expr);
  3118. selector->get(ctx, tgt);
  3119. return;
  3120. }
  3121. case no_field:
  3122. throwUnexpected();
  3123. case no_is_null:
  3124. {
  3125. //Until we have something better in place isNull is the inverse of isValid().
  3126. IHqlExpression * child = expr->queryChild(0);
  3127. OwnedHqlExpr null = createValue(no_not, makeBoolType(), createValue(no_is_valid, makeBoolType(), LINK(child)));
  3128. buildExpr(ctx, null, tgt);
  3129. }
  3130. return;
  3131. case no_is_valid:
  3132. doBuildExprIsValid(ctx, expr, tgt);
  3133. return;
  3134. case no_fromunicode:
  3135. case no_tounicode:
  3136. doBuildExprToFromUnicode(ctx, expr, tgt);
  3137. return;
  3138. case no_keyunicode:
  3139. doBuildExprKeyUnicode(ctx, expr, tgt);
  3140. return;
  3141. case no_xmldecode:
  3142. case no_xmlencode:
  3143. buildTempExpr(ctx, expr, tgt);
  3144. return;
  3145. case no_and:
  3146. case no_or:
  3147. if (requiresTempAfterFirst(ctx, expr))
  3148. buildTempExpr(ctx, expr, tgt);
  3149. else
  3150. doBuildPureSubExpr(ctx, expr, tgt);
  3151. return;
  3152. case no_assertkeyed:
  3153. case no_assertwild:
  3154. {
  3155. StringBuffer s;
  3156. throwError1(HQLERR_KeyedWildNoIndex, getExprECL(expr, s).str());
  3157. }
  3158. case no_assertstepped:
  3159. {
  3160. StringBuffer s;
  3161. throwError1(HQLERR_SteppedNoJoin, getExprECL(expr, s).str());
  3162. }
  3163. case no_id2blob:
  3164. doBuildExprIdToBlob(ctx, expr, tgt);
  3165. return;
  3166. case no_blob2id:
  3167. doBuildExprBlobToId(ctx, expr, tgt);
  3168. return;
  3169. case no_embedbody:
  3170. doBuildExprEmbedBody(ctx, expr, &tgt);
  3171. return;
  3172. case no_null:
  3173. tgt.length.setown(getSizetConstant(0));
  3174. tgt.expr.setown(createValue(no_nullptr, makeReferenceModifier(expr->getType())));
  3175. return;
  3176. case no_clustersize:
  3177. doBuildExprSysFunc(ctx, expr, tgt, getClusterSizeId);
  3178. return;
  3179. case no_deref:
  3180. //Untested
  3181. buildExpr(ctx, expr->queryChild(0), tgt);
  3182. if (tgt.expr->getOperator() == no_address)
  3183. tgt.expr.setown(createValue(no_typetransfer, expr->getType(), LINK(tgt.expr->queryChild(0))));
  3184. else
  3185. tgt.expr.setown(createValue(no_deref, expr->getType(), LINK(tgt.expr)));
  3186. return;
  3187. case no_funcdef:
  3188. tgt.expr.setown(doBuildInternalFunction(expr));
  3189. useFunction(tgt.expr);
  3190. return;
  3191. case no_purevirtual:
  3192. case no_internalselect:
  3193. {
  3194. //This shouldn't happen we should have an no_checkconcrete wrapper inserted into the tree like checkconstant,
  3195. //but it currently can in obscure library contexts (e.g., library3ie2.xhql)
  3196. IAtom * name = expr->queryName();
  3197. throwError1(HQLERR_ConcreteMemberRequired, name ? str(name) : "");
  3198. }
  3199. case NO_AGGREGATEGROUP:
  3200. throwError1(HQLERR_OutsideGroupAggregate, getOpString(op));
  3201. default:
  3202. break;
  3203. }
  3204. EclIR::dbglogIR(expr);
  3205. StringBuffer msg;
  3206. msg.append("Unexpected operator '").append(getOpString(op)).append("' in: HqlCppTranslator::buildExpr(");
  3207. toECL(expr, msg, true);
  3208. msg.append(")");
  3209. throw MakeStringException(HQLERR_UnexpectedOperator, "%s", msg.str());
  3210. }
  3211. void HqlCppTranslator::buildExprOrAssign(BuildCtx & ctx, const CHqlBoundTarget * target, IHqlExpression * expr, CHqlBoundExpr * tgt)
  3212. {
  3213. if (target)
  3214. buildExprAssign(ctx, *target, expr);
  3215. else if (tgt)
  3216. buildExpr(ctx, expr, *tgt);
  3217. else
  3218. buildStmt(ctx, expr);
  3219. }
  3220. bool HqlCppTranslator::specialCaseBoolReturn(BuildCtx & ctx, IHqlExpression * expr)
  3221. {
  3222. if (!options.optimizeBoolReturn)
  3223. return false;
  3224. if ((expr->getOperator() == no_and) && (unwoundCount(expr, no_and) <= MAX_NESTED_CASES))
  3225. return true;
  3226. if (expr->getOperator() == no_not)
  3227. expr = expr->queryChild(0);
  3228. if (!requiresTemp(ctx, expr, true))
  3229. return false;
  3230. if (expr->getOperator() == no_alias_scope)
  3231. expr = expr->queryChild(0);
  3232. if (expr->getOperator() == no_compound)
  3233. expr = expr->queryChild(1);
  3234. if ((expr->getOperator() == no_and) || (expr->getOperator() == no_or))
  3235. return true;
  3236. return false;
  3237. }
  3238. void HqlCppTranslator::buildReturn(BuildCtx & ctx, IHqlExpression * expr, ITypeInfo * retType)
  3239. {
  3240. ITypeInfo * exprType = expr->queryType();
  3241. if (!retType) retType = exprType;
  3242. expr = queryExpandAliasScope(ctx, expr);
  3243. node_operator op = expr->getOperator();
  3244. type_t returntc = retType->getTypeCode();
  3245. if ((retType->getSize() == UNKNOWN_LENGTH) && (returntc == type_varstring))
  3246. {
  3247. if (hasConstModifier(retType) && (hasConstModifier(exprType) || expr->queryValue()))
  3248. {
  3249. OwnedHqlExpr cast = ensureExprType(expr, retType);
  3250. CHqlBoundExpr ret;
  3251. buildCachedExpr(ctx, cast, ret);
  3252. ctx.setNextDestructor();
  3253. ctx.addReturn(ret.expr);
  3254. }
  3255. else
  3256. {
  3257. if (hasConstModifier(retType))
  3258. {
  3259. IIdAtom * funcAtom;
  3260. if (expr->queryType()->getTypeCode()==type_varstring)
  3261. funcAtom = cloneVStringId;
  3262. else
  3263. funcAtom = cloneVStringXId;
  3264. HqlExprArray args;
  3265. args.append(*LINK(expr));
  3266. OwnedHqlExpr call = bindFunctionCall(funcAtom, args);
  3267. CHqlBoundExpr ret;
  3268. buildExpr(ctx, call, ret);
  3269. ctx.addReturn(ret.expr);
  3270. return;
  3271. }
  3272. CHqlBoundTarget retVar;
  3273. retVar.expr.setown(createWrapperTemp(ctx, retType, typemod_none));
  3274. buildExprAssign(ctx, retVar, expr);
  3275. ctx.setNextDestructor();
  3276. StringBuffer s;
  3277. retVar.expr->toString(s);
  3278. switch (retType->getTypeCode())
  3279. {
  3280. case type_varstring:
  3281. s.append(".detachstr()");
  3282. break;
  3283. case type_varunicode:
  3284. s.append(".detachustr()");
  3285. break;
  3286. default:
  3287. UNIMPLEMENTED;
  3288. }
  3289. OwnedHqlExpr temp = createQuoted(s.str(), LINK(exprType));
  3290. ctx.addReturn(temp);
  3291. }
  3292. }
  3293. else if ((returntc == type_row) && hasLinkCountedModifier(retType))
  3294. {
  3295. CHqlBoundTarget result;
  3296. buildTempExpr(ctx, ctx, result, expr, FormatNatural, false);
  3297. //MORE: There should be a cleaner way of doing this
  3298. StringBuffer s;
  3299. result.expr->toString(s);
  3300. s.append(".getClear()");
  3301. OwnedHqlExpr temp = createQuoted(s.str(), LINK(exprType));
  3302. ctx.addReturn(temp);
  3303. }
  3304. else if ((returntc == type_table) && hasStreamedModifier(retType))
  3305. {
  3306. CHqlBoundTarget result;
  3307. buildTempExpr(ctx, ctx, result, expr, FormatStreamedDataset, false);
  3308. //MORE: There should be a cleaner way of doing this
  3309. StringBuffer s;
  3310. result.expr->toString(s);
  3311. s.append(".getClear()");
  3312. OwnedHqlExpr temp = createQuoted(s.str(), LINK(exprType));
  3313. ctx.addReturn(temp);
  3314. }
  3315. else if ((returntc == type_boolean) && specialCaseBoolReturn(ctx, expr))
  3316. {
  3317. bool successValue = true;
  3318. if (op == no_not)
  3319. {
  3320. //!(a and b) is converted into !a || !b. Otherwise just invert the test condition.
  3321. IHqlExpression * child = expr->queryChild(0);
  3322. if (child->getOperator() == no_alias_scope)
  3323. child = child->queryChild(0);
  3324. if (child->getOperator() != no_and)
  3325. {
  3326. successValue = false;
  3327. expr = expr->queryChild(0);
  3328. }
  3329. }
  3330. BuildCtx condctx(ctx);
  3331. buildFilteredReturn(condctx, expr, queryBoolExpr(successValue));
  3332. buildReturn(ctx, queryBoolExpr(!successValue));
  3333. }
  3334. else if (op == no_if)
  3335. {
  3336. OwnedHqlExpr castTrue = ensureExprType(expr->queryChild(1), retType);
  3337. OwnedHqlExpr castFalse = ensureExprType(expr->queryChild(2), retType);
  3338. BuildCtx condctx(ctx);
  3339. buildFilter(condctx, expr->queryChild(0));
  3340. buildReturn(condctx, castTrue);
  3341. buildReturn(ctx, castFalse);
  3342. }
  3343. else if (op == no_map || op == no_case)
  3344. {
  3345. HqlCppCaseInfo info(*this);
  3346. doBuildCaseInfo(expr, info);
  3347. info.buildReturn(ctx);
  3348. }
  3349. else
  3350. {
  3351. CHqlBoundExpr ret;
  3352. OwnedHqlExpr castExpr = ensureExprType(expr, retType);
  3353. buildExpr(ctx, castExpr, ret);
  3354. ctx.addReturn(ret.expr);
  3355. }
  3356. }
  3357. //Assumes that the value being returned is simple.
  3358. //for (a || b || c) gen if (a) return x; if (b) return x; if (c) return x;
  3359. //and !(a && b && c) -> !a || !b || !c
  3360. void HqlCppTranslator::buildFilteredReturn(BuildCtx & ctx, IHqlExpression * filter, IHqlExpression * value)
  3361. {
  3362. filter = queryExpandAliasScope(ctx, filter);
  3363. HqlExprArray conds;
  3364. node_operator op = filter->getOperator();
  3365. if (op == no_or)
  3366. {
  3367. buildFilteredReturn(ctx, filter->queryChild(0), value);
  3368. buildFilteredReturn(ctx, filter->queryChild(1), value);
  3369. return;
  3370. }
  3371. if (op == no_not)
  3372. {
  3373. IHqlExpression * child = filter->queryChild(0);
  3374. node_operator childOp = child->getOperator();
  3375. if (childOp == no_and)
  3376. {
  3377. child->unwindList(conds, no_and);
  3378. ForEachItemIn(i, conds)
  3379. {
  3380. IHqlExpression & cur = conds.item(i);
  3381. OwnedHqlExpr inverse = getInverse(&cur);
  3382. buildFilteredReturn(ctx, inverse, value);
  3383. }
  3384. return;
  3385. }
  3386. if (childOp == no_alias_scope)
  3387. {
  3388. expandAliasScope(ctx, child);
  3389. OwnedHqlExpr inverse = getInverse(child->queryChild(0));
  3390. buildFilteredReturn(ctx, inverse, value);
  3391. return;
  3392. }
  3393. }
  3394. BuildCtx condctx(ctx);
  3395. buildFilter(condctx, filter);
  3396. if (value)
  3397. buildReturn(condctx, value);
  3398. else
  3399. condctx.addReturn(NULL);
  3400. }
  3401. void HqlCppTranslator::buildStmt(BuildCtx & _ctx, IHqlExpression * expr)
  3402. {
  3403. BuildCtx ctx(_ctx);
  3404. node_operator op = expr->getOperator();
  3405. switch (op)
  3406. {
  3407. case no_assign:
  3408. doBuildStmtAssign(ctx, expr->queryChild(0), expr->queryChild(1));
  3409. return;
  3410. case no_assign_addfiles:
  3411. doBuildStmtAssignModify(ctx, expr->queryChild(0), expr->queryChild(1), op);
  3412. return;
  3413. case no_alias:
  3414. doBuildExprAlias(ctx, expr, NULL, NULL);
  3415. return;
  3416. case no_alias_scope:
  3417. {
  3418. expandAliasScope(ctx, expr);
  3419. buildStmt(ctx, expr->queryChild(0));
  3420. return;
  3421. }
  3422. case no_assignall:
  3423. {
  3424. unsigned idx;
  3425. unsigned kids = expr->numChildren();
  3426. for (idx = 0; idx < kids; idx++)
  3427. buildStmt(ctx, expr->queryChild(idx));
  3428. return;
  3429. }
  3430. case no_comma:
  3431. case no_compound:
  3432. buildStmt(ctx, expr->queryChild(0));
  3433. buildStmt(ctx, expr->queryChild(1));
  3434. return;
  3435. case no_if:
  3436. doBuildStmtIf(ctx, expr);
  3437. return;
  3438. case no_call:
  3439. case no_externalcall:
  3440. doBuildStmtCall(ctx, expr);
  3441. return;
  3442. case no_nofold:
  3443. case no_nohoist:
  3444. case no_nothor:
  3445. case no_section:
  3446. case no_sectioninput:
  3447. case no_forcegraph:
  3448. case no_nocombine:
  3449. case no_likely:
  3450. case no_unlikely:
  3451. buildStmt(ctx, expr->queryChild(0));
  3452. return;
  3453. case no_null:
  3454. return;
  3455. case no_fail:
  3456. doBuildStmtFail(ctx, expr);
  3457. return;
  3458. case no_setmeta:
  3459. return;
  3460. case no_update:
  3461. doBuildStmtUpdate(ctx, expr);
  3462. return;
  3463. case no_output:
  3464. if (queryRealChild(expr, 1))
  3465. throwError1(HQLERR_NotSupportedInsideNoThor, "OUTPUT to file");
  3466. doBuildStmtOutput(ctx, expr);
  3467. return;
  3468. case no_subgraph:
  3469. doBuildThorChildSubGraph(ctx, expr, SubGraphRoot);
  3470. return;
  3471. case no_thor:
  3472. doBuildThorGraph(ctx, expr);
  3473. return;
  3474. case no_workflow_action:
  3475. return;
  3476. case no_ensureresult:
  3477. doBuildStmtEnsureResult(ctx, expr);
  3478. return;
  3479. case no_extractresult:
  3480. case no_setresult:
  3481. doBuildStmtSetResult(ctx, expr);
  3482. return;
  3483. case no_parallel:
  3484. case no_sequential:
  3485. case no_actionlist:
  3486. case no_orderedactionlist:
  3487. {
  3488. ForEachChild(idx, expr)
  3489. {
  3490. BuildCtx subctx(ctx);
  3491. //Add a group for each branch of a sequential to ensure all branches are independent
  3492. if (op == no_sequential)
  3493. subctx.addGroup();
  3494. buildStmt(subctx, expr->queryChild(idx));
  3495. }
  3496. return;
  3497. }
  3498. case no_wait:
  3499. doBuildStmtWait(ctx, expr);
  3500. return;
  3501. case no_notify:
  3502. doBuildStmtNotify(ctx, expr);
  3503. return;
  3504. case no_skip:
  3505. doBuildStmtSkip(ctx, expr, NULL);
  3506. return;
  3507. case no_assert:
  3508. doBuildStmtAssert(ctx, expr);
  3509. return;
  3510. case no_embedbody:
  3511. doBuildExprEmbedBody(ctx, expr, NULL);
  3512. return;
  3513. case no_setworkflow_cond:
  3514. {
  3515. HqlExprArray args;
  3516. args.append(*LINK(expr->queryChild(0)));
  3517. buildFunctionCall(ctx, setWorkflowConditionId, args);
  3518. return;
  3519. }
  3520. case no_apply:
  3521. doBuildStmtApply(ctx, expr);
  3522. return;
  3523. case no_cluster:
  3524. doBuildStmtCluster(ctx, expr);
  3525. return;
  3526. case no_choose:
  3527. doBuildChoose(ctx, NULL, expr);
  3528. return;
  3529. case no_persist_check:
  3530. buildWorkflowPersistCheck(ctx, expr);
  3531. return;
  3532. case no_childquery:
  3533. buildChildGraph(ctx, expr);
  3534. return;
  3535. case no_evaluate_stmt:
  3536. expr = expr->queryChild(0);
  3537. if (expr->queryValue())
  3538. return;
  3539. break; // evaluate default behaviour.
  3540. }
  3541. CHqlBoundExpr tgt;
  3542. buildAnyExpr(ctx, expr, tgt);
  3543. ctx.addExpr(tgt.expr);
  3544. tidyupExpr(ctx, tgt);
  3545. }
  3546. class AliasExpansionInfo
  3547. {
  3548. public:
  3549. void pushCondition(IHqlExpression * expr, unsigned branch) { conditions.append(*createAttribute(branchAtom, LINK(expr), getSizetConstant(branch))); }
  3550. void popCondition() { conditions.pop(); }
  3551. void popConditions(unsigned num) { conditions.popn(num); }
  3552. bool isConditional() { return conditions.ordinality() != 0; }
  3553. IHqlExpression * createConditionIntersection(IHqlExpression * prev)
  3554. {
  3555. if (conditions.ordinality() == 0)
  3556. return NULL;
  3557. if (!prev)
  3558. return createValueSafe(no_sortlist, makeSortListType(NULL), conditions);
  3559. unsigned maxPrev = prev->numChildren();
  3560. unsigned max = maxPrev;
  3561. if (max > conditions.ordinality())
  3562. max = conditions.ordinality();
  3563. for (unsigned i=0; i < max; i++)
  3564. {
  3565. if (&conditions.item(i) != prev->queryChild(i))
  3566. {
  3567. if (i == 0)
  3568. return NULL;
  3569. return createValueSafe(no_sortlist, makeSortListType(NULL), conditions, 0, i);
  3570. }
  3571. }
  3572. if (max == maxPrev)
  3573. return LINK(prev);
  3574. return createValueSafe(no_sortlist, makeSortListType(NULL), conditions);
  3575. }
  3576. HqlExprArray conditions;
  3577. };
  3578. void HqlCppTranslator::doExpandAliases(BuildCtx & ctx, IHqlExpression * expr, AliasExpansionInfo & info)
  3579. {
  3580. IHqlExpression * prev = static_cast<IHqlExpression *>(expr->queryTransformExtra());
  3581. if (prev == expr)
  3582. return;
  3583. #ifdef USE_NEW_ALIAS_CODE
  3584. OwnedHqlExpr commonPath = info.createConditionIntersection(prev);
  3585. if (prev == commonPath)
  3586. return;
  3587. if (commonPath)
  3588. expr->setTransformExtra(commonPath);
  3589. else
  3590. expr->setTransformExtraUnlinked(expr);
  3591. node_operator op = expr->getOperator();
  3592. switch (op)
  3593. {
  3594. //MORE: Anything that creates a child query shouldn't be included here...
  3595. case no_select:
  3596. case NO_AGGREGATE:
  3597. case no_alias_scope:
  3598. break;
  3599. case no_alias:
  3600. {
  3601. IHqlExpression * value = expr->queryChild(0);
  3602. if ((commonPath == NULL) && !ctx.queryMatchExpr(value))
  3603. {
  3604. if (containsAliasLocally(value) && !expr->hasAttribute(globalAtom))
  3605. doExpandAliases(ctx, value, info);
  3606. doBuildExprAlias(ctx, expr, NULL, &info);
  3607. }
  3608. break;
  3609. }
  3610. case no_and:
  3611. case no_or:
  3612. {
  3613. HqlExprArray args;
  3614. expr->unwindList(args, op);
  3615. doExpandAliases(ctx, &args.item(0), info);
  3616. unsigned max = args.ordinality();
  3617. for (unsigned i=1; i < max; i++)
  3618. {
  3619. info.pushCondition(expr, i);
  3620. doExpandAliases(ctx, &args.item(i), info);
  3621. }
  3622. info.popConditions(max-1);
  3623. break;
  3624. }
  3625. case no_if:
  3626. {
  3627. doExpandAliases(ctx, expr->queryChild(0), info);
  3628. info.pushCondition(expr, 1);
  3629. doExpandAliases(ctx, expr->queryChild(1), info);
  3630. info.popCondition();
  3631. info.pushCondition(expr, 2);
  3632. doExpandAliases(ctx, expr->queryChild(2), info);
  3633. info.popCondition();
  3634. break;
  3635. }
  3636. case no_case:
  3637. {
  3638. doExpandAliases(ctx, expr->queryChild(0), info);
  3639. unsigned max = expr->numChildren();
  3640. for (unsigned i=1; i < max-1; i++)
  3641. {
  3642. info.pushCondition(expr, i*2);
  3643. doExpandAliases(ctx, expr->queryChild(i)->queryChild(0), info);
  3644. info.popCondition();
  3645. info.pushCondition(expr, i*2+1);
  3646. doExpandAliases(ctx, expr->queryChild(i)->queryChild(1), info);
  3647. info.popCondition();
  3648. }
  3649. info.pushCondition(expr, (max-1)*2);
  3650. doExpandAliases(ctx, expr->queryChild(max-1), info);
  3651. info.popCondition();
  3652. break;
  3653. }
  3654. case no_map:
  3655. {
  3656. //The following is equivalent to old, code; I'm not sure it is the best implementation
  3657. unsigned max = expr->numChildren();
  3658. for (unsigned i=0; i < max-1; i++)
  3659. {
  3660. info.pushCondition(expr, i*2);
  3661. doExpandAliases(ctx, expr->queryChild(i)->queryChild(0), info);
  3662. info.popCondition();
  3663. info.pushCondition(expr, i*2+1);
  3664. doExpandAliases(ctx, expr->queryChild(i)->queryChild(1), info);
  3665. info.popCondition();
  3666. }
  3667. info.pushCondition(expr, (max-1)*2);
  3668. doExpandAliases(ctx, expr->queryChild(max-1), info);
  3669. info.popCondition();
  3670. break;
  3671. }
  3672. default:
  3673. if (containsAliasLocally(expr))
  3674. {
  3675. ForEachChild(i, expr)
  3676. doExpandAliases(ctx, expr->queryChild(i), info);
  3677. }
  3678. break;
  3679. }
  3680. #else
  3681. expr->setTransformExtraUnlinked(expr);
  3682. node_operator op = expr->getOperator();
  3683. switch (op)
  3684. {
  3685. //MORE: Anything that creates a child query shouldn't be included here...
  3686. case no_select:
  3687. case NO_AGGREGATE:
  3688. case no_alias_scope:
  3689. break;
  3690. case no_alias:
  3691. {
  3692. doBuildExprAlias(ctx, expr, NULL, &info);
  3693. break;
  3694. }
  3695. default:
  3696. if (containsAliasLocally(expr))
  3697. {
  3698. ForEachChild(i, expr)
  3699. doExpandAliases(ctx, expr->queryChild(i), info);
  3700. }
  3701. break;
  3702. }
  3703. #endif
  3704. }
  3705. void HqlCppTranslator::expandAliases(BuildCtx & ctx, IHqlExpression * expr, AliasExpansionInfo * parentInfo)
  3706. {
  3707. if (containsAliasLocally(expr))
  3708. {
  3709. if (!parentInfo)
  3710. {
  3711. TransformMutexBlock block;
  3712. AliasExpansionInfo info;
  3713. doExpandAliases(ctx, expr, info);
  3714. }
  3715. else
  3716. doExpandAliases(ctx, expr, *parentInfo);
  3717. }
  3718. }
  3719. void HqlCppTranslator::expandAliasScope(BuildCtx & ctx, IHqlExpression * expr)
  3720. {
  3721. TransformMutexBlock block;
  3722. AliasExpansionInfo info;
  3723. unsigned max = expr->numChildren();
  3724. for (unsigned idx = 1; idx < max; idx++)
  3725. {
  3726. IHqlExpression * child = expr->queryChild(idx);
  3727. if (containsAliasLocally(child))
  3728. doExpandAliases(ctx, child, info);
  3729. }
  3730. }
  3731. //------------------------------------------------------------------------------
  3732. void HqlCppTranslator::gatherActiveCursors(BuildCtx & ctx, HqlExprCopyArray & activeRows)
  3733. {
  3734. AssociationIterator iter(ctx);
  3735. ForEach(iter)
  3736. {
  3737. HqlExprAssociation & cur = iter.get();
  3738. if (cur.isRowAssociation())
  3739. {
  3740. BoundRow & curRow = static_cast<BoundRow &>(cur);
  3741. if ((curRow.querySide() != no_self) && !curRow.isBuilder())
  3742. activeRows.append(*curRow.querySelector());
  3743. }
  3744. else if (cur.represents->getOperator() == no_counter)
  3745. activeRows.append(*cur.represents);
  3746. }
  3747. }
  3748. bool HqlCppTranslator::canEvaluateInContext(BuildCtx & ctx, IHqlExpression * expr)
  3749. {
  3750. HqlExprCopyArray cursors;
  3751. gatherActiveCursors(ctx, cursors);
  3752. return ::canEvaluateInScope(cursors, expr);
  3753. }
  3754. bool mustEvaluateInContext(BuildCtx & ctx, IHqlExpression * expr)
  3755. {
  3756. HqlExprCopyArray required;
  3757. expr->gatherTablesUsed(NULL, &required);
  3758. if (required.ordinality() == 0)
  3759. return false;
  3760. HqlExprCopyArray activeRows;
  3761. HqlExprCopyArray inheritedRows;
  3762. RowAssociationIterator iter(ctx);
  3763. ForEach(iter)
  3764. {
  3765. BoundRow & cur = iter.get();
  3766. if ((cur.querySide() != no_self) && !cur.isBuilder())
  3767. {
  3768. IHqlExpression * selector = cur.querySelector();
  3769. if (cur.isInherited())
  3770. inheritedRows.append(*selector);
  3771. else
  3772. activeRows.append(*selector);
  3773. }
  3774. }
  3775. //Ensure all instances of activeRows which match the inherited rows are removed
  3776. ForEachItemInRev(i, activeRows)
  3777. {
  3778. if (inheritedRows.find(activeRows.item(i)))
  3779. activeRows.remove(i);
  3780. }
  3781. return canEvaluateInScope(activeRows, required);
  3782. }
  3783. bool filterIsTableInvariant(IHqlExpression * expr)
  3784. {
  3785. IHqlExpression * dsSelector = expr->queryChild(0)->queryNormalizedSelector();
  3786. ForEachChildFrom(i, expr, 1)
  3787. {
  3788. IHqlExpression * cur = expr->queryChild(i);
  3789. if (containsSelector(cur, dsSelector))
  3790. return false;
  3791. }
  3792. return true;
  3793. }
  3794. //-----------------------------------------------------------------------------
  3795. bool LoopInvariantHelper::getBestContext(BuildCtx & ctx, IHqlExpression * expr)
  3796. {
  3797. finished();
  3798. active = ctx.selectBestContext(expr);
  3799. return (active != NULL);
  3800. }
  3801. void LoopInvariantHelper::finished()
  3802. {
  3803. if (active)
  3804. {
  3805. active->mergeScopeWithContainer();
  3806. active = NULL;
  3807. }
  3808. }
  3809. //---------------------------------------------------------------------------
  3810. void HqlCppTranslator::buildBlockCopy(BuildCtx & ctx, IHqlExpression * tgt, CHqlBoundExpr & src)
  3811. {
  3812. OwnedHqlExpr size = getBoundSize(src);
  3813. if (!size->queryValue() || size->queryValue()->getIntValue() != 0)
  3814. {
  3815. HqlExprArray args;
  3816. args.append(*getPointer(tgt));
  3817. args.append(*getPointer(src.expr));
  3818. args.append(*size.getClear());
  3819. callProcedure(ctx, memcpyId, args);
  3820. }
  3821. }
  3822. void HqlCppTranslator::buildSimpleExpr(BuildCtx & ctx, IHqlExpression * expr, CHqlBoundExpr & tgt)
  3823. {
  3824. node_operator op = expr->getOperator();
  3825. bool simple = false;
  3826. switch (op)
  3827. {
  3828. case no_alias_scope:
  3829. {
  3830. expandAliasScope(ctx, expr);
  3831. buildSimpleExpr(ctx, expr->queryChild(0), tgt);
  3832. return;
  3833. }
  3834. case no_preservemeta:
  3835. buildSimpleExpr(ctx, expr->queryChild(0), tgt);
  3836. return;
  3837. case no_constant:
  3838. case no_variable:
  3839. case no_getresult: // gets forced into a variable.
  3840. case no_workunit_dataset: // gets forced into a variable.
  3841. case no_getgraphresult: // gets forced into a variable.
  3842. case no_alias:
  3843. case no_list:
  3844. case no_all:
  3845. case no_null:
  3846. case no_id2blob:
  3847. case no_rows:
  3848. case no_libraryinput:
  3849. simple = true;
  3850. break;
  3851. case no_compound:
  3852. buildStmt(ctx, expr->queryChild(0));
  3853. buildSimpleExpr(ctx, expr->queryChild(1), tgt);
  3854. return;
  3855. case no_field:
  3856. throwUnexpected();
  3857. case no_select:
  3858. {
  3859. CHqlBoundExpr bound;
  3860. buildCachedExpr(ctx, expr, bound);
  3861. if (isSimpleTranslatedExpr(bound.expr))
  3862. tgt.set(bound);
  3863. else
  3864. {
  3865. OwnedHqlExpr trans = bound.getTranslatedExpr();
  3866. buildTempExpr(ctx, trans, tgt);
  3867. }
  3868. return;
  3869. }
  3870. break; // should depend on whether conditional etc....
  3871. case no_translated:
  3872. simple = isSimpleTranslatedExpr(expr->queryChild(0));
  3873. break;
  3874. case no_substring:
  3875. {
  3876. SubStringInfo info(expr);
  3877. if (info.canGenerateInline() || expr->hasAttribute(quickAtom))
  3878. simple = true;
  3879. break;
  3880. }
  3881. case no_cast:
  3882. case no_implicitcast:
  3883. {
  3884. //special case casting a string to (string) - saves lots of temporaries.
  3885. ITypeInfo * exprType = expr->queryType();
  3886. IHqlExpression * child = expr->queryChild(0);
  3887. ITypeInfo * childType = child->queryType();
  3888. if ((exprType->getTypeCode() == type_string) && (exprType->getSize() == UNKNOWN_LENGTH))
  3889. {
  3890. if ((childType->getTypeCode() == type_string) && (exprType->queryCharset() == childType->queryCharset()))
  3891. {
  3892. buildSimpleExpr(ctx, child, tgt);
  3893. return;
  3894. }
  3895. }
  3896. if (options.foldConstantCast && (child->getOperator() == no_constant))
  3897. simple = true;
  3898. break;
  3899. }
  3900. case no_sizeof:
  3901. case no_offsetof:
  3902. simple = true;
  3903. break;
  3904. case no_regex_find:
  3905. simple = expr->isBoolean();
  3906. break;
  3907. case no_call:
  3908. case no_externalcall:
  3909. {
  3910. ITypeInfo * type = expr->queryType();
  3911. switch (type->getTypeCode())
  3912. {
  3913. case type_set:
  3914. simple = true;
  3915. break;
  3916. case type_string:
  3917. case type_data:
  3918. case type_qstring:
  3919. if (type->getSize() == UNKNOWN_LENGTH)
  3920. simple = true;
  3921. break;
  3922. }
  3923. }
  3924. }
  3925. if (simple)
  3926. buildCachedExpr(ctx, expr, tgt);
  3927. else
  3928. buildTempExpr(ctx, expr, tgt);
  3929. }
  3930. IHqlExpression * HqlCppTranslator::buildSimplifyExpr(BuildCtx & ctx, IHqlExpression * expr)
  3931. {
  3932. node_operator op = expr->getOperator();
  3933. switch (op)
  3934. {
  3935. case no_constant:
  3936. case no_all:
  3937. case no_null:
  3938. return LINK(expr);
  3939. case no_list:
  3940. if (expr->isConstant())
  3941. return LINK(expr);
  3942. break;
  3943. }
  3944. CHqlBoundExpr bound;
  3945. buildSimpleExpr(ctx, expr, bound);
  3946. return bound.getTranslatedExpr();
  3947. }
  3948. /* In type: not linked. Return: linked */
  3949. IHqlExpression * HqlCppTranslator::createWrapperTemp(BuildCtx & ctx, ITypeInfo * type, typemod_t modifier)
  3950. {
  3951. Linked<ITypeInfo> rawType = queryUnqualifiedType(type);
  3952. if (hasStreamedModifier(type))
  3953. rawType.setown(setStreamedAttr(rawType, true));
  3954. else if (hasLinkCountedModifier(type))
  3955. rawType.setown(makeAttributeModifier(rawType.getClear(), getLinkCountedAttr()));
  3956. Owned<ITypeInfo> declType = makeWrapperModifier(rawType.getClear());
  3957. declType.setown(makeModifier(declType.getClear(), modifier));
  3958. switch (declType->getTypeCode())
  3959. {
  3960. case type_set: //needed if we have sets with link counted elements
  3961. case type_row:
  3962. case type_array:
  3963. case type_dictionary:
  3964. case type_table:
  3965. case type_groupedtable:
  3966. //Ensure row and dataset temporaries are active throughout a function, so pointers to the row
  3967. //or rows within a dataset remain valid (e.g., outside conditions).
  3968. if ((modifier != typemod_member) && ctx.queryMatchExpr(queryConditionalRowMarker()))
  3969. ctx.setNextPriority(BuildCtx::OutermostScopePrio);
  3970. break;
  3971. }
  3972. return ctx.getTempDeclare(declType, NULL);
  3973. }
  3974. void HqlCppTranslator::createTempFor(BuildCtx & ctx, IHqlExpression * expr, CHqlBoundTarget & target)
  3975. {
  3976. createTempFor(ctx, expr->queryType(), target, typemod_none, FormatNatural);
  3977. }
  3978. void HqlCppTranslator::createTempFor(BuildCtx & ctx, ITypeInfo * _exprType, CHqlBoundTarget & target, typemod_t modifier, ExpressionFormat format)
  3979. {
  3980. Owned<ITypeInfo> exprType = makeModifier(LINK(_exprType->queryPromotedType()), modifier);
  3981. type_t tc = exprType->getTypeCode();
  3982. switch (tc)
  3983. {
  3984. case type_array:
  3985. case type_dictionary:
  3986. case type_table:
  3987. case type_groupedtable:
  3988. {
  3989. if (format == FormatStreamedDataset || hasStreamedModifier(exprType))
  3990. {
  3991. }
  3992. else
  3993. {
  3994. if (recordRequiresLinkCount(::queryRecord(exprType)) || hasLinkCountedModifier(_exprType))
  3995. {
  3996. assertex(format != FormatBlockedDataset);
  3997. format = FormatLinkedDataset;
  3998. }
  3999. else if (format == FormatNatural)
  4000. format = FormatLinkedDataset;
  4001. }
  4002. break;
  4003. }
  4004. }
  4005. switch (format)
  4006. {
  4007. case FormatBlockedDataset:
  4008. exprType.setown(setLinkCountedAttr(exprType, false));
  4009. break;
  4010. case FormatLinkedDataset:
  4011. case FormatArrayDataset:
  4012. exprType.setown(setLinkCountedAttr(exprType, true));
  4013. break;
  4014. case FormatStreamedDataset:
  4015. exprType.setown(setStreamedAttr(exprType, true));
  4016. break;
  4017. case FormatNatural:
  4018. if (hasStreamedModifier(exprType))
  4019. format = FormatStreamedDataset;
  4020. else if (hasLinkCountedModifier(exprType))
  4021. format = FormatLinkedDataset;
  4022. else
  4023. format = FormatBlockedDataset;
  4024. break;
  4025. }
  4026. size32_t size = exprType->getSize();
  4027. if (size == UNKNOWN_LENGTH)
  4028. {
  4029. switch (tc)
  4030. {
  4031. case type_string:
  4032. case type_data:
  4033. case type_qstring:
  4034. case type_unicode:
  4035. case type_utf8:
  4036. {
  4037. OwnedITypeInfo lenType = makeModifier(LINK(sizetType), modifier);
  4038. target.expr.setown(createWrapperTemp(ctx, exprType, modifier));
  4039. target.length.setown(ctx.getTempDeclare(lenType, NULL));
  4040. break;
  4041. }
  4042. case type_varstring:
  4043. case type_varunicode:
  4044. {
  4045. target.expr.setown(createWrapperTemp(ctx, exprType, modifier));
  4046. break;
  4047. }
  4048. case type_set:
  4049. case type_array:
  4050. case type_table:
  4051. case type_groupedtable:
  4052. case type_dictionary:
  4053. case type_row:
  4054. break;
  4055. default:
  4056. {
  4057. UNIMPLEMENTED;
  4058. break;
  4059. }
  4060. }
  4061. }
  4062. else if (size > MAX_SIMPLE_VAR_SIZE)
  4063. {
  4064. switch (tc)
  4065. {
  4066. case type_string:
  4067. case type_data:
  4068. case type_qstring:
  4069. case type_unicode:
  4070. case type_varstring:
  4071. case type_varunicode:
  4072. case type_utf8:
  4073. {
  4074. target.expr.setown(createWrapperTemp(ctx, exprType, modifier));
  4075. break;
  4076. }
  4077. }
  4078. }
  4079. switch (tc)
  4080. {
  4081. case type_set:
  4082. {
  4083. OwnedITypeInfo isAllType = makeModifier(LINK(boolType), modifier);
  4084. target.isAll.setown(ctx.getTempDeclare(isAllType, NULL));
  4085. }
  4086. //fall through
  4087. case type_array:
  4088. case type_table:
  4089. case type_groupedtable:
  4090. case type_dictionary:
  4091. {
  4092. OwnedITypeInfo lenType = makeModifier(LINK(sizetType), modifier);
  4093. target.expr.setown(createWrapperTemp(ctx, exprType, modifier));
  4094. if (!hasStreamedModifier(exprType))
  4095. {
  4096. if (isArrayRowset(exprType))
  4097. {
  4098. //A bit of a hack, but the cleanest I could come up with... really access to the count member should be wrapped in
  4099. //member functions, but getting them created needs a whole new level of complication (probably moving out out of hqlwcpp)
  4100. StringBuffer name;
  4101. target.expr->toString(name).append(".count");
  4102. target.count.setown(createVariable(name, LINK(lenType)));
  4103. }
  4104. else
  4105. target.length.setown(ctx.getTempDeclare(lenType, NULL));
  4106. break;
  4107. }
  4108. }
  4109. }
  4110. if (!target.expr)
  4111. {
  4112. target.expr.setown(ctx.getTempDeclare(exprType, NULL));
  4113. }
  4114. }
  4115. void HqlCppTranslator::buildTempExpr(BuildCtx & ctx, BuildCtx & declareCtx, CHqlBoundTarget & tempTarget, IHqlExpression * expr, ExpressionFormat format, bool ignoreSetAll)
  4116. {
  4117. if (options.addLocationToCpp && !options.obfuscateOutput)
  4118. {
  4119. IHqlExpression * location = queryLocation(expr);
  4120. if (location)
  4121. {
  4122. StringBuffer s;
  4123. s.append("// ").append(str(location->querySourcePath())).append("(").append(location->getStartLine()).append(",").append(location->getStartColumn()).append(") ").append(expr->queryName());
  4124. ctx.addQuoted(s);
  4125. }
  4126. else if (expr->queryName())
  4127. {
  4128. StringBuffer s;
  4129. s.append("// ").append(expr->queryName());
  4130. ctx.addQuoted(s);
  4131. }
  4132. }
  4133. typemod_t modifier = !ctx.isSameLocation(declareCtx) ? typemod_member : typemod_none;
  4134. OwnedITypeInfo type = makeModifier(expr->getType(), modifier);
  4135. BuildCtx subctx(ctx);
  4136. switch (type->getTypeCode())
  4137. {
  4138. case type_row:
  4139. {
  4140. Owned<BoundRow> tempRow = declareTempRow(declareCtx, subctx, expr);
  4141. buildRowAssign(subctx, tempRow, expr);
  4142. tempTarget.expr.set(tempRow->queryBound());
  4143. ctx.associate(*tempRow);
  4144. break;
  4145. }
  4146. break;
  4147. case type_table:
  4148. case type_groupedtable:
  4149. case type_dictionary:
  4150. {
  4151. createTempFor(declareCtx, type, tempTarget, modifier, format);
  4152. TemporaryGroup group(subctx);
  4153. buildDatasetAssign(subctx, tempTarget, expr);
  4154. break;
  4155. }
  4156. default:
  4157. {
  4158. createTempFor(declareCtx, type, tempTarget, modifier, format);
  4159. if (ignoreSetAll)
  4160. tempTarget.isAll.clear();
  4161. TemporaryGroup group(subctx);
  4162. buildExprAssign(subctx, tempTarget, expr);
  4163. break;
  4164. }
  4165. }
  4166. }
  4167. void HqlCppTranslator::buildTempExpr(BuildCtx & ctx, IHqlExpression * expr, CHqlBoundExpr & tgt, ExpressionFormat format)
  4168. {
  4169. node_operator op = expr->getOperator();
  4170. if (op == no_alias)
  4171. {
  4172. doBuildExprAlias(ctx, expr, &tgt, NULL);
  4173. return;
  4174. }
  4175. if (isCast(expr))
  4176. {
  4177. ITypeInfo * exprType = expr->queryType();
  4178. if (exprType->getStringLen() == UNKNOWN_LENGTH)
  4179. {
  4180. unsigned bestLen = getBestLengthEstimate(expr);
  4181. if (bestLen != UNKNOWN_LENGTH)
  4182. {
  4183. IHqlExpression * uncast = expr->queryChild(0);
  4184. Owned<ITypeInfo> stretchedType = getStretchedType(bestLen, exprType);
  4185. OwnedHqlExpr castExpr = ensureExprType(uncast, stretchedType);
  4186. buildTempExpr(ctx, castExpr, tgt, format);
  4187. ctx.associateExpr(expr, tgt);
  4188. return;
  4189. }
  4190. }
  4191. }
  4192. BuildCtx bestctx(ctx);
  4193. if (expr->isPure() && ctx.getMatchExpr(expr, tgt))
  4194. return;
  4195. switch (expr->getOperator())
  4196. {
  4197. case no_variable:
  4198. tgt.expr.set(expr);
  4199. return;
  4200. case no_translated:
  4201. {
  4202. if (!expr->queryChild(1))
  4203. {
  4204. IHqlExpression * value = expr->queryChild(0);
  4205. if (value->getOperator() == no_variable)
  4206. {
  4207. tgt.expr.set(value);
  4208. return;
  4209. }
  4210. }
  4211. break;
  4212. }
  4213. case no_getresult:
  4214. case no_deserialize:
  4215. if (expr->isDatarow())
  4216. {
  4217. buildAnyExpr(ctx, expr, tgt);
  4218. return;
  4219. }
  4220. break;
  4221. case no_id2blob:
  4222. buildExpr(ctx, expr, tgt);
  4223. return;
  4224. case no_call:
  4225. if (!isEmbedCall(expr))
  4226. break;
  4227. //fall through
  4228. case no_externalcall:
  4229. if (format == FormatNatural && expr->isDataset())
  4230. {
  4231. ITypeInfo * exprType = expr->queryType();
  4232. if (hasStreamedModifier(exprType))
  4233. format = FormatStreamedDataset;
  4234. else if (hasLinkCountedModifier(exprType) || hasOutOfLineModifier(exprType))
  4235. format = FormatLinkedDataset;
  4236. else
  4237. format = FormatBlockedDataset;
  4238. }
  4239. break;
  4240. }
  4241. LoopInvariantHelper helper;
  4242. if (options.optimizeLoopInvariant)
  4243. helper.getBestContext(bestctx, expr);
  4244. bool canBeAll = canSetBeAll(expr);
  4245. CHqlBoundTarget tempTarget;
  4246. buildTempExpr(bestctx, bestctx, tempTarget, expr, format, !canBeAll);
  4247. tgt.setFromTarget(tempTarget);
  4248. bestctx.associateExpr(expr, tgt);
  4249. }
  4250. void HqlCppTranslator::buildExprViaTypedTemp(BuildCtx & ctx, IHqlExpression * expr, CHqlBoundExpr & tgt, ITypeInfo * type)
  4251. {
  4252. OwnedHqlExpr cast = createValue(no_implicitcast, LINK(type), LINK(expr));
  4253. if (cast->isPure() && ctx.getMatchExpr(cast, tgt))
  4254. return;
  4255. LoopInvariantHelper helper;
  4256. BuildCtx bestctx(ctx);
  4257. if (options.optimizeLoopInvariant)
  4258. helper.getBestContext(bestctx, expr);
  4259. CHqlBoundTarget tempTarget;
  4260. createTempFor(bestctx, type, tempTarget, typemod_none, FormatNatural);
  4261. buildExprAssign(bestctx, tempTarget, expr);
  4262. tgt.setFromTarget(tempTarget);
  4263. if (cast->isPure())
  4264. bestctx.associateExpr(cast, tgt);
  4265. }
  4266. void HqlCppTranslator::buildExprEnsureType(BuildCtx & ctx, IHqlExpression * expr, CHqlBoundExpr & tgt, ITypeInfo * type)
  4267. {
  4268. if (queryUnqualifiedType(expr->queryType()) != queryUnqualifiedType(type))
  4269. buildExprViaTypedTemp(ctx, expr, tgt, type);
  4270. else
  4271. buildExpr(ctx, expr, tgt);
  4272. }
  4273. AliasKind HqlCppTranslator::doBuildAliasValue(BuildCtx & ctx, IHqlExpression * value, CHqlBoundExpr & tgt, AliasExpansionInfo * parentInfo)
  4274. {
  4275. //can happen when this is called for non no_alias arguments
  4276. if (value->getOperator() == no_alias)
  4277. value = value->queryChild(0);
  4278. EvalContext * instance = queryEvalContext(ctx);
  4279. if (instance)
  4280. return instance->evaluateExpression(ctx, value, tgt, true);
  4281. expandAliases(ctx, value, parentInfo);
  4282. buildTempExpr(ctx, value, tgt);
  4283. return RuntimeAlias;
  4284. }
  4285. void HqlCppTranslator::doBuildExprAlias(BuildCtx & ctx, IHqlExpression * expr, CHqlBoundExpr * tgt, AliasExpansionInfo * parentInfo)
  4286. {
  4287. //MORE These will be declared in a different context later.
  4288. IHqlExpression * value = expr->queryChild(0);
  4289. while (value->getOperator() == no_alias)
  4290. value = value->queryChild(0);
  4291. CHqlBoundExpr bound;
  4292. if (!tgt)
  4293. tgt = &bound;
  4294. //The second half of this test could cause aliases to be duplicated, but has the significant effect of reducing the amount of data that is serialised.
  4295. //so far on my examples it does the latter, but doesn't seem to cause the former
  4296. if (expr->hasAttribute(localAtom) || (insideOnCreate(ctx) && !expr->hasAttribute(globalAtom)))
  4297. {
  4298. expandAliases(ctx, value, parentInfo);
  4299. switch (value->getOperator())
  4300. {
  4301. // these operations generate temporaries anyway, and the row versions are inefficient via a buildTempExpr
  4302. case no_getresult:
  4303. case no_getgraphresult:
  4304. case no_getgraphloopresult:
  4305. buildAnyExpr(ctx, value, *tgt);
  4306. break;
  4307. default:
  4308. buildTempExpr(ctx, value, *tgt);
  4309. break;
  4310. }
  4311. }
  4312. else
  4313. {
  4314. doBuildAliasValue(ctx, value, *tgt, parentInfo);
  4315. }
  4316. }
  4317. void HqlCppTranslator::doBuildBoolAssign(BuildCtx & ctx, const CHqlBoundTarget & target, IHqlExpression * expr)
  4318. {
  4319. if (requiresTemp(ctx, expr, true))
  4320. {
  4321. BuildCtx subctx(ctx);
  4322. assignBound(subctx, target, queryBoolExpr(false));
  4323. buildFilter(subctx, expr);
  4324. assignBound(subctx, target, queryBoolExpr(true));
  4325. }
  4326. else
  4327. {
  4328. CHqlBoundExpr temp;
  4329. buildCachedExpr(ctx, expr, temp);
  4330. assign(ctx, target, temp);
  4331. }
  4332. }
  4333. void HqlCppTranslator::doBuildExprAssign(BuildCtx & ctx, const CHqlBoundTarget & target, IHqlExpression * expr)
  4334. {
  4335. CHqlBoundExpr temp;
  4336. buildExpr(ctx, expr, temp);
  4337. assign(ctx, target, temp);
  4338. }
  4339. void HqlCppTranslator::ensureSimpleExpr(BuildCtx & ctx, CHqlBoundExpr & tgt)
  4340. {
  4341. if (!isSimpleTranslatedExpr(tgt.expr))
  4342. {
  4343. OwnedHqlExpr bound = tgt.getTranslatedExpr();
  4344. buildTempExpr(ctx, bound, tgt);
  4345. }
  4346. }
  4347. IHqlExpression * HqlCppTranslator::ensureSimpleTranslatedExpr(BuildCtx & ctx, IHqlExpression * expr)
  4348. {
  4349. if (isSimpleTranslatedExpr(expr))
  4350. return LINK(expr);
  4351. OwnedHqlExpr translated = createTranslated(expr);
  4352. CHqlBoundExpr bound;
  4353. buildTempExpr(ctx, translated, bound);
  4354. return LINK(bound.expr);
  4355. }
  4356. void HqlCppTranslator::ensureHasAddress(BuildCtx & ctx, CHqlBoundExpr & tgt)
  4357. {
  4358. IHqlExpression * expr = tgt.expr;
  4359. node_operator op = expr->getOperator();
  4360. switch (op)
  4361. {
  4362. case no_deref:
  4363. case no_variable:
  4364. break;
  4365. default:
  4366. if (!isTypePassedByAddress(expr->queryType()) || (expr->getOperator() == no_decimalstack))
  4367. {
  4368. OwnedHqlExpr bound = tgt.getTranslatedExpr();
  4369. buildTempExpr(ctx, bound, tgt);
  4370. }
  4371. break;
  4372. }
  4373. }
  4374. //---------------------------------------------------------------------------
  4375. bool optimizeVarStringCompare(node_operator op, const CHqlBoundExpr & lhs, const CHqlBoundExpr & rhs, CHqlBoundExpr & tgt)
  4376. {
  4377. IHqlExpression * rhsExpr = rhs.expr;
  4378. if ((rhsExpr->getOperator() == no_constant) && (rhsExpr->queryType()->getStringLen() == 0))
  4379. {
  4380. if ((op == no_eq) || (op == no_ne))
  4381. {
  4382. tgt.expr.setown(createValue(op, LINK(boolType), createValue(no_deref, makeCharType(), lhs.expr.getLink()), getZero()));
  4383. return true;
  4384. }
  4385. }
  4386. return false;
  4387. }
  4388. void HqlCppTranslator::doBuildExprSetCompareAll(BuildCtx & ctx, IHqlExpression * set, CHqlBoundExpr & tgt, bool invert)
  4389. {
  4390. Owned<IHqlCppSetCursor> cursor = createSetSelector(ctx, set);
  4391. cursor->buildIsAll(ctx, tgt);
  4392. if (invert)
  4393. tgt.expr.setown(getInverse(tgt.expr));
  4394. }
  4395. void HqlCppTranslator::doBuildExprSetCompareNone(BuildCtx & ctx, IHqlExpression * set, CHqlBoundExpr & tgt, bool invert)
  4396. {
  4397. Owned<IHqlCppSetCursor> cursor = createSetSelector(ctx, set);
  4398. cursor->buildExists(ctx, tgt);
  4399. if (!invert)
  4400. tgt.expr.setown(getInverse(tgt.expr));
  4401. }
  4402. bool HqlCppTranslator::doBuildExprSetCompare(BuildCtx & ctx, IHqlExpression * expr, CHqlBoundExpr & tgt)
  4403. {
  4404. //Special case comparison against all and null set. All other work goes through a the order code.
  4405. node_operator exprOp = expr->getOperator();
  4406. if ((exprOp == no_eq) || (exprOp == no_ne))
  4407. {
  4408. OwnedHqlExpr left = normalizeListCasts(expr->queryChild(0));
  4409. OwnedHqlExpr right = normalizeListCasts(expr->queryChild(1));
  4410. if (right->getOperator() == no_all)
  4411. doBuildExprSetCompareAll(ctx, left, tgt, exprOp==no_ne);
  4412. else if (left->getOperator() == no_all)
  4413. doBuildExprSetCompareAll(ctx, right, tgt, exprOp==no_ne);
  4414. else if (isNullList(right))
  4415. doBuildExprSetCompareNone(ctx, left, tgt, exprOp==no_ne);
  4416. else if (isNullList(left))
  4417. doBuildExprSetCompareNone(ctx, right, tgt, exprOp==no_ne);
  4418. else
  4419. return false;
  4420. return true;
  4421. }
  4422. return false;
  4423. }
  4424. IHqlExpression * HqlCppTranslator::convertBoundStringToChar(const CHqlBoundExpr & bound)
  4425. {
  4426. OwnedHqlExpr element = getElementPointer(bound.expr);
  4427. Owned<ITypeInfo> charType = makeCharType(true);
  4428. switch (element->getOperator())
  4429. {
  4430. case no_constant:
  4431. {
  4432. IValue * value = element->queryValue();
  4433. return createConstant(value->castTo(charType));
  4434. }
  4435. case no_address:
  4436. return LINK(element->queryChild(0));
  4437. }
  4438. return createValue(no_deref, charType.getClear(), element.getClear());
  4439. }
  4440. void HqlCppTranslator::doBuildExprCompare(BuildCtx & ctx, IHqlExpression * expr, CHqlBoundExpr & tgt)
  4441. {
  4442. IHqlExpression * left = expr->queryChild(0);
  4443. IHqlExpression * right = expr->queryChild(1);
  4444. ITypeInfo * leftType = left->queryType()->queryPromotedType();
  4445. ITypeInfo * rightType = right->queryType()->queryPromotedType();
  4446. assertex(areTypesComparable(leftType,rightType));
  4447. OwnedHqlExpr orderExpr;
  4448. CHqlBoundExpr lhs, rhs;
  4449. node_operator compareOp = expr->getOperator();
  4450. type_t tc = leftType->getTypeCode();
  4451. switch (tc)
  4452. {
  4453. case type_string:
  4454. case type_data:
  4455. case type_qstring:
  4456. {
  4457. OwnedHqlExpr simpleLeft = getSimplifyCompareArg(left);
  4458. OwnedHqlExpr simpleRight = getSimplifyCompareArg(right);
  4459. HqlExprArray args;
  4460. buildCachedExpr(ctx, simpleLeft, lhs);
  4461. buildCachedExpr(ctx, simpleRight, rhs);
  4462. //update types - lengths may be constant by now..
  4463. leftType = lhs.queryType();
  4464. rightType = rhs.queryType();
  4465. IIdAtom * func = queryStrCompareFunc(leftType);
  4466. //MORE: Move blank string compare here?
  4467. if (lhs.length || rhs.length || needVarStringCompare(leftType, rightType))
  4468. {
  4469. args.append(*getBoundLength(lhs));
  4470. args.append(*getElementPointer(lhs.expr));
  4471. if (func == compareStrStrId && isBlankString(rhs.expr))
  4472. {
  4473. func = compareStrBlankId;
  4474. }
  4475. else
  4476. {
  4477. args.append(*getBoundLength(rhs));
  4478. args.append(*getElementPointer(rhs.expr));
  4479. }
  4480. orderExpr.setown(bindTranslatedFunctionCall(func, args));
  4481. }
  4482. else if (options.optimizeString1Compare &&
  4483. ((tc == type_string && isAscii(leftType)) || tc == type_data) &&
  4484. (leftType->getSize() == 1) && (rightType->getSize() == 1) &&
  4485. ((compareOp == no_eq) || (compareOp == no_ne)))
  4486. {
  4487. //Optimize equality/non equality of a single character string.
  4488. //Not done for > etc because of potential issues with signed/unsigned chars
  4489. args.append(*convertBoundStringToChar(lhs));
  4490. args.append(*convertBoundStringToChar(rhs));
  4491. tgt.expr.setown(createValue(compareOp, makeBoolType(), args));
  4492. return;
  4493. }
  4494. else
  4495. {
  4496. args.append(*getElementPointer(lhs.expr));
  4497. args.append(*getElementPointer(rhs.expr));
  4498. args.append(*getSizetConstant(leftType->getSize()));
  4499. orderExpr.setown(bindTranslatedFunctionCall(memcmpId, args));
  4500. }
  4501. break;
  4502. }
  4503. case type_unicode:
  4504. {
  4505. OwnedHqlExpr simpleLeft = LINK(left);//getSimplifyCompareArg(left);
  4506. OwnedHqlExpr simpleRight = LINK(right);//getSimplifyCompareArg(right);
  4507. HqlExprArray args;
  4508. buildCachedExpr(ctx, simpleLeft, lhs);
  4509. buildCachedExpr(ctx, simpleRight, rhs);
  4510. assertex(haveCommonLocale(leftType, rightType));
  4511. char const * locale = str(getCommonLocale(leftType, rightType));
  4512. args.append(*getBoundLength(lhs));
  4513. args.append(*getElementPointer(lhs.expr));
  4514. args.append(*getBoundLength(rhs));
  4515. args.append(*getElementPointer(rhs.expr));
  4516. args.append(*createConstant(locale));
  4517. orderExpr.setown(bindTranslatedFunctionCall(compareUnicodeUnicodeId, args));
  4518. break;
  4519. }
  4520. case type_varunicode:
  4521. {
  4522. HqlExprArray args;
  4523. buildCachedExpr(ctx, left, lhs);
  4524. buildCachedExpr(ctx, right, rhs);
  4525. assertex(haveCommonLocale(leftType, rightType));
  4526. char const * locale = str(getCommonLocale(leftType, rightType));
  4527. args.append(*getElementPointer(lhs.expr));
  4528. args.append(*getElementPointer(rhs.expr));
  4529. args.append(*createConstant(locale));
  4530. orderExpr.setown(bindTranslatedFunctionCall(compareVUnicodeVUnicodeId, args));
  4531. break;
  4532. }
  4533. case type_utf8:
  4534. {
  4535. OwnedHqlExpr simpleLeft = LINK(left);//getSimplifyCompareArg(left);
  4536. OwnedHqlExpr simpleRight = LINK(right);//getSimplifyCompareArg(right);
  4537. HqlExprArray args;
  4538. buildCachedExpr(ctx, simpleLeft, lhs);
  4539. buildCachedExpr(ctx, simpleRight, rhs);
  4540. assertex(haveCommonLocale(leftType, rightType));
  4541. char const * locale = str(getCommonLocale(leftType, rightType));
  4542. args.append(*getBoundLength(lhs));
  4543. args.append(*getElementPointer(lhs.expr));
  4544. args.append(*getBoundLength(rhs));
  4545. args.append(*getElementPointer(rhs.expr));
  4546. args.append(*createConstant(locale));
  4547. orderExpr.setown(bindTranslatedFunctionCall(compareUtf8Utf8Id, args));
  4548. break;
  4549. }
  4550. case type_varstring:
  4551. {
  4552. HqlExprArray args;
  4553. buildCachedExpr(ctx, left, lhs);
  4554. buildCachedExpr(ctx, right, rhs);
  4555. //optimize comparison against null string.
  4556. if (optimizeVarStringCompare(compareOp, lhs, rhs, tgt) || optimizeVarStringCompare(compareOp, rhs, lhs, tgt))
  4557. return;
  4558. args.append(*getElementPointer(lhs.expr));
  4559. args.append(*getElementPointer(rhs.expr));
  4560. orderExpr.setown(bindTranslatedFunctionCall(compareVStrVStrId, args));
  4561. break;
  4562. }
  4563. case type_decimal:
  4564. {
  4565. HqlExprArray args;
  4566. buildCachedExpr(ctx, left, lhs);
  4567. buildCachedExpr(ctx, right, rhs);
  4568. if (!isPushed(lhs) && !isPushed(rhs) && isSameBasicType(lhs.queryType(), rhs.queryType()))
  4569. {
  4570. args.append(*getSizetConstant(leftType->getSize()));
  4571. args.append(*getPointer(lhs.expr));
  4572. args.append(*getPointer(rhs.expr));
  4573. orderExpr.setown(bindTranslatedFunctionCall(leftType->isSigned() ? DecCompareDecimalId : DecCompareUDecimalId, args));
  4574. }
  4575. else
  4576. {
  4577. bool pushedLhs = ensurePushed(ctx, lhs);
  4578. bool pushedRhs = ensurePushed(ctx, rhs);
  4579. //NB: Arguments could be pushed in opposite order 1 <=> x *2
  4580. if (pushedLhs && !pushedRhs)
  4581. orderExpr.setown(bindTranslatedFunctionCall(DecDistinctRId, args));
  4582. else
  4583. orderExpr.setown(bindTranslatedFunctionCall(DecDistinctId, args));
  4584. }
  4585. break;
  4586. }
  4587. case type_set:
  4588. case type_array:
  4589. {
  4590. if (doBuildExprSetCompare(ctx, expr, tgt))
  4591. return;
  4592. //fallthrough....
  4593. }
  4594. case type_dictionary:
  4595. case type_table:
  4596. case type_groupedtable:
  4597. case type_row:
  4598. case type_record:
  4599. {
  4600. IHqlExpression * orderExpr = createValue(no_order, LINK(signedType), LINK(left), LINK(right));
  4601. OwnedHqlExpr cmpExpr = createBoolExpr(compareOp, orderExpr, createConstant(signedType->castFrom(true, 0)));
  4602. buildExpr(ctx, cmpExpr, tgt);
  4603. return;
  4604. }
  4605. case type_swapint:
  4606. case type_packedint:
  4607. {
  4608. Owned<ITypeInfo> type = makeIntType(leftType->getSize(), leftType->isSigned());
  4609. IHqlExpression * intLeft = createValue(no_implicitcast, type.getLink(), LINK(left));
  4610. IHqlExpression * intRight = createValue(no_implicitcast, type.getLink(), LINK(right));
  4611. OwnedHqlExpr transformed = createValue(compareOp, makeBoolType(), intLeft, intRight);
  4612. doBuildPureSubExpr(ctx, transformed, tgt);
  4613. return;
  4614. }
  4615. default:
  4616. doBuildPureSubExpr(ctx, expr, tgt);
  4617. return;
  4618. }
  4619. tgt.expr.setown(createValue(compareOp, LINK(boolType), LINK(orderExpr), createConstant(orderExpr->queryType()->castFrom(true, 0))));
  4620. }
  4621. //---------------------------------------------------------------------------
  4622. //-- no_and --
  4623. void HqlCppTranslator::doBuildFilterToTarget(BuildCtx & ctx, const CHqlBoundTarget & isOk, HqlExprArray & conds, bool invert)
  4624. {
  4625. LinkedHqlExpr test = isOk.expr;
  4626. if (invert)
  4627. test.setown(getInverse(test));
  4628. unsigned max = conds.ordinality();
  4629. unsigned curIndex = 0;
  4630. for (unsigned outer =0; curIndex < max; outer += MAX_NESTED_CASES)
  4631. {
  4632. BuildCtx subctx(ctx);
  4633. if (outer != 0)
  4634. subctx.addFilter(test);
  4635. buildExprAssign(subctx, isOk, queryBoolExpr(invert)); // if(!invert,false,true) => invert
  4636. doBuildFilterNextAndRange(subctx, curIndex, MAX_NESTED_CASES, conds);
  4637. buildExprAssign(subctx, isOk, queryBoolExpr(!invert)); // if(!invert,true,false) => !invert
  4638. }
  4639. }
  4640. void HqlCppTranslator::doBuildFilterAnd(BuildCtx & ctx, IHqlExpression * expr)
  4641. {
  4642. HqlExprArray conds;
  4643. expr->unwindList(conds, no_and);
  4644. //Estimate the depth generated by the conditions - it may be wrong because
  4645. //aliases generated in outer levels can stop temporaries being required later.
  4646. unsigned numConds = conds.ordinality();
  4647. unsigned depthEstimate = 1;
  4648. for (unsigned i=1; i < numConds; i++)
  4649. if (requiresTemp(ctx, &conds.item(i), true))
  4650. depthEstimate++;
  4651. if (depthEstimate < MAX_NESTED_CASES)
  4652. {
  4653. unsigned curIndex = 0;
  4654. doBuildFilterNextAndRange(ctx, curIndex, numConds, conds);
  4655. }
  4656. else
  4657. {
  4658. CHqlBoundTarget isOk;
  4659. createTempFor(ctx, expr, isOk);
  4660. doBuildFilterToTarget(ctx, isOk, conds, false);
  4661. ctx.addFilter(isOk.expr);
  4662. }
  4663. }
  4664. void HqlCppTranslator::doBuildFilterNextAndRange(BuildCtx & ctx, unsigned & curIndex, unsigned maxIterations, HqlExprArray & conds)
  4665. {
  4666. unsigned max = conds.ordinality();
  4667. for (unsigned i=0; (i < maxIterations) && (curIndex != max); i++)
  4668. {
  4669. unsigned last;
  4670. expandAliases(ctx, &conds.item(curIndex), NULL);
  4671. for (last = curIndex+1; last < max; last++)
  4672. if (requiresTemp(ctx, &conds.item(last), true))
  4673. break;
  4674. doBuildFilterAndRange(ctx, curIndex, last, conds);
  4675. curIndex = last;
  4676. }
  4677. }
  4678. void HqlCppTranslator::doBuildFilterAndRange(BuildCtx & ctx, unsigned first, unsigned last, HqlExprArray & conds)
  4679. {
  4680. if (first+1 == last)
  4681. buildFilter(ctx, &conds.item(first));
  4682. else
  4683. {
  4684. HqlExprArray args;
  4685. for (unsigned k = first; k < last; k++)
  4686. args.append(OLINK(conds.item(k)));
  4687. OwnedHqlExpr test = createValue(no_and, makeBoolType(), args);
  4688. buildFilterViaExpr(ctx, test);
  4689. }
  4690. }
  4691. void HqlCppTranslator::doBuildAssignAnd(BuildCtx & ctx, const CHqlBoundTarget & target, IHqlExpression * expr, bool invert)
  4692. {
  4693. HqlExprArray conds;
  4694. expr->unwindList(conds, no_and);
  4695. doBuildFilterToTarget(ctx, target, conds, invert);
  4696. }
  4697. void HqlCppTranslator::doBuildAssignOr(BuildCtx & ctx, const CHqlBoundTarget & target, IHqlExpression * expr)
  4698. {
  4699. BuildCtx subctx(ctx);
  4700. HqlExprArray conds;
  4701. expr->unwindList(conds, no_or);
  4702. unsigned first = 0;
  4703. unsigned max = conds.ordinality();
  4704. while (first < max)
  4705. {
  4706. unsigned last = first+1;
  4707. //special case no_in because it always creates a temporary, so may as well assign on its own
  4708. if (conds.item(first).getOperator() != no_in)
  4709. {
  4710. for (; last < max; last++)
  4711. {
  4712. IHqlExpression & cur = conds.item(last);
  4713. if ((cur.getOperator() == no_in) || requiresTemp(subctx, &cur, true))
  4714. break;
  4715. }
  4716. }
  4717. if (first != 0 || last != max)
  4718. {
  4719. if (last != first+1)
  4720. {
  4721. OwnedHqlExpr left = createBalanced(no_or, queryBoolType(), conds, first, last);
  4722. doBuildExprAssign(subctx, target, left);
  4723. }
  4724. else
  4725. buildExprAssign(subctx, target, &conds.item(first));
  4726. if (last != max)
  4727. {
  4728. OwnedHqlExpr inverse = getInverse(target.expr);
  4729. subctx.addFilter(inverse);
  4730. }
  4731. }
  4732. else
  4733. {
  4734. doBuildExprAssign(subctx, target, expr);
  4735. }
  4736. first = last;
  4737. }
  4738. }
  4739. //---------------------------------------------------------------------------
  4740. //-- no_case --
  4741. void HqlCppTranslator::doBuildCaseInfo(IHqlExpression * expr, HqlCppCaseInfo & info)
  4742. {
  4743. unsigned maxMaps = expr->numChildren()-1;
  4744. unsigned index = 0;
  4745. if (expr->getOperator() == no_case)
  4746. {
  4747. info.setCond(expr->queryChild(0));
  4748. index++;
  4749. }
  4750. for (;index<maxMaps;index++)
  4751. info.addPair(expr->queryChild(index));
  4752. info.setDefault(expr->queryChild(maxMaps));
  4753. }
  4754. void HqlCppTranslator::doBuildInCaseInfo(IHqlExpression * expr, HqlCppCaseInfo & info, IHqlExpression * normalized)
  4755. {
  4756. bool valueIfMatch = (expr->getOperator() == no_in);
  4757. HqlExprArray args;
  4758. LinkedHqlExpr values = normalized;
  4759. if (!normalized)
  4760. values.setown(normalizeListCasts(expr->queryChild(1)));
  4761. info.setCond(expr->queryChild(0));
  4762. cvtInListToPairs(args, values, valueIfMatch);
  4763. info.addPairs(args);
  4764. info.setDefault(queryBoolExpr(!valueIfMatch));
  4765. }
  4766. void HqlCppTranslator::doBuildAssignInStored(BuildCtx & ctx, const CHqlBoundTarget & target, IHqlExpression * expr)
  4767. {
  4768. OwnedHqlExpr values = normalizeListCasts(expr->queryChild(1));
  4769. CHqlBoundExpr bound;
  4770. if (values->isPure())
  4771. ctx.getMatchExpr(values, bound);
  4772. if (options.optimizeLoopInvariant && !bound.expr)
  4773. {
  4774. if (values->getOperator() == no_createset)
  4775. {
  4776. IHqlExpression * ds = values->queryChild(0);
  4777. //MORE: This is a special case - it should check if the dataset can be iterated without any projection
  4778. switch (ds->getOperator())
  4779. {
  4780. case no_select:
  4781. case no_rows:
  4782. break;
  4783. default:
  4784. {
  4785. //Evaluate the dataset into a temporary - the iterator will match it up later
  4786. //Evaluating the dataset instead of the set since this is likely to avoid a memcpy with link counted rows enabled.
  4787. LoopInvariantHelper helper;
  4788. BuildCtx bestctx(ctx);
  4789. if (helper.getBestContext(bestctx, ds))
  4790. {
  4791. CHqlBoundExpr temp;
  4792. buildTempExpr(ctx, ds, temp);
  4793. }
  4794. break;
  4795. }
  4796. }
  4797. }
  4798. else
  4799. {
  4800. LoopInvariantHelper helper;
  4801. BuildCtx bestctx(ctx);
  4802. if (helper.getBestContext(bestctx, values))
  4803. {
  4804. //Unfortunately this will do strength reduction again, but shouldn't take too long.
  4805. buildTempExpr(bestctx, values, bound);
  4806. }
  4807. }
  4808. }
  4809. if (bound.expr)
  4810. values.setown(bound.getTranslatedExpr());
  4811. bool valueIfMatch = (expr->getOperator() == no_in);
  4812. ITypeInfo * elementType = values->queryType()->queryChildType();
  4813. Owned<ITypeInfo> compareType = getPromotedECLCompareType(expr->queryChild(0)->queryType(), elementType);
  4814. CHqlBoundExpr boundSearch;
  4815. OwnedHqlExpr castSearch = ensureExprType(expr->queryChild(0), compareType);
  4816. BuildCtx subctx(ctx);
  4817. Owned<IHqlCppSetCursor> cursor = createSetSelector(ctx, values);
  4818. CHqlBoundExpr isAll;
  4819. cursor->buildIsAll(ctx, isAll);
  4820. if (isAll.expr->queryValue())
  4821. {
  4822. if (isAll.expr->queryValue()->getBoolValue())
  4823. {
  4824. buildExprAssign(subctx, target, queryBoolExpr(valueIfMatch));
  4825. //If this inverted the test we would need to do more.
  4826. return;
  4827. }
  4828. }
  4829. else
  4830. {
  4831. IHqlStmt * stmt = subctx.addFilter(isAll.expr);
  4832. buildExprAssign(subctx, target, queryBoolExpr(valueIfMatch));
  4833. subctx.selectElse(stmt);
  4834. }
  4835. //result = false/true
  4836. buildSimpleExpr(subctx, castSearch, boundSearch);
  4837. buildExprAssign(subctx, target, queryBoolExpr(!valueIfMatch));
  4838. //iterate through the set
  4839. bool needToBreak = !cursor->isSingleValued();
  4840. CHqlBoundExpr boundCurElement;
  4841. cursor->buildIterateLoop(subctx, boundCurElement, needToBreak);
  4842. OwnedHqlExpr curElement = boundCurElement.getTranslatedExpr();
  4843. OwnedHqlExpr castCurElement = ensureExprType(curElement, compareType);
  4844. //if match then
  4845. OwnedHqlExpr matchTest = createValue(no_eq, makeBoolType(), boundSearch.getTranslatedExpr(), LINK(castCurElement));
  4846. buildFilter(subctx, matchTest);
  4847. //result = true/false + break loop.
  4848. buildExprAssign(subctx, target, queryBoolExpr(valueIfMatch));
  4849. if (needToBreak)
  4850. subctx.addBreak();
  4851. }
  4852. void HqlCppTranslator::doBuildAssignInCreateSet(BuildCtx & ctx, const CHqlBoundTarget & target, IHqlExpression * expr)
  4853. {
  4854. IHqlExpression * setExpr = expr->queryChild(1);
  4855. IHqlExpression * dataset = setExpr->queryChild(0);
  4856. IHqlExpression * selected = setExpr->queryChild(1);
  4857. bool valueIfMatch = (expr->getOperator() == no_in);
  4858. ITypeInfo * elementType = setExpr->queryType()->queryChildType();
  4859. Owned<ITypeInfo> compareType = getPromotedECLCompareType(expr->queryChild(0)->queryType(), elementType);
  4860. CHqlBoundExpr boundSearch;
  4861. OwnedHqlExpr castSearch = ensureExprType(expr->queryChild(0), compareType);
  4862. //result = false/true
  4863. buildSimpleExpr(ctx, castSearch, boundSearch);
  4864. buildExprAssign(ctx, target, queryBoolExpr(!valueIfMatch));
  4865. //iterate through the set
  4866. bool needToBreak = !hasNoMoreRowsThan(dataset, 1);
  4867. BuildCtx loopctx(ctx);
  4868. buildDatasetIterate(loopctx, dataset, needToBreak);
  4869. //if match then
  4870. OwnedHqlExpr matchTest = createValue(no_eq, makeBoolType(), boundSearch.getTranslatedExpr(), ensureExprType(selected, compareType));
  4871. buildFilter(loopctx, matchTest);
  4872. //result = true/false + break loop.
  4873. buildExprAssign(loopctx, target, queryBoolExpr(valueIfMatch));
  4874. if (needToBreak)
  4875. loopctx.addBreak();
  4876. }
  4877. void HqlCppTranslator::doBuildAssignIn(BuildCtx & ctx, const CHqlBoundTarget & target, IHqlExpression * expr)
  4878. {
  4879. OwnedHqlExpr values = normalizeListCasts(expr->queryChild(1));
  4880. node_operator op = expr->getOperator();
  4881. bool valueIfMatch = (op == no_in);
  4882. switch (values->getOperator())
  4883. {
  4884. case no_all:
  4885. buildExprAssign(ctx, target, queryBoolExpr(valueIfMatch));
  4886. break;
  4887. case no_null:
  4888. buildExprAssign(ctx, target, queryBoolExpr(!valueIfMatch));
  4889. break;
  4890. case no_list:
  4891. {
  4892. HqlCppCaseInfo info(*this);
  4893. doBuildInCaseInfo(expr, info, values);
  4894. info.buildAssign(ctx, target);
  4895. break;
  4896. }
  4897. case no_createset:
  4898. {
  4899. //Look further at bug #52745.eclxml and what causes the poor code.
  4900. //if (canIterateInline(&ctx, values->queryChild(0)))
  4901. // doBuildAssignInCreateSet(ctx, target, expr);
  4902. //else
  4903. doBuildAssignInStored(ctx, target, expr);
  4904. break;
  4905. }
  4906. #if 0
  4907. //Possible optimizations, but I don't have any examples that would trigger them, so don't enable
  4908. case no_if:
  4909. {
  4910. BuildCtx subctx(ctx);
  4911. OwnedHqlExpr inLhs = createValue(expr->getOperator(), LINK(expr->queryChild(0)), LINK(values->queryChild(1)));
  4912. OwnedHqlExpr inRhs = createValue(expr->getOperator(), LINK(expr->queryChild(0)), LINK(values->queryChild(2)));
  4913. IHqlStmt * stmt = buildFilterViaExpr(subctx, values->queryChild(0));
  4914. buildExprAssign(subctx, target, inLhs);
  4915. subctx.selectElse(stmt);
  4916. buildExprAssign(subctx, target, inRhs);
  4917. break;
  4918. }
  4919. case no_addsets:
  4920. if (op == no_in)
  4921. {
  4922. BuildCtx subctx(ctx);
  4923. OwnedHqlExpr inLhs = createValue(expr->getOperator(), LINK(expr->queryChild(0)), LINK(values->queryChild(0)));
  4924. OwnedHqlExpr inRhs = createValue(expr->getOperator(), LINK(expr->queryChild(0)), LINK(values->queryChild(1)));
  4925. buildExprAssign(ctx, target, inLhs);
  4926. OwnedHqlExpr test = getInverse(target.expr);
  4927. subctx.addFilter(test);
  4928. buildExprAssign(subctx, target, inRhs);
  4929. break;
  4930. }
  4931. #endif
  4932. //fall through
  4933. default:
  4934. doBuildAssignInStored(ctx, target, expr);
  4935. break;
  4936. }
  4937. }
  4938. //---------------------------------------------------------------------------
  4939. void HqlCppTranslator::doBuildExprInDict(BuildCtx & ctx, IHqlExpression * expr, CHqlBoundExpr & tgt)
  4940. {
  4941. IHqlExpression *dict = expr->queryChild(1);
  4942. Owned<IHqlCppDatasetCursor> cursor = createDatasetSelector(ctx, dict);
  4943. cursor->buildInDataset(ctx, expr, tgt);
  4944. }
  4945. void HqlCppTranslator::doBuildExprCountDict(BuildCtx & ctx, IHqlExpression * expr, CHqlBoundExpr & tgt)
  4946. {
  4947. IHqlExpression *dict = expr->queryChild(0);
  4948. Owned<IHqlCppDatasetCursor> cursor = createDatasetSelector(ctx, dict);
  4949. cursor->buildCountDict(ctx, tgt); // not the same as buildCount - that is the size of the table, we want the number of populated entries
  4950. }
  4951. void HqlCppTranslator::doBuildExprExistsDict(BuildCtx & ctx, IHqlExpression * expr, CHqlBoundExpr & tgt)
  4952. {
  4953. IHqlExpression *dict = expr->queryChild(0);
  4954. Owned<IHqlCppDatasetCursor> cursor = createDatasetSelector(ctx, dict);
  4955. cursor->buildExistsDict(ctx, tgt);
  4956. }
  4957. //---------------------------------------------------------------------------
  4958. void HqlCppTranslator::doBuildExprArith(BuildCtx & ctx, IHqlExpression * expr, CHqlBoundExpr & tgt)
  4959. {
  4960. ITypeInfo * type = expr->queryType();
  4961. switch (type->getTypeCode())
  4962. {
  4963. case type_decimal:
  4964. {
  4965. bindAndPush(ctx, expr->queryChild(0));
  4966. bindAndPush(ctx, expr->queryChild(1));
  4967. HqlExprArray args;
  4968. IIdAtom * func;
  4969. switch (expr->getOperator())
  4970. {
  4971. case no_add:
  4972. func = DecAddId;
  4973. break;
  4974. case no_div:
  4975. func = DecDivideId;
  4976. args.append(*getSizetConstant(options.divideByZeroAction));
  4977. break;
  4978. case no_modulus:
  4979. func = DecModulusId;
  4980. args.append(*getSizetConstant(options.divideByZeroAction));
  4981. break;
  4982. case no_mul:
  4983. func = DecMulId;
  4984. break;
  4985. case no_sub:
  4986. func = DecSubId;
  4987. break;
  4988. default: UNIMPLEMENTED;
  4989. }
  4990. callProcedure(ctx, func, args);
  4991. tgt.expr.setown(createValue(no_decimalstack, LINK(type)));
  4992. }
  4993. break;
  4994. case type_swapint:
  4995. case type_packedint:
  4996. {
  4997. //someone is being deliberately perverse....
  4998. ITypeInfo * type = expr->queryType();
  4999. ITypeInfo * intType = makeIntType(type->getSize(), type->isSigned());
  5000. IHqlExpression * lhs = expr->queryChild(0);
  5001. IHqlExpression * rhs = expr->queryChild(1);
  5002. assertex(isSameBasicType(type, lhs->queryType()));
  5003. assertex(isSameBasicType(type, rhs->queryType()));
  5004. lhs = createValue(no_implicitcast, LINK(intType), LINK(lhs));
  5005. rhs = createValue(no_implicitcast, LINK(intType), LINK(rhs));
  5006. IHqlExpression * newExpr = createValue(expr->getOperator(), intType, lhs, rhs);
  5007. OwnedHqlExpr castNewExpr = createValue(no_implicitcast, LINK(type), newExpr);
  5008. buildExpr(ctx, castNewExpr, tgt);
  5009. break;
  5010. }
  5011. default:
  5012. doBuildPureSubExpr(ctx, expr, tgt);
  5013. break;
  5014. }
  5015. }
  5016. void HqlCppTranslator::doBuildExprAdd(BuildCtx & ctx, IHqlExpression * expr, CHqlBoundExpr & tgt)
  5017. {
  5018. if (expr->queryType()->getTypeCode() != type_int)
  5019. {
  5020. doBuildExprArith(ctx, expr, tgt);
  5021. return;
  5022. }
  5023. CHqlBoundExpr boundL, boundR;
  5024. buildExpr(ctx, expr->queryChild(0), boundL);
  5025. bool WORK_AROUND_GCC_CONDITION_BUG = (options.targetCompiler == GccCppCompiler);
  5026. if (WORK_AROUND_GCC_CONDITION_BUG && expr->queryChild(1)->getOperator() == no_if)
  5027. buildTempExpr(ctx, expr->queryChild(1), boundR);
  5028. else
  5029. buildExpr(ctx, expr->queryChild(1), boundR);
  5030. tgt.expr.setown(adjustBoundIntegerValues(boundL.expr, boundR.expr, false));
  5031. }
  5032. //---------------------------------------------------------------------------
  5033. void HqlCppTranslator::doBuildExprNegate(BuildCtx & ctx, IHqlExpression * expr, CHqlBoundExpr & tgt)
  5034. {
  5035. ITypeInfo * type = expr->queryType();
  5036. switch (type->getTypeCode())
  5037. {
  5038. case type_decimal:
  5039. {
  5040. bindAndPush(ctx, expr->queryChild(0));
  5041. HqlExprArray args;
  5042. callProcedure(ctx, DecNegateId, args);
  5043. tgt.expr.setown(createValue(no_decimalstack, LINK(type)));
  5044. }
  5045. break;
  5046. case type_data:
  5047. case type_qstring:
  5048. case type_string:
  5049. case type_varstring:
  5050. case type_unicode:
  5051. case type_varunicode:
  5052. case type_utf8:
  5053. throwError(HQLERR_MinusOnString);
  5054. default:
  5055. doBuildPureSubExpr(ctx, expr, tgt);
  5056. break;
  5057. }
  5058. }
  5059. //---------------------------------------------------------------------------
  5060. void HqlCppTranslator::doBuildExprRound(BuildCtx & ctx, IHqlExpression * expr, CHqlBoundExpr & tgt)
  5061. {
  5062. IHqlExpression * arg = expr->queryChild(0);
  5063. node_operator op = expr->getOperator();
  5064. switch (arg->queryType()->getTypeCode())
  5065. {
  5066. case type_decimal:
  5067. {
  5068. bindAndPush(ctx, arg);
  5069. HqlExprArray args;
  5070. if (op == no_round)
  5071. {
  5072. IHqlExpression * places = queryRealChild(expr, 1);
  5073. if (places)
  5074. {
  5075. args.append(*LINK(places));
  5076. callProcedure(ctx, DecRoundToId, args);
  5077. }
  5078. else
  5079. callProcedure(ctx, DecRoundId, args);
  5080. }
  5081. else
  5082. callProcedure(ctx, DecRoundUpId, args);
  5083. assertex(expr->queryType()->getTypeCode() == type_decimal);
  5084. tgt.expr.setown(createValue(no_decimalstack, expr->getType()));
  5085. }
  5086. break;
  5087. default:
  5088. {
  5089. if (op == no_round)
  5090. {
  5091. if (queryRealChild(expr, 1))
  5092. doBuildExprSysFunc(ctx, expr, tgt, roundToId);
  5093. else
  5094. doBuildExprSysFunc(ctx, expr, tgt, roundId);
  5095. }
  5096. else
  5097. doBuildExprSysFunc(ctx, expr, tgt, roundupId);
  5098. break;
  5099. }
  5100. }
  5101. }
  5102. void HqlCppTranslator::doBuildExprTrunc(BuildCtx & ctx, IHqlExpression * expr, CHqlBoundExpr & tgt)
  5103. {
  5104. IHqlExpression * arg = expr->queryChild(0);
  5105. switch (arg->queryType()->getTypeCode())
  5106. {
  5107. case type_decimal:
  5108. {
  5109. bindAndPush(ctx, arg);
  5110. HqlExprArray args;
  5111. callProcedure(ctx, DecTruncateId, args);
  5112. assertex(expr->queryType()->getTypeCode() == type_decimal);
  5113. tgt.expr.setown(createValue(no_decimalstack, expr->getType()));
  5114. }
  5115. break;
  5116. default:
  5117. {
  5118. doBuildExprSysFunc(ctx, expr, tgt, truncateId);
  5119. break;
  5120. }
  5121. }
  5122. }
  5123. //---------------------------------------------------------------------------
  5124. void HqlCppTranslator::doBuildExprAbs(BuildCtx & ctx, IHqlExpression * expr, CHqlBoundExpr & tgt)
  5125. {
  5126. ITypeInfo * type = expr->queryType();
  5127. switch (type->getTypeCode())
  5128. {
  5129. case type_decimal:
  5130. {
  5131. bindAndPush(ctx, expr->queryChild(0));
  5132. HqlExprArray args;
  5133. IIdAtom * func = DecAbsId;
  5134. callProcedure(ctx, func, args);
  5135. tgt.expr.setown(createValue(no_decimalstack, LINK(type)));
  5136. }
  5137. break;
  5138. case type_int:
  5139. case type_swapint:
  5140. case type_packedint:
  5141. case type_real:
  5142. {
  5143. CHqlBoundExpr temp;
  5144. buildTempExpr(ctx, expr->queryChild(0), temp);
  5145. ITypeInfo * type = expr->getType();
  5146. IHqlExpression * cond = createValue(no_ge, makeBoolType(), temp.expr.getLink(), createConstant(type->castFrom(true, 0)));
  5147. tgt.expr.setown(createValue(no_if, type, cond, temp.expr.getLink(), createValue(no_negate, LINK(type), temp.expr.getLink())));
  5148. }
  5149. break;
  5150. default:
  5151. buildExpr(ctx, expr->queryChild(0), tgt);
  5152. break;
  5153. }
  5154. }
  5155. //---------------------------------------------------------------------------
  5156. void HqlCppTranslator::doBuildAssignCatch(BuildCtx & ctx, const CHqlBoundTarget & target, IHqlExpression * expr)
  5157. {
  5158. BuildCtx subctx(ctx);
  5159. subctx.addGroup();
  5160. BuildCtx tryctx(subctx);
  5161. tryctx.addQuotedCompoundLiteral("try");
  5162. buildExprAssign(tryctx, target, expr->queryChild(0));
  5163. BuildCtx catchctx(subctx);
  5164. catchctx.addQuotedCompoundLiteral("catch (IException * e)");
  5165. IHqlExpression * exceptVar = associateLocalFailure(catchctx, "e");
  5166. buildExprAssign(catchctx, target, expr->queryChild(1));
  5167. HqlExprArray args;
  5168. args.append(*LINK(exceptVar));
  5169. callProcedure(catchctx, freeExceptionId, args);
  5170. }
  5171. //---------------------------------------------------------------------------
  5172. //-- no_externalcall --
  5173. static IHqlExpression * getCastParameter(IHqlExpression * curParam, ITypeInfo * argType, const char * argName)
  5174. {
  5175. type_t atc = argType->getTypeCode();
  5176. //Remove a few unnecessary casts which clutter up the code/or make it less efficient
  5177. if (isCast(curParam) && (curParam->queryType() == argType))
  5178. {
  5179. //If the following code is incorrect, the casts should get added back by the code that follows
  5180. IHqlExpression * uncast = curParam->queryChild(0);
  5181. //casts to larger size integers
  5182. if (atc == type_int)
  5183. curParam = uncast;
  5184. else if ((atc == type_unicode) && (uncast->queryType()->getTypeCode() == type_varunicode))
  5185. curParam = uncast;
  5186. else if ((atc == type_string) && (uncast->queryType()->getTypeCode() == type_varstring))
  5187. curParam = uncast;
  5188. }
  5189. ITypeInfo * paramType = curParam->queryType();
  5190. type_t ptc = paramType->getTypeCode();
  5191. if (atc != ptc)
  5192. {
  5193. switch (atc)
  5194. {
  5195. case type_unicode:
  5196. if ((argType->getSize() == UNKNOWN_LENGTH) && (ptc == type_varunicode))
  5197. return LINK(curParam);
  5198. break;
  5199. case type_string:
  5200. if ((argType->getSize() == UNKNOWN_LENGTH) &&
  5201. ((ptc == type_varstring) && (argType->queryCharset() == paramType->queryCharset())))
  5202. return LINK(curParam);
  5203. break;
  5204. case type_row:
  5205. if (curParam->isDataset())
  5206. throwError1(HQLERR_DatasetPassedToRowArg, argName);
  5207. // fallthrough
  5208. case type_dictionary:
  5209. case type_table:
  5210. case type_groupedtable:
  5211. {
  5212. IHqlExpression * record = queryRecord(argType);
  5213. if (record && (record->numChildren() != 0))
  5214. {
  5215. //really need to project instead.
  5216. //return ensureExprType(curParam, argType);
  5217. }
  5218. return LINK(curParam);
  5219. }
  5220. }
  5221. return ensureExprType(curParam, argType);
  5222. }
  5223. ITypeInfo * childType = argType->queryChildType();
  5224. if (paramType->queryTypeBase() != argType)
  5225. {
  5226. size32_t argSize = argType->getSize();
  5227. switch (atc)
  5228. {
  5229. case type_int:
  5230. case type_real:
  5231. if (argSize <= paramType->getSize())
  5232. return ensureExprType(curParam, argType);
  5233. break;
  5234. case type_decimal:
  5235. //Don't need explicit cast to cast between different sizes of these types...
  5236. //Will be done automatically by the compiler.
  5237. break;
  5238. case type_unicode:
  5239. //Don't need cast between different locales
  5240. if ((argSize != paramType->getSize()) && (argSize != UNKNOWN_LENGTH))
  5241. {
  5242. Owned<ITypeInfo> modArgType = makeUnicodeType(argType->getStringLen(), curParam->queryType()->queryLocale());
  5243. return ensureExprType(curParam, modArgType);
  5244. }
  5245. break;
  5246. case type_varunicode:
  5247. //Don't need cast between different locales
  5248. if(argSize != paramType->getSize())
  5249. {
  5250. Owned<ITypeInfo> modArgType = makeVarUnicodeType(argType->getStringLen(), curParam->queryType()->queryLocale());
  5251. return ensureExprType(curParam, modArgType);
  5252. }
  5253. break;
  5254. case type_utf8:
  5255. //Don't need cast between different locales
  5256. if(argSize != paramType->getSize())
  5257. {
  5258. Owned<ITypeInfo> modArgType = makeUtf8Type(argType->getStringLen(), curParam->queryType()->queryLocale());
  5259. return ensureExprType(curParam, modArgType);
  5260. }
  5261. break;
  5262. case type_set:
  5263. if (childType)
  5264. return ensureExprType(curParam, argType);
  5265. break;
  5266. case type_dictionary:
  5267. case type_table:
  5268. case type_groupedtable:
  5269. case type_row:
  5270. {
  5271. IHqlExpression * record = queryRecord(argType);
  5272. if (record && (record->numChildren() != 0))
  5273. {
  5274. //really need to project instead.
  5275. //return ensureExprType(curParam, argType);
  5276. }
  5277. break;
  5278. }
  5279. default:
  5280. return ensureExprType(curParam, argType);
  5281. }
  5282. }
  5283. // if (argType->queryCharset() != paramType->queryCharset())
  5284. // return ensureExprType(curParam, argType);
  5285. return LINK(curParam);
  5286. }
  5287. void HqlCppTranslator::normalizeBoundExpr(BuildCtx & ctx, CHqlBoundExpr & bound)
  5288. {
  5289. bound.expr.setown(convertWrapperToPointer(bound.expr));
  5290. }
  5291. IHqlExpression * HqlCppTranslator::doBuildInternalFunction(IHqlExpression * funcdef)
  5292. {
  5293. if (funcdef->queryChild(0)->getOperator() == no_external)
  5294. return LINK(funcdef);
  5295. unsigned match = internalFunctions.find(*funcdef);
  5296. if (match != NotFound)
  5297. return LINK(&internalFunctionExternals.item(match));
  5298. OwnedHqlExpr externalFuncdef = createExternalFuncdefFromInternal(funcdef);
  5299. internalFunctions.append(*LINK(funcdef));
  5300. internalFunctionExternals.append(*LINK(externalFuncdef));
  5301. buildFunctionDefinition(funcdef);
  5302. return LINK(externalFuncdef);
  5303. }
  5304. void HqlCppTranslator::doBuildCall(BuildCtx & ctx, const CHqlBoundTarget * tgt, IHqlExpression * expr, CHqlBoundExpr * result)
  5305. {
  5306. if (result && expr->isPure() && ctx.getMatchExpr(expr, *result))
  5307. return;
  5308. LinkedHqlExpr funcdef;
  5309. if (expr->getOperator() == no_externalcall)
  5310. {
  5311. funcdef.set(expr->queryExternalDefinition());
  5312. assertex(funcdef != NULL);
  5313. useFunction(funcdef);
  5314. }
  5315. else
  5316. {
  5317. IHqlExpression * def = expr->queryBody()->queryFunctionDefinition();
  5318. assertex(def);
  5319. if (def->getOperator() == no_param)
  5320. {
  5321. ITypeInfo * type = def->queryType();
  5322. IFunctionTypeExtra * extra = queryFunctionTypeExtra(type);
  5323. IHqlExpression * params = (IHqlExpression *)extra->queryParameters();
  5324. IHqlExpression * defaults = (IHqlExpression *)extra->queryDefaults();
  5325. HqlExprArray attrs;
  5326. attrs.append(*LINK(params));
  5327. if (defaults)
  5328. attrs.append(*LINK(defaults));
  5329. attrs.append(*createAttribute(contextAtom));
  5330. ITypeInfo * returnType = type->queryChildType();
  5331. OwnedHqlExpr externalExpr = createExternalReference(def->queryId(), LINK(returnType), attrs);
  5332. funcdef.setown(createFunctionDefinition(def->queryId(), LINK(externalExpr), LINK(params), LINK(defaults), NULL));
  5333. }
  5334. else
  5335. {
  5336. assertex(def && def->getOperator() == no_funcdef);
  5337. funcdef.setown(doBuildInternalFunction(def));
  5338. }
  5339. }
  5340. IHqlExpression * external = funcdef->queryChild(0);
  5341. IHqlExpression * formals = funcdef->queryChild(1);
  5342. if (external->hasAttribute(ctxmethodAtom) || external->hasAttribute(contextAtom))
  5343. ensureContextAvailable(ctx);
  5344. if (external->hasAttribute(gctxmethodAtom) || external->hasAttribute(globalContextAtom))
  5345. {
  5346. if (!ctx.queryMatchExpr(globalContextMarkerExpr))
  5347. throwError1(HQLERR_FuncNotInGlobalContext, str(external->queryName()));
  5348. }
  5349. unsigned maxArg = formals->numChildren();
  5350. unsigned maxParam = expr->numChildren();
  5351. bool returnByReference = false;
  5352. bool returnMustAssign = false;
  5353. HqlExprArray args;
  5354. unsigned arg = 0;
  5355. unsigned param;
  5356. unsigned firstParam = 0;
  5357. bool isMethod = external->hasAttribute(methodAtom) || external->hasAttribute(omethodAtom) ;
  5358. bool newFormatSet = !external->hasAttribute(oldSetFormatAtom);
  5359. bool translateSetReturn = false;
  5360. if (isMethod)
  5361. {
  5362. CHqlBoundExpr bound;
  5363. buildExpr(ctx, expr->queryChild(firstParam++), bound);
  5364. args.append(*bound.expr.getClear());
  5365. }
  5366. if (external->hasAttribute(userMatchFunctionAtom))
  5367. {
  5368. //MORE: Test valid in this location...
  5369. args.append(*createVariable("walker", makeBoolType()));
  5370. }
  5371. IHqlExpression * extendAttr = external->queryAttribute(extendAtom);
  5372. bool doneAssign = false;
  5373. CHqlBoundExpr localBound;
  5374. CHqlBoundTarget localTarget;
  5375. Linked<BoundRow> resultRow;
  5376. Linked<BoundRow> resultRowBuilder;
  5377. Owned<ITypeInfo> targetType = tgt ? LINK(tgt->queryType()) : makeVoidType();
  5378. ITypeInfo * retType = funcdef->queryType()->queryChildType();
  5379. BoundRow * resultSelfCursor = NULL;
  5380. switch (retType->getTypeCode())
  5381. {
  5382. case type_varstring:
  5383. case type_varunicode:
  5384. if (retType->getSize() == UNKNOWN_LENGTH)
  5385. {
  5386. if (hasConstModifier(retType))
  5387. break;
  5388. returnMustAssign = true;
  5389. if (tgt && !tgt->isFixedSize())
  5390. {
  5391. doneAssign = true;
  5392. localBound.expr.set(tgt->expr);
  5393. }
  5394. else
  5395. localBound.expr.setown(createWrapperTemp(ctx, retType, typemod_none));
  5396. break;
  5397. }
  5398. else
  5399. {
  5400. //fixed size strings => just pass pointer
  5401. IHqlExpression * resultVar = ctx.getTempDeclare(retType, NULL);
  5402. args.append(*getElementPointer(resultVar));
  5403. localBound.expr.setown(resultVar);
  5404. }
  5405. returnByReference = true;
  5406. break;
  5407. case type_string:
  5408. case type_data:
  5409. case type_qstring:
  5410. case type_unicode:
  5411. case type_utf8:
  5412. if (retType->getSize() == UNKNOWN_LENGTH)
  5413. {
  5414. OwnedHqlExpr lenVar;
  5415. OwnedHqlExpr strVar;
  5416. if (tgt && !tgt->isFixedSize())
  5417. {
  5418. doneAssign = true;
  5419. strVar.set(tgt->expr);
  5420. lenVar.set(tgt->length);
  5421. }
  5422. else
  5423. {
  5424. strVar.setown(createWrapperTemp(ctx, retType, typemod_none));
  5425. lenVar.setown(ctx.getTempDeclare(sizetType, NULL));
  5426. }
  5427. args.append(*LINK(lenVar));
  5428. args.append(*createValue(no_reference, strVar->getType(), LINK(strVar), LINK(extendAttr)));
  5429. localBound.length.set(lenVar);
  5430. localBound.expr.set(strVar);
  5431. }
  5432. else
  5433. {
  5434. //fixed size strings => just pass pointer
  5435. if (tgt && tgt->isFixedSize() && targetType->queryPromotedType() == retType)
  5436. {
  5437. doneAssign = true;
  5438. args.append(*getElementPointer(tgt->expr));
  5439. localBound.expr.set(tgt->expr);
  5440. }
  5441. else
  5442. {
  5443. IHqlExpression * resultVar = ctx.getTempDeclare(retType, NULL);
  5444. args.append(*getElementPointer(resultVar));
  5445. localBound.expr.setown(resultVar);
  5446. }
  5447. }
  5448. returnByReference = true;
  5449. break;
  5450. case type_set:
  5451. {
  5452. translateSetReturn = !newFormatSet;
  5453. OwnedHqlExpr lenVar;
  5454. OwnedHqlExpr strVar;
  5455. OwnedHqlExpr isAll;
  5456. if (tgt && !tgt->isFixedSize())
  5457. {
  5458. doneAssign = true;
  5459. strVar.set(tgt->expr);
  5460. if (translateSetReturn)
  5461. lenVar.setown(ctx.getTempDeclare(unsignedType, NULL));
  5462. else
  5463. lenVar.set(tgt->length);
  5464. assertex(tgt->isAll != NULL);
  5465. isAll.set(tgt->isAll);
  5466. }
  5467. else
  5468. {
  5469. Owned<ITypeInfo> dataType = makeDataType(UNKNOWN_LENGTH);
  5470. strVar.setown(createWrapperTemp(ctx, dataType, typemod_none));
  5471. if (translateSetReturn)
  5472. {
  5473. lenVar.setown(ctx.getTempDeclare(unsignedType, NULL));
  5474. }
  5475. else
  5476. {
  5477. lenVar.setown(ctx.getTempDeclare(sizetType, NULL));
  5478. isAll.setown(ctx.getTempDeclare(boolType, NULL));
  5479. }
  5480. }
  5481. if (newFormatSet)
  5482. args.append(*LINK(isAll));
  5483. args.append(*LINK(lenVar));
  5484. args.append(*createValue(no_reference, strVar->getType(), LINK(strVar)));
  5485. if (newFormatSet)
  5486. {
  5487. localBound.length.set(lenVar);
  5488. }
  5489. else
  5490. {
  5491. localBound.count.set(lenVar);
  5492. if (isAll && !isAll->queryValue())
  5493. ctx.addAssign(isAll, queryBoolExpr(false));
  5494. }
  5495. localBound.isAll.set(isAll);
  5496. localBound.expr.setown(createValue(no_implicitcast, makeReferenceModifier(LINK(retType)), LINK(strVar)));
  5497. returnByReference = true;
  5498. break;
  5499. }
  5500. case type_dictionary:
  5501. case type_table:
  5502. case type_groupedtable:
  5503. {
  5504. if (hasStreamedModifier(retType))
  5505. {
  5506. if (getBoolAttribute(external, allocatorAtom, true))
  5507. args.append(*createRowAllocator(ctx, ::queryRecord(retType)));
  5508. returnMustAssign = true;
  5509. if (tgt && hasStreamedModifier(targetType) && recordTypesMatch(targetType, retType))
  5510. {
  5511. doneAssign = true;
  5512. localBound.expr.set(tgt->expr);
  5513. }
  5514. else
  5515. localBound.expr.setown(createWrapperTemp(ctx, retType, typemod_none));
  5516. break;
  5517. }
  5518. const CHqlBoundTarget * curTarget;
  5519. if (tgt && !tgt->isFixedSize() &&
  5520. (hasLinkCountedModifier(targetType) == hasLinkCountedModifier(retType)))
  5521. {
  5522. doneAssign = true;
  5523. curTarget = tgt;
  5524. }
  5525. else
  5526. {
  5527. curTarget = &localTarget;
  5528. ExpressionFormat format = (hasLinkCountedModifier(retType) ? FormatLinkedDataset : FormatBlockedDataset);
  5529. createTempFor(ctx, retType, localTarget, typemod_none, format);
  5530. }
  5531. if (curTarget->count)
  5532. args.append(*LINK(curTarget->count));
  5533. if (curTarget->length)
  5534. args.append(*LINK(curTarget->length));
  5535. args.append(*createValue(no_reference, curTarget->expr->getType(), LINK(curTarget->expr)));
  5536. if (hasLinkCountedModifier(retType) && getBoolAttribute(external, allocatorAtom, true))
  5537. args.append(*createRowAllocator(ctx, ::queryRecord(retType)));
  5538. localBound.setFromTarget(*curTarget);
  5539. // localBound.expr.setown(createValue(no_implicitcast, makeReferenceModifier(LINK(retType)), LINK(strVar)));
  5540. returnByReference = true;
  5541. break;
  5542. }
  5543. case type_row:
  5544. {
  5545. if (hasLinkCountedModifier(retType))
  5546. {
  5547. if (hasNonNullRecord(retType) && getBoolAttribute(external, allocatorAtom, true))
  5548. args.append(*createRowAllocator(ctx, ::queryRecord(retType)));
  5549. //Always assign link counted rows to a temporary (or the target) to ensure the are not leaked.
  5550. returnMustAssign = true;
  5551. if (tgt && hasLinkCountedModifier(targetType) && recordTypesMatch(targetType, retType))
  5552. {
  5553. doneAssign = true;
  5554. localBound.expr.set(tgt->expr);
  5555. }
  5556. else
  5557. localBound.expr.setown(createWrapperTemp(ctx, retType, typemod_none));
  5558. }
  5559. else
  5560. {
  5561. //row, just pass pointer
  5562. if (tgt && recordTypesMatch(targetType, retType) && !hasLinkCountedModifier(targetType))
  5563. {
  5564. doneAssign = true;
  5565. args.append(*getPointer(tgt->expr));
  5566. localBound.expr.set(tgt->expr);
  5567. }
  5568. else
  5569. {
  5570. if (isVariableSizeRecord(expr->queryRecord()))
  5571. {
  5572. const char * name = str(expr->queryName());
  5573. throwError1(HQLERR_VariableRowMustBeLinked, name ? name : "");
  5574. }
  5575. resultRow.setown(declareTempRow(ctx, ctx, expr));
  5576. resultRowBuilder.setown(createRowBuilder(ctx, resultRow));
  5577. IHqlExpression * bound = resultRowBuilder->queryBound();
  5578. args.append(*getPointer(bound));
  5579. localBound.expr.setown(getPointer(resultRow->queryBound()));
  5580. }
  5581. returnByReference = true;
  5582. }
  5583. break;
  5584. }
  5585. case type_transform:
  5586. {
  5587. //Ugly, but target selector is passed in as the target
  5588. assertex(tgt && tgt->expr);
  5589. resultSelfCursor = resolveSelectorDataset(ctx, tgt->expr);
  5590. assertex(resultSelfCursor);
  5591. if (resultSelfCursor->queryBuilder())
  5592. args.append(*LINK(resultSelfCursor->queryBuilder()));
  5593. else
  5594. throwUnexpectedX("Expected a dynamic target for a transform - legacy not supported");
  5595. returnByReference = true;
  5596. doneAssign = true;
  5597. break;
  5598. }
  5599. case type_array:
  5600. UNIMPLEMENTED;
  5601. }
  5602. for (param = firstParam; param < maxParam; param++)
  5603. {
  5604. IHqlExpression * curParam = expr->queryChild(param);
  5605. if (curParam->isAttribute())
  5606. continue;
  5607. if (arg >= maxArg)
  5608. {
  5609. PrintLog("Too many parameters passed to function '%s'", str(expr->queryName()));
  5610. throwError1(HQLERR_TooManyParameters, str(expr->queryName()));
  5611. }
  5612. CHqlBoundExpr bound;
  5613. IHqlExpression * curArg = formals->queryChild(arg);
  5614. ITypeInfo * argType = curArg->queryType();
  5615. OwnedHqlExpr castParam = getCastParameter(curParam, argType, curArg->queryId()->queryStr());
  5616. type_t atc = argType->getTypeCode();
  5617. switch (atc)
  5618. {
  5619. case type_dictionary:
  5620. case type_table:
  5621. case type_groupedtable:
  5622. {
  5623. IHqlExpression* passParamAttr = getFunctionBodyAttribute(external, passParameterMetaAtom);
  5624. if (getBoolAttributeValue(passParamAttr))
  5625. args.append(*buildMetaParameter(curParam));
  5626. ExpressionFormat format = queryNaturalFormat(argType);
  5627. doBuildDataset(ctx, castParam, bound, format);
  5628. ensureDatasetFormat(ctx, argType, bound, format);
  5629. break;
  5630. }
  5631. case type_row:
  5632. {
  5633. IHqlExpression* passParamAttr = getFunctionBodyAttribute(external, passParameterMetaAtom);
  5634. if (getBoolAttributeValue(passParamAttr))
  5635. args.append(*buildMetaParameter(curParam));
  5636. Owned<IReferenceSelector> selector = buildNewRow(ctx, castParam);
  5637. if (hasLinkCountedModifier(argType))
  5638. selector.setown(ensureLinkCountedRow(ctx, selector));
  5639. selector->buildAddress(ctx, bound);
  5640. break;
  5641. }
  5642. case type_set:
  5643. {
  5644. ITypeInfo * elemType = argType->queryChildType();
  5645. if (newFormatSet && elemType && (elemType->getTypeCode() != type_any) && (elemType != castParam->queryType()->queryChildType()))
  5646. buildExprEnsureType(ctx, castParam, bound, argType);
  5647. else
  5648. buildExpr(ctx, castParam, bound);
  5649. if (isUnknownSize(elemType))
  5650. {
  5651. ITypeInfo * boundElemType = bound.queryType()->queryChildType();
  5652. assertex(!boundElemType || isUnknownSize(boundElemType));
  5653. }
  5654. normalizeBoundExpr(ctx, bound);
  5655. break;
  5656. }
  5657. case type_decimal:
  5658. {
  5659. buildSimpleExpr(ctx, castParam, bound);
  5660. normalizeBoundExpr(ctx, bound);
  5661. break;
  5662. }
  5663. case type_record:
  5664. {
  5665. args.append(*buildMetaParameter(curParam));
  5666. break;
  5667. }
  5668. default:
  5669. {
  5670. buildExpr(ctx, castParam, bound);
  5671. normalizeBoundExpr(ctx, bound);
  5672. break;
  5673. }
  5674. }
  5675. bool done = false;
  5676. switch (atc)
  5677. {
  5678. case type_string:
  5679. case type_data:
  5680. case type_qstring:
  5681. case type_unicode:
  5682. case type_utf8:
  5683. if (argType->getSize() == UNKNOWN_LENGTH)
  5684. args.append(*getBoundLength(bound));
  5685. /*
  5686. Ensure parameter is passed as non-const if the argument does not have const.
  5687. if (!curArg->hasAttribute(constAtom))//!argType->isConstantType())// && bound.queryType()->isConstantType())
  5688. bound.expr.setown(createValue(no_cast, LINK(argType), LINK(bound.expr)));
  5689. */
  5690. break;
  5691. case type_varstring:
  5692. case type_varunicode:
  5693. //MORE: pass length only if an out parameter
  5694. break;
  5695. case type_set:
  5696. {
  5697. if (newFormatSet)
  5698. {
  5699. args.append(*bound.getIsAll());
  5700. args.append(*getBoundSize(bound));
  5701. }
  5702. else
  5703. {
  5704. if (castParam->getOperator() == no_all)
  5705. throwError(HQLERR_AllPassedExternal);
  5706. args.append(*getBoundCount(bound));
  5707. }
  5708. break;
  5709. }
  5710. case type_dictionary:
  5711. case type_table:
  5712. case type_groupedtable:
  5713. {
  5714. if (!hasStreamedModifier(argType))
  5715. {
  5716. if (isArrayRowset(argType))
  5717. args.append(*getBoundCount(bound));
  5718. else
  5719. args.append(*getBoundSize(bound));
  5720. }
  5721. bound.expr.setown(getPointer(bound.expr));
  5722. break;
  5723. }
  5724. case type_record:
  5725. done = true;
  5726. break;
  5727. case type_array:
  5728. UNIMPLEMENTED;
  5729. }
  5730. if (!done)
  5731. args.append(*bound.expr.getClear());
  5732. arg++;
  5733. }
  5734. if (arg < maxArg)
  5735. {
  5736. //MORE: Process default parameters...
  5737. PrintLog("Not enough parameters passed to function '%s'", str(expr->queryName()));
  5738. throwError1(HQLERR_TooFewParameters, str(expr->queryName()));
  5739. }
  5740. OwnedHqlExpr call = bindTranslatedFunctionCall(funcdef, args);
  5741. CHqlBoundExpr boundTimer, boundStart;
  5742. if (external->hasAttribute(timeAtom))
  5743. buildStartTimer(ctx, boundTimer, boundStart, str(external->queryId()));
  5744. //either copy the integral value across, or a var string to fixed string
  5745. if (returnMustAssign)
  5746. {
  5747. ctx.addAssign(localBound.expr, call);
  5748. }
  5749. else if (resultSelfCursor)
  5750. {
  5751. OwnedHqlExpr sizeVar = ctx.getTempDeclare(unsignedType, call);
  5752. OwnedHqlExpr sizeOfExpr = createSizeof(resultSelfCursor->querySelector());
  5753. ctx.associateExpr(sizeOfExpr, sizeVar);
  5754. }
  5755. else if (returnByReference || (!tgt && !result))
  5756. {
  5757. ctx.addExpr(call);
  5758. if (translateSetReturn && tgt)
  5759. {
  5760. CHqlBoundTarget targetLength;
  5761. CHqlBoundExpr boundLength;
  5762. boundLength.expr.setown(getBoundLength(localBound));
  5763. targetLength.expr.set(tgt->length);
  5764. assign(ctx, targetLength, boundLength);
  5765. }
  5766. }
  5767. else
  5768. localBound.expr.set(call);
  5769. if (localBound.expr)
  5770. localBound.expr.setown(convertWrapperToPointer(localBound.expr));
  5771. if (tgt && !doneAssign)
  5772. assign(ctx, *tgt, localBound);
  5773. else if (result)
  5774. result->set(localBound);
  5775. //Old style row target where the row is passed in as a parameter
  5776. if (resultRow)
  5777. finalizeTempRow(ctx, resultRow, resultRowBuilder);
  5778. if (external->hasAttribute(timeAtom))
  5779. buildStopTimer(ctx, boundTimer, boundStart);
  5780. if (returnByReference)
  5781. ctx.associateExpr(expr, localBound);
  5782. }
  5783. void HqlCppTranslator::doBuildExprCall(BuildCtx & ctx, IHqlExpression * expr, CHqlBoundExpr & tgt)
  5784. {
  5785. doBuildCall(ctx, NULL, expr, &tgt);
  5786. }
  5787. void HqlCppTranslator::doBuildAssignCall(BuildCtx & ctx, const CHqlBoundTarget & target, IHqlExpression * expr)
  5788. {
  5789. ITypeInfo * exprType = expr->queryType()->queryPromotedType();
  5790. if ((exprType->getSize() == UNKNOWN_LENGTH) && target.isFixedSize())
  5791. {
  5792. doBuildExprAssign(ctx, target, expr);
  5793. return;
  5794. }
  5795. ITypeInfo * targetType = target.queryType()->queryPromotedType();
  5796. if ((isStringType(exprType) || isUnicodeType(exprType) || isStringType(targetType) || isUnicodeType(targetType)) && exprType->getTypeCode() != targetType->getTypeCode())
  5797. {
  5798. doBuildExprAssign(ctx, target, expr);
  5799. return;
  5800. }
  5801. if ((exprType->getTypeCode() == type_set) && (queryUnqualifiedType(targetType) != queryUnqualifiedType(exprType)))
  5802. {
  5803. if (exprType->queryChildType()) // allow direct assignment of generic set functions. (e.g., internal)
  5804. {
  5805. doBuildExprAssign(ctx, target, expr);
  5806. return;
  5807. }
  5808. }
  5809. doBuildCall(ctx, &target, expr, NULL);
  5810. }
  5811. void HqlCppTranslator::doBuildStmtCall(BuildCtx & ctx, IHqlExpression * expr)
  5812. {
  5813. doBuildCall(ctx, NULL, expr, NULL);
  5814. }
  5815. //---------------------------------------------------------------------------
  5816. void HqlCppTranslator::doBuildXmlEncode(BuildCtx & ctx, const CHqlBoundTarget * tgt, IHqlExpression * expr, CHqlBoundExpr * result)
  5817. {
  5818. node_operator op = expr->getOperator();
  5819. bool isUnicode = isUnicodeType(expr->queryType());
  5820. HqlExprArray args;
  5821. IIdAtom * func;
  5822. args.append(*LINK(expr->queryChild(0)));
  5823. if (op == no_xmldecode)
  5824. func = isUnicode ? xmlDecodeUStrId : xmlDecodeStrId;
  5825. else
  5826. {
  5827. func = isUnicode ? xmlEncodeUStrId : xmlEncodeStrId;
  5828. __int64 flags = 0;
  5829. if (expr->hasAttribute(allAtom))
  5830. flags = ENCODE_WHITESPACE;
  5831. args.append(*createConstant(flags));
  5832. }
  5833. OwnedHqlExpr call = bindFunctionCall(func, args);
  5834. if (tgt)
  5835. buildExprAssign(ctx, *tgt, call);
  5836. else
  5837. buildExpr(ctx, call, *result);
  5838. }
  5839. //---------------------------------------------------------------------------
  5840. //-- no_cast --
  5841. //-- no_implicitcast --
  5842. IValue * getCastValue(ITypeInfo * cast, IHqlExpression * arg)
  5843. {
  5844. Owned<IValue> value;
  5845. switch (arg->getOperator())
  5846. {
  5847. case no_constant:
  5848. value.set(arg->queryValue());
  5849. break;
  5850. case no_cast: case no_implicitcast:
  5851. value.setown(getCastValue(arg->queryType(), arg->queryChild(0)));
  5852. break;
  5853. default:
  5854. return NULL;
  5855. }
  5856. if (!value)
  5857. return NULL;
  5858. return value->castTo(cast);
  5859. }
  5860. inline IHqlExpression * getCastExpr(ITypeInfo * cast, IHqlExpression * arg)
  5861. {
  5862. IValue * value = getCastValue(cast, arg);
  5863. if (value)
  5864. return createConstant(value);
  5865. return NULL;
  5866. }
  5867. void HqlCppTranslator::doBuildAssignCast(BuildCtx & ctx, const CHqlBoundTarget & target, IHqlExpression * expr)
  5868. {
  5869. IHqlExpression * left = expr->queryChild(0);
  5870. if (options.foldConstantCast)
  5871. {
  5872. //remove when we have constant folding installed...
  5873. OwnedHqlExpr cast = getCastExpr(expr->queryType(), left);
  5874. if (cast)
  5875. {
  5876. buildExprAssign(ctx, target, cast);
  5877. return;
  5878. }
  5879. }
  5880. #if 0
  5881. StringBuffer s;
  5882. s.append("assign cast from=");
  5883. left->queryType()->getECLType(s);
  5884. target.queryType()->getECLType(s.append(" target="));
  5885. expr->queryType()->getECLType(s.append(" expr="));
  5886. PrintLog(s.str());
  5887. #endif
  5888. ITypeInfo * targetType = target.queryType();
  5889. ITypeInfo * exprType = expr->queryType();
  5890. if ((targetType->queryPromotedType() == exprType->queryPromotedType()))
  5891. {
  5892. CHqlBoundExpr bound;
  5893. if (ctx.getMatchExpr(left, bound))
  5894. assignAndCast(ctx, target, bound);
  5895. else
  5896. {
  5897. OwnedHqlExpr values = normalizeListCasts(expr);
  5898. if (values != expr)
  5899. {
  5900. buildExprAssign(ctx, target, values);
  5901. return;
  5902. }
  5903. node_operator leftOp = left->getOperator();
  5904. bool useTemp = requiresTemp(ctx, left, false) && (leftOp != no_concat) && (leftOp != no_createset);
  5905. if (useTemp && target.isFixedSize())
  5906. {
  5907. bool ignoreStretched = false;
  5908. switch (leftOp)
  5909. {
  5910. case no_case:
  5911. case no_map:
  5912. //MORE: If the length of the bulk of the branches match then it is worth ignoring.
  5913. //would be worthwhile if isStringType(targetType) || isUnicodeType(targetType);
  5914. break;
  5915. case no_substring:
  5916. //don't do this if the target type is unicode at the moment
  5917. ignoreStretched = isStringType(targetType);
  5918. break;
  5919. }
  5920. if (ignoreStretched)
  5921. {
  5922. Owned<ITypeInfo> stretchedType = getStretchedType(targetType->getStringLen(), exprType);
  5923. if (isSameBasicType(stretchedType, targetType->queryPromotedType()))
  5924. useTemp = false;
  5925. }
  5926. }
  5927. if (useTemp)
  5928. {
  5929. buildExpr(ctx, left, bound);
  5930. assignAndCast(ctx, target, bound);
  5931. }
  5932. else
  5933. {
  5934. buildExprAssign(ctx, target, left);
  5935. }
  5936. }
  5937. return;
  5938. }
  5939. ITypeInfo * leftType = left->queryType();
  5940. if ((targetType->queryPromotedType() == left->queryType()->queryPromotedType()))
  5941. {
  5942. if (preservesValue(exprType, leftType))
  5943. {
  5944. buildExprAssign(ctx, target, left);
  5945. return;
  5946. }
  5947. }
  5948. CHqlBoundExpr pure;
  5949. bool assignDirect = false;
  5950. if ((exprType->getSize() == UNKNOWN_LENGTH) && (targetType->getTypeCode() == exprType->getTypeCode()) &&
  5951. (isStringType(exprType) || isUnicodeType(exprType)))
  5952. {
  5953. OwnedITypeInfo stretched = getStretchedType(UNKNOWN_LENGTH, targetType->queryPromotedType());
  5954. if (stretched == exprType->queryPromotedType())
  5955. assignDirect = true;
  5956. }
  5957. if (assignDirect)
  5958. buildExpr(ctx, left, pure);
  5959. else
  5960. buildExpr(ctx, expr, pure);
  5961. assignAndCast(ctx, target, pure);
  5962. }
  5963. void HqlCppTranslator::doBuildExprCast(BuildCtx & ctx, IHqlExpression * expr, CHqlBoundExpr & tgt)
  5964. {
  5965. IHqlExpression * arg = expr->queryChild(0);
  5966. ITypeInfo * exprType = expr->queryType();
  5967. if (exprType->getTypeCode() == type_set)
  5968. {
  5969. OwnedHqlExpr castArg = ensureExprType(arg, exprType);
  5970. if ((castArg->getOperator() != no_cast) && (castArg->getOperator() != no_implicitcast))
  5971. {
  5972. buildExpr(ctx, castArg, tgt);
  5973. return;
  5974. }
  5975. //The case is almost certainly going to go via a temporary anyway, better code is generated if we can avoid an extra assign
  5976. buildTempExpr(ctx, expr, tgt);
  5977. return;
  5978. }
  5979. if (options.foldConstantCast)
  5980. {
  5981. //remove when we have constant folding installed...
  5982. IHqlExpression * cast = getCastExpr(exprType, arg);
  5983. if (cast)
  5984. {
  5985. tgt.expr.setown(cast);
  5986. return;
  5987. }
  5988. }
  5989. // weird special case.... (int4)(swapint4)int4_x - remove both casts...
  5990. switch (arg->getOperator())
  5991. {
  5992. case no_cast:
  5993. case no_implicitcast:
  5994. {
  5995. IHqlExpression * child = arg->queryChild(0);
  5996. if (child->queryType() == exprType)
  5997. {
  5998. if ((exprType->getTypeCode() == type_int) || (exprType->getTypeCode() == type_swapint) || (exprType->getTypeCode() == type_packedint))
  5999. {
  6000. ITypeInfo * argType = arg->queryType();
  6001. if ((argType->getTypeCode() == type_int) || (argType->getTypeCode() == type_swapint) || (argType->getTypeCode() == type_packedint))
  6002. {
  6003. if (argType->getSize() == exprType->getSize())
  6004. {
  6005. buildExpr(ctx, child, tgt);
  6006. return;
  6007. }
  6008. }
  6009. }
  6010. }
  6011. break;
  6012. }
  6013. #if 0
  6014. case no_if:
  6015. {
  6016. //optimize cast of an if where one of the arguments is already the correct type
  6017. //this doesn't really improve things
  6018. IHqlExpression * lhs = arg->queryChild(1);
  6019. IHqlExpression * rhs = arg->queryChild(2);
  6020. if (lhs->queryType() == exprType || lhs->getOperator() == no_constant ||
  6021. rhs->queryType() == exprType || rhs->getOperator() == no_constant)
  6022. {
  6023. HqlExprArray args;
  6024. args.append(*LINK(arg->queryChild(0)));
  6025. args.append(*ensureExprType(lhs, exprType));
  6026. args.append(*ensureExprType(rhs, exprType));
  6027. unwindChildren(args, arg, 3);
  6028. OwnedHqlExpr next = createValue(no_if, LINK(exprType), args);
  6029. buildExpr(ctx, next, tgt);
  6030. return;
  6031. }
  6032. break;
  6033. }
  6034. #endif
  6035. case no_substring:
  6036. {
  6037. ITypeInfo * argType = arg->queryType();
  6038. if ((exprType->getSize() != UNKNOWN_LENGTH) && (argType->getSize() == UNKNOWN_LENGTH) && (exprType->getTypeCode() == argType->getTypeCode()))
  6039. {
  6040. OwnedITypeInfo stretched = getStretchedType(exprType->getStringLen(), argType);
  6041. if (stretched == exprType)
  6042. {
  6043. buildTempExpr(ctx, expr, tgt);
  6044. return;
  6045. }
  6046. }
  6047. break;
  6048. }
  6049. }
  6050. CHqlBoundExpr pure;
  6051. buildExpr(ctx, arg, pure);
  6052. doBuildExprCast(ctx, exprType, pure, tgt);
  6053. if ((arg->queryType()->getTypeCode() == type_decimal) && !isTypePassedByAddress(exprType))
  6054. {
  6055. OwnedHqlExpr translated = tgt.getTranslatedExpr();
  6056. buildTempExpr(ctx, translated, tgt);
  6057. }
  6058. }
  6059. void HqlCppTranslator::doBuildCastViaTemp(BuildCtx & ctx, ITypeInfo * to, CHqlBoundExpr & pure, CHqlBoundExpr & tgt)
  6060. {
  6061. CHqlBoundTarget boundTarget;
  6062. Linked<ITypeInfo> targetType = to;
  6063. //If the temporary size can be deduced, then use a fixed length temporary to save a heap operation.
  6064. ITypeInfo * fromType = pure.expr->queryType();
  6065. if (isStringType(to) && to->getSize() == UNKNOWN_LENGTH && isStringType(fromType) && !pure.length)
  6066. {
  6067. assertex(fromType->getSize() != UNKNOWN_LENGTH);
  6068. targetType.setown(getStretchedType(fromType->getStringLen(), to));
  6069. }
  6070. OwnedHqlExpr translated = pure.getTranslatedExpr();
  6071. OwnedHqlExpr cast = ensureExprType(translated, targetType);
  6072. buildTempExpr(ctx, cast, tgt);
  6073. }
  6074. void HqlCppTranslator::doBuildCastViaString(BuildCtx & ctx, ITypeInfo * to, const CHqlBoundExpr & pure, CHqlBoundExpr & tgt)
  6075. {
  6076. ITypeInfo * from = pure.expr->queryType();
  6077. Owned<ITypeInfo> tempType = makeStringType(from->getStringLen(), NULL, NULL);
  6078. OwnedHqlExpr temp = createValue(no_implicitcast, LINK(to),
  6079. createValue(no_implicitcast, tempType.getLink(), pure.getTranslatedExpr()));
  6080. buildExpr(ctx, temp, tgt);
  6081. }
  6082. void HqlCppTranslator::doBuildExprCast(BuildCtx & ctx, ITypeInfo * to, CHqlBoundExpr & pure, CHqlBoundExpr & tgt)
  6083. {
  6084. ITypeInfo * from = pure.expr->queryType();
  6085. HqlExprArray args;
  6086. IIdAtom * funcName = NULL;
  6087. OwnedHqlExpr op;
  6088. switch (to->getTypeCode())
  6089. {
  6090. case type_boolean:
  6091. {
  6092. switch (from->getTypeCode())
  6093. {
  6094. case type_string:
  6095. funcName = an2bId;
  6096. args.append(*getBoundLength(pure));
  6097. args.append(*getElementPointer(pure.expr));
  6098. break;
  6099. case type_data:
  6100. funcName = data2BoolId;
  6101. args.append(*getBoundLength(pure));
  6102. args.append(*getElementPointer(pure.expr));
  6103. break;
  6104. case type_qstring:
  6105. funcName = qstr2BoolId;
  6106. args.append(*getBoundLength(pure));
  6107. args.append(*getElementPointer(pure.expr));
  6108. break;
  6109. case type_varstring:
  6110. funcName = vn2bId;
  6111. args.append(*getElementPointer(pure.expr));
  6112. break;
  6113. case type_decimal:
  6114. ensurePushed(ctx, pure);
  6115. funcName = DecCompareNullId;
  6116. break;
  6117. case type_unicode:
  6118. case type_varunicode:
  6119. case type_utf8:
  6120. doBuildCastViaString(ctx, to, pure, tgt);
  6121. return;
  6122. case type_real:
  6123. default:
  6124. //default action
  6125. break;
  6126. }
  6127. break;
  6128. }
  6129. case type_packedint:
  6130. {
  6131. ITypeInfo * logicalType = to->queryPromotedType();
  6132. size32_t toSize = logicalType->getSize();
  6133. if ((from->getTypeCode() != type_int) || (toSize != from->getSize()))
  6134. {
  6135. OwnedHqlExpr translated = pure.getTranslatedExpr();
  6136. OwnedHqlExpr castTranslated = createValue(no_implicitcast, LINK(logicalType), LINK(translated));
  6137. buildExpr(ctx, castTranslated, tgt);
  6138. return;
  6139. }
  6140. tgt.set(pure);
  6141. return;
  6142. }
  6143. case type_swapint:
  6144. {
  6145. if ((from->getTypeCode() == type_swapint) && (to->getSize() == from->getSize()))
  6146. {
  6147. break; // default behaviour
  6148. //MORE: Could special case cast between diff size swapints, but a bit too complicated.
  6149. }
  6150. if (from->getTypeCode() != type_int)
  6151. {
  6152. ITypeInfo * tempType = makeIntType(to->getSize(), to->isSigned());
  6153. IHqlExpression * translated = pure.getTranslatedExpr();
  6154. translated = createValue(no_implicitcast, tempType, translated);
  6155. OwnedHqlExpr castTranslated = createValue(no_implicitcast, LINK(to), translated);
  6156. buildExpr(ctx, castTranslated, tgt);
  6157. return;
  6158. }
  6159. unsigned toSize = to->getSize();
  6160. unsigned fromSize = from->getSize();
  6161. if ((toSize == 1) && (fromSize == 1))
  6162. {
  6163. if (to->isSigned() != from->isSigned())
  6164. break;
  6165. tgt.expr.setown(createValue(no_typetransfer, LINK(to), LINK(pure.expr)));
  6166. return;
  6167. }
  6168. if (toSize != fromSize)
  6169. {
  6170. Owned<ITypeInfo> tempType = makeIntType(toSize, from->isSigned());
  6171. CHqlBoundExpr tempInt;
  6172. tempInt.expr.setown(ensureExprType(pure.expr, tempType));
  6173. doBuildCastViaTemp(ctx, to, tempInt, tgt);
  6174. }
  6175. else
  6176. doBuildCastViaTemp(ctx, to, pure, tgt);
  6177. return;
  6178. }
  6179. case type_int:
  6180. {
  6181. switch (from->getTypeCode())
  6182. {
  6183. case type_qstring:
  6184. {
  6185. //Need to go via a temporary string.
  6186. Owned<ITypeInfo> tempType = makeStringType(from->getStringLen(), NULL, NULL);
  6187. OwnedHqlExpr temp = createValue(no_implicitcast, LINK(to),
  6188. createValue(no_implicitcast, tempType.getLink(), pure.getTranslatedExpr()));
  6189. buildExpr(ctx, temp, tgt);
  6190. return;
  6191. }
  6192. case type_string:
  6193. case type_data:
  6194. {
  6195. IAtom * charset = from->queryCharset()->queryName();
  6196. if (charset == ebcdicAtom)
  6197. {
  6198. if (to->isSigned())
  6199. funcName = (to->getSize() > 4 ? en2ls8Id : en2ls4Id);
  6200. else
  6201. funcName = (to->getSize() > 4 ? en2l8Id : en2l4Id);
  6202. }
  6203. else if ((charset == asciiAtom) || (charset == dataAtom))
  6204. {
  6205. if (to->isSigned())
  6206. funcName = (to->getSize() > 4 ? an2ls8Id : an2ls4Id);
  6207. else
  6208. funcName = (to->getSize() > 4 ? an2l8Id : an2l4Id);
  6209. }
  6210. else
  6211. assertex(!"Unknown character set");
  6212. //MORE: This should really cast the result to the real width to remove extra bytes.
  6213. //e.g. (unsigned3)-1 should be 0xffffff, not 0xffffffff
  6214. args.append(*getBoundLength(pure));
  6215. args.append(*getElementPointer(pure.expr));
  6216. break;
  6217. }
  6218. case type_varstring:
  6219. if (to->isSigned())
  6220. funcName = (to->getSize() > 4 ? vn2ls8Id : vn2ls4Id);
  6221. else
  6222. funcName = (to->getSize() > 4 ? vn2l8Id : vn2l4Id);
  6223. args.append(*getElementPointer(pure.expr));
  6224. break;
  6225. case type_decimal:
  6226. ensurePushed(ctx, pure);
  6227. if (to->getSize() > 4)
  6228. funcName = DecPopInt64Id;
  6229. else
  6230. funcName = DecPopLongId;
  6231. break;
  6232. case type_packedint:
  6233. {
  6234. if (to->getSize() < from->getSize())
  6235. {
  6236. funcName = castIntId[to->getSize()][to->isSigned()];
  6237. args.append(*LINK(pure.expr));
  6238. }
  6239. else
  6240. tgt.set(pure);
  6241. break;
  6242. }
  6243. case type_swapint:
  6244. {
  6245. unsigned toSize = to->getSize();
  6246. unsigned fromSize = from->getSize();
  6247. if ((toSize == 1) && (fromSize == 1))
  6248. {
  6249. if (to->isSigned() != from->isSigned())
  6250. break;
  6251. tgt.expr.setown(createValue(no_typetransfer, LINK(to), LINK(pure.expr)));
  6252. return;
  6253. }
  6254. if (toSize != fromSize)
  6255. {
  6256. Owned<ITypeInfo> tempType = makeIntType(fromSize, from->isSigned());
  6257. CHqlBoundExpr tempInt;
  6258. doBuildCastViaTemp(ctx, tempType, pure, tempInt);
  6259. funcName = castIntId[to->getSize()][to->isSigned()];
  6260. if (funcName && toSize < fromSize)
  6261. {
  6262. args.append(*LINK(tempInt.expr));
  6263. tgt.expr.setown(bindTranslatedFunctionCall(funcName, args));
  6264. }
  6265. else
  6266. tgt.expr.setown(ensureExprType(tempInt.expr, to));
  6267. }
  6268. else
  6269. doBuildCastViaTemp(ctx, to, pure, tgt);
  6270. return;
  6271. }
  6272. case type_int:
  6273. if (to->getSize() < from->getSize())
  6274. {
  6275. IIdAtom * name = castIntId[to->getSize()][to->isSigned()];
  6276. if (name)
  6277. {
  6278. args.append(*LINK(pure.expr));
  6279. IHqlExpression * call = bindTranslatedFunctionCall(name, args);
  6280. op.setown(createValue(no_typetransfer, LINK(to), call));
  6281. }
  6282. }
  6283. break;
  6284. case type_unicode:
  6285. case type_varunicode:
  6286. case type_utf8:
  6287. doBuildCastViaString(ctx, to, pure, tgt);
  6288. return;
  6289. case type_real:
  6290. default:
  6291. //default action
  6292. break;
  6293. }
  6294. break;
  6295. }
  6296. case type_real:
  6297. switch (from->getTypeCode())
  6298. {
  6299. case type_qstring:
  6300. case type_unicode:
  6301. case type_varunicode:
  6302. case type_utf8:
  6303. doBuildCastViaString(ctx, to, pure, tgt);
  6304. return;
  6305. case type_data:
  6306. case type_string:
  6307. funcName = from->queryCharset()->queryName() != ebcdicAtom ? an2fId : en2fId;
  6308. args.append(*getBoundLength(pure));
  6309. args.append(*getElementPointer(pure.expr));
  6310. break;
  6311. case type_varstring:
  6312. funcName = from->queryCharset()->queryName() != ebcdicAtom ? vn2fId : ex2fId;
  6313. args.append(*getElementPointer(pure.expr));
  6314. break;
  6315. case type_decimal:
  6316. ensurePushed(ctx, pure);
  6317. funcName = DecPopRealId;
  6318. break;
  6319. case type_swapint:
  6320. {
  6321. //cast via intermediate int.
  6322. ITypeInfo * type = makeIntType(from->getSize(), from->isSigned());
  6323. IHqlExpression * translated = pure.getTranslatedExpr();
  6324. OwnedHqlExpr castTranslated = createValue(no_implicitcast, type, translated);
  6325. pure.clear();
  6326. buildExpr(ctx, castTranslated, pure);
  6327. from = type;
  6328. }
  6329. //fallthrough
  6330. case type_int:
  6331. case type_boolean:
  6332. case type_packedint:
  6333. default:
  6334. //default action
  6335. break;
  6336. }
  6337. break;
  6338. case type_decimal:
  6339. {
  6340. ensurePushed(ctx, pure);
  6341. bool needToSetPrecision = true;
  6342. unsigned toDigits = to->getDigits();
  6343. unsigned toPrecision = to->getPrecision();
  6344. switch (from->getTypeCode())
  6345. {
  6346. case type_int:
  6347. case type_swapint:
  6348. if (toDigits >= from->getDigits())
  6349. needToSetPrecision = false;
  6350. break;
  6351. case type_decimal:
  6352. {
  6353. unsigned fromDigits = from->getDigits();
  6354. unsigned fromPrecision = from->getPrecision();
  6355. if (((toDigits - toPrecision) >= (fromDigits - fromPrecision)) &&
  6356. (toPrecision >= fromPrecision))
  6357. needToSetPrecision = false;
  6358. break;
  6359. }
  6360. }
  6361. if ((toDigits == MAX_DECIMAL_DIGITS) && (toPrecision == MAX_DECIMAL_PRECISION))
  6362. needToSetPrecision = false;
  6363. if (needToSetPrecision)
  6364. {
  6365. args.append(*createConstant(createIntValue(to->getDigits(), 1, false)));
  6366. args.append(*createConstant(createIntValue(to->getPrecision(), 1, false)));
  6367. callProcedure(ctx, DecSetPrecisionId, args);
  6368. }
  6369. op.setown(createValue(no_decimalstack, LINK(to)));
  6370. }
  6371. break;
  6372. case type_set:
  6373. {
  6374. //MORE: Shouldn't have to create this node...
  6375. OwnedHqlExpr cast = createValue(no_implicitcast, LINK(to), pure.getTranslatedExpr());
  6376. buildTempExpr(ctx, cast, tgt);
  6377. return;
  6378. }
  6379. case type_pointer:
  6380. case type_row:
  6381. break;
  6382. case type_varstring:
  6383. if ((to->getSize() == UNKNOWN_LENGTH) && (from->getTypeCode() == type_varstring))
  6384. tgt.set(pure);
  6385. else
  6386. doBuildCastViaTemp(ctx, to, pure, tgt);
  6387. return;
  6388. case type_string:
  6389. case type_data:
  6390. {
  6391. if (canRemoveStringCast(to, from))
  6392. {
  6393. ICharsetInfo * srcset = from->queryCharset();
  6394. ICharsetInfo * tgtset = to->queryCharset();
  6395. //Data never calls a conversion function... but does add a cast
  6396. if ((srcset == tgtset) || (to->getTypeCode() == type_data) || (from->getTypeCode() == type_data))
  6397. {
  6398. if (from->getTypeCode() == type_varstring)
  6399. tgt.length.setown(getBoundLength(pure));
  6400. else
  6401. tgt.length.set(pure.length);
  6402. Owned<ITypeInfo> newType;
  6403. if (to->getSize() == UNKNOWN_LENGTH)
  6404. newType.setown(getStretchedType(from->getSize(), to));
  6405. else
  6406. newType.set(to);
  6407. if (from->getTypeCode() != type_data)
  6408. {
  6409. newType.setown(cloneModifiers(from, newType));
  6410. tgt.expr.setown(createValue(no_typetransfer, newType.getClear(), pure.expr.getLink()));
  6411. }
  6412. else
  6413. {
  6414. IHqlExpression * base = queryStripCasts(pure.expr);
  6415. newType.setown(makeReferenceModifier(newType.getClear()));
  6416. tgt.expr.setown(createValue(no_cast, newType.getClear(), LINK(base)));
  6417. }
  6418. return;
  6419. }
  6420. }
  6421. }
  6422. doBuildCastViaTemp(ctx, to, pure, tgt);
  6423. return;
  6424. case type_unicode:
  6425. case type_varunicode:
  6426. case type_utf8:
  6427. if ((from->getTypeCode() == to->getTypeCode()) && (to->getSize() == UNKNOWN_LENGTH))
  6428. {
  6429. tgt.set(pure);
  6430. return;
  6431. }
  6432. doBuildCastViaTemp(ctx, to, pure, tgt);
  6433. return;
  6434. default:
  6435. doBuildCastViaTemp(ctx, to, pure, tgt);
  6436. return;
  6437. }
  6438. if (funcName)
  6439. op.setown(bindTranslatedFunctionCall(funcName, args));
  6440. if (!op)
  6441. op.setown(ensureExprType(pure.expr, to));
  6442. if (queryUnqualifiedType(op->queryType()) != queryUnqualifiedType(to))
  6443. {
  6444. OwnedHqlExpr translated = createTranslated(op);
  6445. OwnedHqlExpr cast = ensureExprType(translated, to);
  6446. buildExpr(ctx, cast, tgt);
  6447. }
  6448. else
  6449. tgt.expr.setown(op.getClear());
  6450. }
  6451. //---------------------------------------------------------------------------
  6452. //-- no_char_length --
  6453. //NB: parameter is expression to take length of - not the length node.
  6454. IHqlExpression * HqlCppTranslator::doBuildCharLength(BuildCtx & ctx, IHqlExpression * expr)
  6455. {
  6456. CHqlBoundExpr bound;
  6457. buildCachedExpr(ctx, expr, bound);
  6458. return getBoundLength(bound);
  6459. }
  6460. //---------------------------------------------------------------------------
  6461. //-- no_choose
  6462. void HqlCppTranslator::doBuildChoose(BuildCtx & ctx, const CHqlBoundTarget * target, IHqlExpression * expr)
  6463. {
  6464. CHqlBoundExpr test;
  6465. BuildCtx subctx(ctx);
  6466. buildExpr(subctx, expr->queryChild(0), test);
  6467. IHqlStmt * stmt = subctx.addSwitch(test.expr);
  6468. unsigned max = expr->numChildren()-1;
  6469. unsigned idx;
  6470. for (idx = 1; idx < max; idx++)
  6471. {
  6472. OwnedHqlExpr branch = getSizetConstant(idx);
  6473. subctx.addCase(stmt, branch);
  6474. buildExprOrAssign(subctx, target, expr->queryChild(idx), NULL);
  6475. }
  6476. IHqlExpression * defaultExpr = expr->queryChild(max);
  6477. if (target || !isNullAction(defaultExpr))
  6478. {
  6479. subctx.addDefault(stmt);
  6480. buildExprOrAssign(subctx, target, defaultExpr, NULL);
  6481. }
  6482. }
  6483. void HqlCppTranslator::doBuildAssignChoose(BuildCtx & ctx, const CHqlBoundTarget & target, IHqlExpression * expr)
  6484. {
  6485. unsigned max = expr->numChildren()-1;
  6486. unsigned idx;
  6487. bool allConstant = true;
  6488. for (idx = 1; idx < max; idx++)
  6489. {
  6490. if (!expr->queryChild(idx)->queryValue())
  6491. allConstant = false;
  6492. }
  6493. if (allConstant)
  6494. {
  6495. //MORE: Need to calculate the correct type.
  6496. HqlExprArray args;
  6497. args.append(*LINK(expr->queryChild(0)));
  6498. for (idx = 1; idx < max; idx++)
  6499. {
  6500. IHqlExpression * v1 = createConstant(createIntValue(idx, LINK(unsignedType)));
  6501. IHqlExpression * v2 = expr->queryChild(idx);
  6502. ITypeInfo * type = v2->queryType();
  6503. args.append(* createValue(no_mapto, LINK(type), v1, LINK(v2)));
  6504. }
  6505. args.append(*LINK(expr->queryChild(max)));
  6506. OwnedHqlExpr caseExpr = createValue(no_case, expr->getType(), args);
  6507. buildExprAssign(ctx, target, caseExpr);
  6508. }
  6509. else
  6510. {
  6511. doBuildChoose(ctx, &target, expr);
  6512. }
  6513. }
  6514. //---------------------------------------------------------------------------
  6515. //-- compare (no_eq,no_ne,no_lt,no_gt,no_le,no_ge) --
  6516. //Are the arguments scalar? If so return the scalar arguments.
  6517. static bool getIsScalarCompare(IHqlExpression * expr, SharedHqlExpr & left, SharedHqlExpr & right)
  6518. {
  6519. left.set(expr->queryChild(0));
  6520. right.set(expr->queryChild(1));
  6521. ITypeInfo * lType = left->queryType();
  6522. ITypeInfo * rType = right->queryType();
  6523. type_t leftTypeCode = lType ? lType->getTypeCode() : type_row;
  6524. type_t rightTypeCode = rType ? rType->getTypeCode() : type_row;
  6525. assertex(leftTypeCode == rightTypeCode);
  6526. switch (leftTypeCode)
  6527. {
  6528. case type_row:
  6529. case type_set:
  6530. return false;
  6531. default:
  6532. return true;
  6533. }
  6534. }
  6535. void HqlCppTranslator::doBuildAssignCompare(BuildCtx & ctx, const CHqlBoundTarget & target, IHqlExpression * expr)
  6536. {
  6537. OwnedHqlExpr left;
  6538. OwnedHqlExpr right;
  6539. if (getIsScalarCompare(expr, left, right))
  6540. {
  6541. doBuildExprAssign(ctx, target, expr);
  6542. return;
  6543. }
  6544. //Comparing a row - calculate the ordering and then compare against zero...
  6545. IHqlExpression * compare = createValue(no_order, LINK(signedType), LINK(left), LINK(right));
  6546. OwnedHqlExpr temp = createBoolExpr(expr->getOperator(), compare, getZero());
  6547. buildExprAssign(ctx, target, temp);
  6548. }
  6549. //---------------------------------------------------------------------------
  6550. //-- no_concat --
  6551. void HqlCppTranslator::buildConcatFArgs(HqlExprArray & args, BuildCtx & ctx, const HqlExprArray & values, ITypeInfo * argType)
  6552. {
  6553. ForEachItemIn(idx, values)
  6554. {
  6555. IHqlExpression * cur = &values.item(idx);
  6556. OwnedHqlExpr value = getCastParameter(cur, argType, "");
  6557. CHqlBoundExpr bound;
  6558. buildCachedExpr(ctx, value, bound);
  6559. //The function takes ... so the length must be 4 bytes or the stack
  6560. //gets out of sync...
  6561. OwnedHqlExpr length = getBoundLength(bound);
  6562. args.append(*ensureExprType(length, unsignedType));
  6563. args.append(*getElementPointer(bound.expr));
  6564. }
  6565. args.append(*createConstant(signedType->castFrom(true, -1))); // correct size terminator
  6566. }
  6567. void HqlCppTranslator::doBuildVarLengthConcatF(BuildCtx & ctx, const CHqlBoundTarget & target, const HqlExprArray & values)
  6568. {
  6569. ITypeInfo * targetType = target.queryType();
  6570. Linked<ITypeInfo> argType = targetType;
  6571. type_t ttc = targetType->getTypeCode();
  6572. HqlExprArray args;
  6573. IIdAtom * func;
  6574. if (ttc == type_varstring)
  6575. {
  6576. func = concatVStrId;
  6577. argType.setown(makeStringType(UNKNOWN_LENGTH, LINK(targetType->queryCharset()), LINK(targetType->queryCollation())));
  6578. }
  6579. else if (ttc == type_unicode)
  6580. {
  6581. args.append(*target.length.getLink());
  6582. func = concatUnicodeId;
  6583. }
  6584. else if (ttc == type_utf8)
  6585. {
  6586. args.append(*target.length.getLink());
  6587. func = concatUtf8Id;
  6588. }
  6589. else if (ttc == type_varunicode)
  6590. {
  6591. func = concatVUnicodeId;
  6592. argType.setown(makeUnicodeType(UNKNOWN_LENGTH, targetType->queryLocale()));
  6593. }
  6594. else
  6595. {
  6596. args.append(*target.length.getLink());
  6597. func = concatId;
  6598. }
  6599. IHqlExpression * tgt = createValue(no_address, makeVoidType(), LINK(target.expr));
  6600. if (ttc == type_data)
  6601. tgt = createValue(no_implicitcast, makePointerType(makeStringType(UNKNOWN_LENGTH, NULL, NULL)), tgt);
  6602. args.append(*tgt);
  6603. buildConcatFArgs(args, ctx, values, argType);
  6604. callProcedure(ctx, func, args);
  6605. }
  6606. bool HqlCppTranslator::doBuildFixedLengthConcatF(BuildCtx & ctx, const CHqlBoundTarget & target, const HqlExprArray & values)
  6607. {
  6608. ITypeInfo * targetType = target.queryType();
  6609. Owned<ITypeInfo> argType = getStretchedType(UNKNOWN_LENGTH, targetType);
  6610. type_t ttc = targetType->getTypeCode();
  6611. HqlExprArray args;
  6612. IIdAtom * func = NULL;
  6613. OwnedHqlExpr fill;
  6614. switch (ttc)
  6615. {
  6616. case type_varstring:
  6617. func = concatVStrFId;
  6618. argType.setown(makeStringType(UNKNOWN_LENGTH, LINK(targetType->queryCharset()), LINK(targetType->queryCollation())));
  6619. break;
  6620. case type_varunicode:
  6621. func = concatVUnicodeFId;
  6622. argType.setown(makeUnicodeType(UNKNOWN_LENGTH, targetType->queryLocale()));
  6623. break;
  6624. case type_unicode:
  6625. func = concatUnicodeFId;
  6626. break;
  6627. case type_string:
  6628. func = concatStrFId;
  6629. if (targetType->queryCharset()->queryName() == ebcdicAtom)
  6630. fill.setown(getSizetConstant('@'));
  6631. else
  6632. fill.setown(getSizetConstant(' '));
  6633. break;
  6634. case type_data:
  6635. func = concatStrFId;
  6636. fill.setown(getSizetConstant(0));
  6637. break;
  6638. }
  6639. if (func)
  6640. {
  6641. args.append(*getSizetConstant(targetType->getStringLen()));
  6642. args.append(*getPointer(target.expr));
  6643. if (fill)
  6644. args.append(*LINK(fill));
  6645. buildConcatFArgs(args, ctx, values, argType);
  6646. callProcedure(ctx, func, args);
  6647. return true;
  6648. }
  6649. return false;
  6650. }
  6651. void HqlCppTranslator::doBuildAssignConcat(BuildCtx & ctx, const CHqlBoundTarget & target, IHqlExpression * expr)
  6652. {
  6653. ITypeInfo * targetType = target.queryType();
  6654. ICharsetInfo * targetCharset = targetType->queryCharset();
  6655. type_t ttc = targetType->getTypeCode();
  6656. switch (ttc)
  6657. {
  6658. case type_unicode:
  6659. case type_varunicode:
  6660. case type_utf8:
  6661. break;
  6662. case type_string:
  6663. case type_varstring:
  6664. case type_data:
  6665. if (!queryDefaultTranslation(targetCharset, expr->queryType()->queryCharset()))
  6666. break;
  6667. //fallthrough
  6668. default:
  6669. doBuildExprAssign(ctx, target, expr);
  6670. return;
  6671. }
  6672. HqlExprArray values;
  6673. expr->unwindList(values, no_concat);
  6674. //Combine adjacent constants (not folded previously because of the tree structure)
  6675. //Not optimized in the folder since it can mess up cse evaluation
  6676. for (unsigned i=0; i < values.ordinality()-1; i++)
  6677. {
  6678. IValue * firstValue = values.item(i).queryValue();
  6679. if (firstValue)
  6680. {
  6681. Linked<IValue> combinedValue = firstValue;
  6682. while (i+1 < values.ordinality())
  6683. {
  6684. IValue * nextValue = values.item(i+1).queryValue();
  6685. if (!nextValue)
  6686. break;
  6687. combinedValue.setown(concatValues(combinedValue, nextValue));
  6688. values.remove(i+1);
  6689. }
  6690. if (combinedValue->queryType()->getStringLen() == 0)
  6691. {
  6692. values.remove(i);
  6693. i--; // not nice, but will be incremented before comparison so safe
  6694. }
  6695. else if (combinedValue != firstValue)
  6696. values.replace(*createConstant(combinedValue.getClear()), i);
  6697. }
  6698. }
  6699. if (!target.isFixedSize())
  6700. {
  6701. doBuildVarLengthConcatF(ctx, target, values);
  6702. }
  6703. else if (!doBuildFixedLengthConcatF(ctx, target, values))
  6704. {
  6705. Owned<ITypeInfo> varType = getStretchedType(UNKNOWN_LENGTH, targetType);
  6706. OwnedHqlExpr castValue = createValue(no_concat, LINK(varType), values);
  6707. doBuildExprAssign(ctx, target, castValue);
  6708. }
  6709. }
  6710. void HqlCppTranslator::doBuildAssignExecuteWhen(BuildCtx & ctx, const CHqlBoundTarget & target, IHqlExpression * expr)
  6711. {
  6712. IHqlExpression * value = expr->queryChild(0);
  6713. IHqlExpression * action = expr->queryChild(1);
  6714. if (expr->hasAttribute(beforeAtom))
  6715. {
  6716. buildStmt(ctx, action);
  6717. buildExprAssign(ctx, target, value);
  6718. }
  6719. else if (expr->hasAttribute(failureAtom))
  6720. {
  6721. BuildCtx tryctx(ctx);
  6722. tryctx.addTry();
  6723. buildExprAssign(tryctx, target, value);
  6724. BuildCtx catchctx(ctx);
  6725. catchctx.addCatch(NULL);
  6726. buildStmt(catchctx, action);
  6727. catchctx.addThrow(NULL);
  6728. }
  6729. else
  6730. {
  6731. buildExprAssign(ctx, target, value);
  6732. buildStmt(ctx, action);
  6733. }
  6734. }
  6735. //---------------------------------------------------------------------------
  6736. //-- no_div --
  6737. // also used for no_modulus
  6738. void HqlCppTranslator::doBuildExprDivide(BuildCtx & ctx, IHqlExpression * expr, CHqlBoundExpr & tgt)
  6739. {
  6740. if (expr->queryType()->getTypeCode() == type_decimal)
  6741. {
  6742. doBuildExprArith(ctx, expr, tgt);
  6743. return;
  6744. }
  6745. IHqlExpression * left = expr->queryChild(0);
  6746. IHqlExpression * right = expr->queryChild(1);
  6747. assertex(left->queryType() == right->queryType());
  6748. IValue * value = right->queryValue();
  6749. if (value)
  6750. {
  6751. Owned<IValue> zero = right->queryType()->castFrom(false, 0);
  6752. int cmp = value->compare(zero);
  6753. if (cmp == 0)
  6754. {
  6755. OwnedHqlExpr eZero = createConstant(LINK(zero));
  6756. doBuildDivideByZero(ctx, NULL, eZero, &tgt);
  6757. }
  6758. else
  6759. doBuildPureSubExpr(ctx, expr, tgt);
  6760. }
  6761. else
  6762. {
  6763. buildTempExpr(ctx, expr, tgt);
  6764. }
  6765. }
  6766. void HqlCppTranslator::doBuildDivideByZero(BuildCtx & ctx, const CHqlBoundTarget * target, IHqlExpression * zero, CHqlBoundExpr * bound)
  6767. {
  6768. //Always assign something to bound - even if it is replaced further down.
  6769. if (bound)
  6770. buildExpr(ctx, zero, *bound);
  6771. switch (options.divideByZeroAction)
  6772. {
  6773. case DBZzero:
  6774. if (target)
  6775. assignBound(ctx, *target, zero);
  6776. break;
  6777. case DBZfail:
  6778. {
  6779. HqlExprArray noArgs;
  6780. buildFunctionCall(ctx, failDivideByZeroId, noArgs);
  6781. break;
  6782. }
  6783. case DBZnan:
  6784. {
  6785. LinkedHqlExpr nan = zero;
  6786. if (zero->queryType()->getTypeCode() == type_real)
  6787. {
  6788. HqlExprArray noArgs;
  6789. nan.setown(bindFunctionCall(createRealInfId, noArgs));
  6790. }
  6791. if (target)
  6792. assignBound(ctx, *target, nan);
  6793. else
  6794. buildExpr(ctx, nan, *bound);
  6795. break;
  6796. }
  6797. default:
  6798. throwUnexpected();
  6799. }
  6800. }
  6801. void HqlCppTranslator::doBuildAssignDivide(BuildCtx & ctx, const CHqlBoundTarget & target, IHqlExpression * expr)
  6802. {
  6803. if (expr->queryType()->getTypeCode() == type_decimal)
  6804. {
  6805. doBuildExprAssign(ctx, target, expr);
  6806. return;
  6807. }
  6808. IHqlExpression * left = expr->queryChild(0);
  6809. IHqlExpression * right = expr->queryChild(1);
  6810. assertex(left->queryType() == right->queryType());
  6811. CHqlBoundExpr lhs,rhs;
  6812. buildExpr(ctx, left, lhs);
  6813. buildSimpleExpr(ctx, right, rhs);
  6814. IHqlExpression * divisor = rhs.expr.get();
  6815. OwnedHqlExpr pureExpr = createValue(expr->getOperator(), left->getType(), lhs.expr.getClear(), LINK(divisor));
  6816. IValue * zero = pureExpr->queryType()->castFrom(false, 0);
  6817. OwnedHqlExpr eZero = createConstant(zero);
  6818. IValue * value = rhs.expr->queryValue();
  6819. if (value)
  6820. {
  6821. int cmp = value->compare(eZero->queryValue());
  6822. if (cmp == 0)
  6823. doBuildDivideByZero(ctx, &target, eZero, NULL);
  6824. else
  6825. assignBound(ctx, target, pureExpr);
  6826. }
  6827. else
  6828. {
  6829. BuildCtx subctx(ctx);
  6830. IHqlStmt * stmt = subctx.addFilter(divisor);
  6831. assignBound(subctx, target, pureExpr);
  6832. subctx.selectElse(stmt);
  6833. doBuildDivideByZero(subctx, &target, eZero, NULL);
  6834. }
  6835. }
  6836. //---------------------------------------------------------------------------
  6837. //-- no_if --
  6838. bool HqlCppTranslator::ifRequiresAssignment(BuildCtx & ctx, IHqlExpression * expr)
  6839. {
  6840. IHqlExpression * trueExpr = expr->queryChild(1);
  6841. IHqlExpression * falseExpr = expr->queryChild(2);
  6842. if (requiresTemp(ctx, trueExpr, true) || requiresTemp(ctx, falseExpr, true) || expr->queryType()->getSize() == UNKNOWN_LENGTH)
  6843. return true;
  6844. if (trueExpr->queryType() != falseExpr->queryType() && isStringType(expr->queryType()))
  6845. return true;
  6846. type_t tc = expr->queryType()->getTypeCode();
  6847. if ((tc == type_decimal) || (tc == type_data))
  6848. return true;
  6849. return false;
  6850. }
  6851. void HqlCppTranslator::doBuildAssignIf(BuildCtx & ctx, const CHqlBoundTarget & target, IHqlExpression * expr)
  6852. {
  6853. CHqlBoundExpr bound;
  6854. if (expr->isPure() && ctx.getMatchExpr(expr, bound))
  6855. {
  6856. assign(ctx, target, bound);
  6857. return;
  6858. }
  6859. if (!ifRequiresAssignment(ctx, expr))
  6860. {
  6861. doBuildExprAssign(ctx, target, expr);
  6862. return;
  6863. }
  6864. IHqlExpression * trueExpr = expr->queryChild(1);
  6865. IHqlExpression * falseExpr = expr->queryChild(2);
  6866. BuildCtx subctx(ctx);
  6867. CHqlBoundExpr cond;
  6868. buildCachedExpr(subctx, expr->queryChild(0), cond);
  6869. IHqlStmt * test = subctx.addFilter(cond.expr);
  6870. buildExprAssign(subctx, target, trueExpr);
  6871. subctx.selectElse(test);
  6872. buildExprAssign(subctx, target, falseExpr);
  6873. }
  6874. void HqlCppTranslator::doBuildExprIf(BuildCtx & ctx, IHqlExpression * expr, CHqlBoundExpr & tgt)
  6875. {
  6876. if (ifRequiresAssignment(ctx, expr))
  6877. {
  6878. buildTempExpr(ctx, expr, tgt);
  6879. return;
  6880. }
  6881. IHqlExpression * trueExpr = expr->queryChild(1);
  6882. IHqlExpression * falseExpr = expr->queryChild(2);
  6883. //Length should not be conditional...
  6884. CHqlBoundExpr cond;
  6885. CHqlBoundExpr boundTrue;
  6886. CHqlBoundExpr boundFalse;
  6887. buildCachedExpr(ctx, expr->queryChild(0), cond);
  6888. buildCachedExpr(ctx, trueExpr, boundTrue);
  6889. buildCachedExpr(ctx, falseExpr, boundFalse);
  6890. //true and false should have same type...
  6891. tgt.expr.setown(createValue(no_if, expr->getType(), cond.expr.getClear(), boundTrue.expr.getClear(), boundFalse.expr.getClear()));
  6892. }
  6893. void HqlCppTranslator::doBuildStmtIf(BuildCtx & ctx, IHqlExpression * expr)
  6894. {
  6895. BuildCtx subctx(ctx);
  6896. CHqlBoundExpr cond;
  6897. buildCachedExpr(subctx, expr->queryChild(0), cond);
  6898. IHqlStmt * test = subctx.addFilter(cond.expr);
  6899. optimizeBuildActionList(subctx, expr->queryChild(1));
  6900. IHqlExpression * elseExpr = queryRealChild(expr, 2);
  6901. if (elseExpr && elseExpr->getOperator() != no_null)
  6902. {
  6903. subctx.selectElse(test);
  6904. optimizeBuildActionList(subctx, elseExpr);
  6905. }
  6906. }
  6907. //---------------------------------------------------------------------------
  6908. //-- no_intformat --
  6909. IHqlExpression * HqlCppTranslator::createFormatCall(IIdAtom * func, IHqlExpression * expr)
  6910. {
  6911. HqlExprArray args;
  6912. unsigned max = expr->numChildren();
  6913. unsigned idx;
  6914. for (idx=0; idx < max; idx++)
  6915. {
  6916. IHqlExpression * cur = expr->queryChild(idx);;
  6917. args.append(*LINK(cur));
  6918. }
  6919. return bindFunctionCall(func, args);
  6920. }
  6921. void HqlCppTranslator::doBuildExprFormat(IIdAtom * func, BuildCtx & ctx, IHqlExpression * expr, CHqlBoundExpr & tgt)
  6922. {
  6923. OwnedHqlExpr call = createFormatCall(func, expr);
  6924. buildExpr(ctx, call, tgt);
  6925. }
  6926. void HqlCppTranslator::doBuildAssignFormat(IIdAtom * func, BuildCtx & ctx, const CHqlBoundTarget & target, IHqlExpression * expr)
  6927. {
  6928. OwnedHqlExpr call = createFormatCall(func, expr);
  6929. buildExprAssign(ctx, target, call);
  6930. }
  6931. //---------------------------------------------------------------------------
  6932. void HqlCppTranslator::doBuildAssignToXmlorJson(BuildCtx & ctx, const CHqlBoundTarget & target, IHqlExpression * expr)
  6933. {
  6934. IHqlExpression * row = expr->queryChild(0);
  6935. HqlExprArray args;
  6936. args.append(*buildMetaParameter(row));
  6937. args.append(*LINK(row));
  6938. args.append(*getSizetConstant(XWFtrim|XWFopt|XWFnoindent));
  6939. node_operator op = expr->getOperator();
  6940. OwnedHqlExpr call = bindFunctionCall((op==no_tojson) ? ctxGetRowJsonId : ctxGetRowXmlId, args);
  6941. buildExprAssign(ctx, target, call);
  6942. }
  6943. //---------------------------------------------------------------------------
  6944. void HqlCppTranslator::processCppBodyDirectives(IHqlExpression * expr)
  6945. {
  6946. ForEachChild(i, expr)
  6947. {
  6948. IHqlExpression * cur = expr->queryChild(i);
  6949. if (cur->isAttribute())
  6950. {
  6951. IAtom * name = cur->queryName();
  6952. if (name == linkAtom)
  6953. {
  6954. //MORE: Add code to add the argument to the linker options.
  6955. }
  6956. else if (name == libraryAtom)
  6957. {
  6958. StringBuffer libraryName;
  6959. getStringValue(libraryName, cur->queryChild(0));
  6960. if (libraryName.length())
  6961. useLibrary(libraryName.str());
  6962. }
  6963. else if (name == sourceAtom)
  6964. {
  6965. StringBuffer sourceName;
  6966. getStringValue(sourceName, cur->queryChild(0));
  6967. if (sourceName.length())
  6968. code->useSourceFile(sourceName.str());
  6969. }
  6970. }
  6971. }
  6972. }
  6973. void HqlCppTranslator::doBuildExprEmbedBody(BuildCtx & ctx, IHqlExpression * expr, CHqlBoundExpr * tgt)
  6974. {
  6975. if (expr->hasAttribute(_disallowed_Atom))
  6976. throwError(HQLERR_EmbeddedCppNotAllowed);
  6977. processCppBodyDirectives(expr);
  6978. IHqlExpression *languageAttr = expr->queryAttribute(languageAtom);
  6979. if (languageAttr)
  6980. {
  6981. UNIMPLEMENTED; // It's not clear if this can ever happen - perhaps a parameterless function that used EMBED ?
  6982. }
  6983. StringBuffer text;
  6984. expr->queryChild(0)->queryValue()->getUTF8Value(text);
  6985. text.setLength(cleanupEmbeddedCpp(text.length(), (char*)text.str()));
  6986. OwnedHqlExpr quoted = createQuoted(text.str(), expr->getType());
  6987. if (tgt)
  6988. {
  6989. ITypeInfo * type = expr->queryType();
  6990. assertex(type->getTypeCode() == type_varstring || type->getSize() != UNKNOWN_LENGTH);
  6991. tgt->expr.set(quoted);
  6992. }
  6993. else
  6994. ctx.addExpr(quoted);
  6995. }
  6996. //---------------------------------------------------------------------------
  6997. //-- no_index --
  6998. IHqlExpression * getSimpleListIndex(BuildCtx & ctx, IHqlExpression * expr)
  6999. {
  7000. IHqlExpression * index = expr->queryChild(1);
  7001. if (!index->isConstant())
  7002. return NULL;
  7003. OwnedHqlExpr set = normalizeListCasts(expr->queryChild(0));
  7004. switch (set->getOperator())
  7005. {
  7006. case no_null:
  7007. case no_list:
  7008. break;
  7009. default:
  7010. return NULL;
  7011. }
  7012. OwnedHqlExpr folded = foldHqlExpression(index);
  7013. assertex(folded->queryValue());
  7014. __int64 which = folded->queryValue()->getIntValue();
  7015. if ((which > 0) && (which <= set->numChildren()))
  7016. return LINK(set->queryChild((unsigned)which-1));
  7017. return createNullExpr(expr);
  7018. }
  7019. void HqlCppTranslator::doBuildExprIndex(BuildCtx & ctx, IHqlExpression * expr, CHqlBoundExpr & tgt)
  7020. {
  7021. OwnedHqlExpr simple = getSimpleListIndex(ctx, expr);
  7022. if (simple)
  7023. buildExpr(ctx, simple, tgt);
  7024. else
  7025. {
  7026. OwnedHqlExpr simpleList = simplifyFixedLengthList(expr->queryChild(0));
  7027. Owned<IHqlCppSetCursor> cursor = createSetSelector(ctx, simpleList);
  7028. cursor->buildExprSelect(ctx, expr, tgt);
  7029. }
  7030. }
  7031. void HqlCppTranslator::doBuildAssignIndex(BuildCtx & ctx, const CHqlBoundTarget & target, IHqlExpression * expr)
  7032. {
  7033. OwnedHqlExpr simple = getSimpleListIndex(ctx, expr);
  7034. if (simple)
  7035. buildExprAssign(ctx, target, simple);
  7036. else
  7037. {
  7038. OwnedHqlExpr simpleList = simplifyFixedLengthList(expr->queryChild(0));
  7039. Owned<IHqlCppSetCursor> cursor = createSetSelector(ctx, simpleList);
  7040. cursor->buildAssignSelect(ctx, target, expr);
  7041. }
  7042. }
  7043. //---------------------------------------------------------------------------
  7044. //-- no_list --
  7045. bool isComplexSet(ITypeInfo * type, bool isConstant)
  7046. {
  7047. ITypeInfo * childType = type->queryChildType();
  7048. if (!childType)
  7049. return false;
  7050. switch (childType->getTypeCode())
  7051. {
  7052. case type_alien:
  7053. return true;
  7054. case type_string:
  7055. case type_qstring:
  7056. case type_data:
  7057. case type_unicode:
  7058. case type_varstring:
  7059. case type_varunicode:
  7060. return isUnknownSize(childType) && !isConstant;
  7061. case type_utf8:
  7062. case type_swapint:
  7063. case type_packedint:
  7064. return true;
  7065. case type_int:
  7066. switch (childType->getSize())
  7067. {
  7068. case 3: case 5: case 6: case 7:
  7069. return true;
  7070. }
  7071. return false;
  7072. }
  7073. return false;
  7074. }
  7075. bool isComplexSet(IHqlExpression * expr)
  7076. {
  7077. return isComplexSet(expr->queryType(), isConstantSet(expr));
  7078. }
  7079. bool isConstantSet(IHqlExpression * expr)
  7080. {
  7081. unsigned max = expr->numChildren();
  7082. unsigned idx;
  7083. for (idx = 0; idx < max; idx++)
  7084. {
  7085. IHqlExpression * child = expr->queryChild(idx);
  7086. if (!child->queryValue())
  7087. return false;
  7088. }
  7089. return true;
  7090. }
  7091. bool createUnknownLengthStringSet(MemoryBuffer & target, IHqlExpression * set)
  7092. {
  7093. ITypeInfo * elemType = set->queryType()->queryChildType();
  7094. type_t tc = elemType->getTypeCode();
  7095. ForEachChild(i, set)
  7096. {
  7097. IHqlExpression * cur = set->queryChild(i);
  7098. IValue * curValue = cur->queryValue();
  7099. if (!curValue)
  7100. return false;
  7101. size32_t sizeValue = curValue->getSize();
  7102. if ((tc != type_varstring) && (tc != type_varunicode))
  7103. {
  7104. size32_t lenValue = curValue->queryType()->getStringLen();
  7105. rtlWriteInt4(target.reserve(sizeof(size32_t)), lenValue);
  7106. }
  7107. curValue->toMem(target.reserve(sizeValue));
  7108. }
  7109. return true;
  7110. }
  7111. void HqlCppTranslator::doBuildExprConstList(BuildCtx & ctx, IHqlExpression * expr, CHqlBoundExpr & tgt)
  7112. {
  7113. BuildCtx declareCtx(*code, literalAtom);
  7114. if (!declareCtx.getMatchExpr(expr, tgt))
  7115. {
  7116. ITypeInfo * type = expr->queryType();
  7117. Linked<ITypeInfo> elementType = type->queryChildType();
  7118. if (!elementType)
  7119. throwError(HQLERR_NullSetCannotGenerate);
  7120. Owned<ITypeInfo> transferType;
  7121. OwnedHqlExpr table;
  7122. OwnedHqlExpr declare;
  7123. unsigned numElements = expr->numChildren();
  7124. LinkedHqlExpr values = expr;
  7125. if ((isTypePassedByAddress(elementType) && ((elementType->getTypeCode() != type_varstring) || isUnknownSize(elementType))))
  7126. {
  7127. if (isUnknownSize(elementType))
  7128. {
  7129. MemoryBuffer serialized;
  7130. bool ok = createUnknownLengthStringSet(serialized, values);
  7131. assertex(ok);
  7132. Owned<IValue> value = createDataValue(serialized.toByteArray(), serialized.length());
  7133. OwnedHqlExpr constValue = createConstant(LINK(value));
  7134. OwnedHqlExpr initializer = createValue(no_create_initializer, constValue->getType(), LINK(constValue));
  7135. Owned<ITypeInfo> declareType = makeConstantModifier(constValue->getType());
  7136. declare.setown(declareCtx.getTempDeclare(declareType, initializer));
  7137. ITypeInfo * arrayType = makeReferenceModifier(makeSetType(LINK(elementType)));
  7138. table.setown(createValue(no_typetransfer, arrayType, LINK(declare)));
  7139. tgt.length.setown(getSizetConstant(serialized.length()));
  7140. }
  7141. else if (elementType->isReference())
  7142. {
  7143. // use a var string type to get better C++ generated...
  7144. transferType.set(elementType);
  7145. elementType.setown(makeVarStringType(UNKNOWN_LENGTH));
  7146. }
  7147. else
  7148. {
  7149. // for string, data and qstring we need to initialize the array with a list of characters instead of
  7150. // a cstring e.g., char[][2] = { { 'a','b' }, { 'c', 'd' } };
  7151. HqlExprArray newValues;
  7152. ForEachChild(idx, expr)
  7153. {
  7154. IHqlExpression * next = expr->queryChild(idx);
  7155. newValues.append(*createValue(no_create_initializer, next->getType(), LINK(next)));
  7156. }
  7157. values.setown(createValue(no_list, makeSetType(LINK(elementType)), newValues));
  7158. }
  7159. }
  7160. if (!declare)
  7161. {
  7162. Owned<ITypeInfo> t = makeConstantModifier(makeArrayType(LINK(elementType), numElements));
  7163. declare.setown(declareCtx.getTempDeclare(t, values));
  7164. if (transferType)
  7165. {
  7166. ITypeInfo * arrayType = makeArrayType(LINK(transferType), numElements);
  7167. table.setown(createValue(no_typetransfer, arrayType, LINK(declare)));
  7168. }
  7169. }
  7170. tgt.count.setown(getSizetConstant(numElements));
  7171. tgt.expr.set(table ? table : declare);
  7172. //make sure tables get added before any global functions
  7173. declareCtx.associateExpr(expr, tgt);
  7174. if (options.spanMultipleCpp)
  7175. {
  7176. BuildCtx protoctx(*code, mainprototypesAtom);
  7177. protoctx.addDeclareExternal(declare);
  7178. }
  7179. }
  7180. }
  7181. void HqlCppTranslator::doBuildExprDynList(BuildCtx & ctx, IHqlExpression * expr, CHqlBoundExpr & tgt)
  7182. {
  7183. if (!ctx.getMatchExpr(expr, tgt))
  7184. {
  7185. ITypeInfo * type = expr->queryType();
  7186. ITypeInfo * elementType = type->queryChildType();
  7187. unsigned max = expr->numChildren();
  7188. //MORE: What if this is an array of variable length strings?
  7189. Owned<ITypeInfo> t = makeArrayType(LINK(elementType), max);
  7190. IHqlExpression * table = ctx.getTempDeclare(t, NULL);
  7191. // new code - should really use a selector...
  7192. unsigned idx;
  7193. CHqlBoundTarget boundTarget;
  7194. for (idx = 0; idx < max; idx++)
  7195. {
  7196. IHqlExpression * child = expr->queryChild(idx);
  7197. boundTarget.expr.setown(createValue(no_index, LINK(elementType), LINK(table), createConstant((int)idx)));
  7198. buildExprAssign(ctx, boundTarget, child);
  7199. }
  7200. tgt.count.setown(getSizetConstant(max));
  7201. tgt.expr.setown(table);
  7202. ctx.associateExpr(expr, tgt);
  7203. }
  7204. }
  7205. void HqlCppTranslator::doBuildExprList(BuildCtx & ctx, IHqlExpression * expr, CHqlBoundExpr & tgt)
  7206. {
  7207. ITypeInfo * type = expr->queryType();
  7208. switch (type->getTypeCode())
  7209. {
  7210. case type_set:
  7211. case type_array:
  7212. {
  7213. LinkedHqlExpr values = expr;
  7214. //MORE: Also alien data types and other weird things...
  7215. //if (childType->getSize() == UNKNOWN_LENGTH)
  7216. if (expr->numChildren() == 0)
  7217. {
  7218. tgt.length.setown(getSizetConstant(0));
  7219. tgt.expr.setown(createValue(no_nullptr, makeReferenceModifier(LINK(type))));
  7220. return;
  7221. }
  7222. else if (isComplexSet(expr))
  7223. {
  7224. buildTempExpr(ctx, expr, tgt);
  7225. return;
  7226. }
  7227. if (isConstantSet(expr))
  7228. doBuildExprConstList(ctx, values, tgt);
  7229. else
  7230. doBuildExprDynList(ctx, values, tgt);
  7231. tgt.isAll.set(queryBoolExpr(false));
  7232. }
  7233. break;
  7234. default:
  7235. throwUnexpectedX("This type of list not supported yet");
  7236. }
  7237. }
  7238. void HqlCppTranslator::doBuildAssignList(BuildCtx & ctx, const CHqlBoundTarget & target, IHqlExpression * _expr)
  7239. {
  7240. OwnedHqlExpr expr = simplifyFixedLengthList(_expr);
  7241. node_operator op = expr->getOperator();
  7242. assertex(op == no_list);
  7243. ITypeInfo * type = expr->queryType();
  7244. switch (type->getTypeCode())
  7245. {
  7246. case type_set:
  7247. case type_array:
  7248. break;
  7249. default:
  7250. throwUnexpectedX("This type of list not supported yet");
  7251. }
  7252. //This is an assignment, a non-constant set would end up creating two temporaries.
  7253. unsigned numItems = expr->numChildren();
  7254. ITypeInfo * elementType = type->queryChildType();
  7255. if (((numItems > 0) && (numItems < 3) && !isUnknownSize(elementType)) || isComplexSet(expr) || !isConstantSet(expr))
  7256. {
  7257. Owned<IHqlCppSetBuilder> builder = createTempSetBuilder(target.queryType()->queryChildType(), target.isAll);
  7258. builder->buildDeclare(ctx);
  7259. buildSetAssign(ctx, builder, expr);
  7260. builder->buildFinish(ctx, target);
  7261. }
  7262. else if (numItems == 0)
  7263. {
  7264. CHqlBoundExpr temp;
  7265. buildExpr(ctx, expr, temp);
  7266. if (target.isAll)
  7267. {
  7268. if (temp.isAll)
  7269. ctx.addAssign(target.isAll, temp.isAll);
  7270. else
  7271. ctx.addAssign(target.isAll, queryBoolExpr(false));
  7272. }
  7273. ctx.addAssign(target.length, temp.length);
  7274. ctx.addAssign(target.expr, temp.expr);
  7275. }
  7276. else
  7277. {
  7278. OwnedHqlExpr cast = ensureExprType(expr, target.queryType());
  7279. OwnedHqlExpr simpleCast = simplifyFixedLengthList(cast);
  7280. // can do a direct assignment without any casts
  7281. doBuildExprAssign(ctx, target, simpleCast);
  7282. }
  7283. }
  7284. void HqlCppTranslator::doBuildExprAll(BuildCtx & ctx, IHqlExpression * expr, CHqlBoundExpr & tgt)
  7285. {
  7286. tgt.isAll.set(queryBoolExpr(true));
  7287. tgt.length.setown(getSizetConstant(0));
  7288. tgt.expr.setown(createQuoted("0", makeSetType(NULL)));
  7289. }
  7290. void HqlCppTranslator::doBuildAssignAll(BuildCtx & ctx, const CHqlBoundTarget & target, IHqlExpression * expr)
  7291. {
  7292. CHqlBoundExpr temp;
  7293. buildExpr(ctx, expr, temp);
  7294. ctx.addAssign(target.isAll, temp.isAll);
  7295. ctx.addAssign(target.length, temp.length);
  7296. ctx.addAssign(target.expr, temp.expr);
  7297. }
  7298. //---------------------------------------------------------------------------
  7299. //-- no_not --
  7300. void HqlCppTranslator::doBuildExprNot(BuildCtx & ctx, IHqlExpression * expr, CHqlBoundExpr & tgt)
  7301. {
  7302. assertex(expr->queryChild(0)->isBoolean());
  7303. CHqlBoundExpr bound;
  7304. buildExpr(ctx, expr->queryChild(0), bound);
  7305. tgt.expr.setown(getInverse(bound.expr));
  7306. }
  7307. //---------------------------------------------------------------------------
  7308. //-- no_or --
  7309. IHqlExpression * HqlCppTranslator::convertOrToAnd(IHqlExpression * expr)
  7310. {
  7311. bool invert = true;
  7312. if (expr->getOperator() == no_not)
  7313. {
  7314. invert = false;
  7315. expr = expr->queryChild(0);
  7316. }
  7317. assertex(expr->getOperator() == no_or);
  7318. HqlExprArray original, inverted;
  7319. expr->unwindList(original, no_or);
  7320. ForEachItemIn(idx, original)
  7321. inverted.append(*getInverse(&original.item(idx)));
  7322. IHqlExpression * ret = createValue(no_and, makeBoolType(), inverted);
  7323. if (invert)
  7324. ret = createValue(no_not, makeBoolType(), ret);
  7325. return ret;
  7326. }
  7327. //---------------------------------------------------------------------------
  7328. // no_unicodeorder
  7329. void HqlCppTranslator::doBuildAssignUnicodeOrder(BuildCtx & ctx, const CHqlBoundTarget & target, IHqlExpression * expr)
  7330. {
  7331. CHqlBoundExpr lhs, rhs, locale, strength;
  7332. buildCachedExpr(ctx, expr->queryChild(0), lhs);
  7333. buildCachedExpr(ctx, expr->queryChild(1), rhs);
  7334. buildCachedExpr(ctx, expr->queryChild(2), locale);
  7335. buildCachedExpr(ctx, expr->queryChild(3), strength);
  7336. Owned<IHqlExpression> op;
  7337. HqlExprArray args;
  7338. ITypeInfo * realType = lhs.expr->queryType()->queryPromotedType();
  7339. switch(realType->getTypeCode())
  7340. {
  7341. case type_unicode:
  7342. args.append(*getBoundLength(lhs));
  7343. args.append(*getElementPointer(lhs.expr));
  7344. args.append(*getBoundLength(rhs));
  7345. args.append(*getElementPointer(rhs.expr));
  7346. args.append(*getElementPointer(locale.expr));
  7347. args.append(*strength.expr.getLink());
  7348. op.setown(bindTranslatedFunctionCall(compareUnicodeUnicodeStrengthId, args));
  7349. break;
  7350. case type_varunicode:
  7351. args.append(*getElementPointer(lhs.expr));
  7352. args.append(*getElementPointer(rhs.expr));
  7353. args.append(*getElementPointer(locale.expr));
  7354. args.append(*strength.expr.getLink());
  7355. op.setown(bindTranslatedFunctionCall(compareVUnicodeVUnicodeStrengthId, args));
  7356. break;
  7357. case type_utf8:
  7358. args.append(*getBoundLength(lhs));
  7359. args.append(*getElementPointer(lhs.expr));
  7360. args.append(*getBoundLength(rhs));
  7361. args.append(*getElementPointer(rhs.expr));
  7362. args.append(*getElementPointer(locale.expr));
  7363. args.append(*strength.expr.getLink());
  7364. op.setown(bindTranslatedFunctionCall(compareUtf8Utf8StrengthId, args));
  7365. break;
  7366. default:
  7367. UNIMPLEMENTED;
  7368. }
  7369. assignBound(ctx, target, op);
  7370. }
  7371. //---------------------------------------------------------------------------
  7372. //-- no_order --
  7373. static void buildIteratorIsValid(BuildCtx & ctx, IHqlExpression * iter, IHqlExpression * row, CHqlBoundExpr & bound)
  7374. {
  7375. bound.expr.set(row);
  7376. }
  7377. void HqlCppTranslator::buildIteratorFirst(BuildCtx & ctx, IHqlExpression * iter, IHqlExpression * row)
  7378. {
  7379. StringBuffer s;
  7380. generateExprCpp(s, row).append(" = (byte*)");
  7381. generateExprCpp(s, iter).append(".first();");
  7382. ctx.addQuoted(s);
  7383. }
  7384. void HqlCppTranslator::buildIteratorNext(BuildCtx & ctx, IHqlExpression * iter, IHqlExpression * row)
  7385. {
  7386. StringBuffer s;
  7387. generateExprCpp(s, row).append(" = (byte*)");
  7388. generateExprCpp(s, iter).append(".next();");
  7389. ctx.addQuoted(s);
  7390. }
  7391. void HqlCppTranslator::doBuildAssignCompareRow(BuildCtx & ctx, EvaluateCompareInfo & info, IHqlExpression * left, IHqlExpression * right)
  7392. {
  7393. HqlExprArray leftValues, rightValues;
  7394. expandRowOrder(left->queryNormalizedSelector(), left->queryRecord(), leftValues, false);
  7395. expandRowOrder(right->queryNormalizedSelector(), right->queryRecord(), rightValues, false);
  7396. optimizeOrderValues(leftValues, rightValues, false);
  7397. doBuildAssignCompare(ctx, info, leftValues, rightValues, true, false); //MORE: ,no_break,true
  7398. }
  7399. void HqlCppTranslator::doBuildAssignCompareTable(BuildCtx & ctx, EvaluateCompareInfo & info, IHqlExpression * left, IHqlExpression * right)
  7400. {
  7401. ITypeInfo * targetType = info.target.queryType();
  7402. OwnedHqlExpr zeroTarget = createConstant(targetType->castFrom(true, 0));
  7403. OwnedHqlExpr plusOne = createConstant(targetType->castFrom(true, +1));
  7404. OwnedHqlExpr minusOne = createConstant(targetType->castFrom(true, -1));
  7405. // cmp = 0;
  7406. assignBound(ctx, info.target, zeroTarget);
  7407. BuildCtx subctx(ctx);
  7408. subctx.addGroup(); // stop bound cursors leaking outside the testing block.
  7409. // i1 iter1; i1.first();
  7410. HqlExprAttr leftIter, leftRow;
  7411. Owned<IHqlCppDatasetCursor> cursor = createDatasetSelector(subctx, left);
  7412. cursor->buildIterateClass(subctx, leftIter, leftRow);
  7413. buildIteratorFirst(subctx, leftIter, leftRow);
  7414. // i2; forEachIn(i2); {
  7415. CHqlBoundExpr isValid;
  7416. BuildCtx loopctx(subctx);
  7417. if (buildDatasetIterate(loopctx, right, true))
  7418. {
  7419. bindTableCursor(loopctx, left, leftRow);
  7420. // if (!i1.isValid()) { cmp = -1; break; }
  7421. buildIteratorIsValid(loopctx, leftIter, leftRow, isValid);
  7422. OwnedHqlExpr test = createValue(no_not, makeBoolType(), isValid.expr.getClear());
  7423. BuildCtx moreRightCtx(loopctx);
  7424. moreRightCtx.addFilter(test);
  7425. if (info.actionIfDiffer == return_stmt)
  7426. {
  7427. if (info.isEqualityCompare())
  7428. {
  7429. OwnedHqlExpr returnValue = info.getEqualityReturnValue();
  7430. moreRightCtx.addReturn(returnValue);
  7431. }
  7432. else
  7433. moreRightCtx.addReturn(minusOne);
  7434. }
  7435. else
  7436. {
  7437. buildExprAssign(moreRightCtx, info.target, minusOne);
  7438. moreRightCtx.addBreak();
  7439. }
  7440. //Now do the comparison....
  7441. {
  7442. EvaluateCompareInfo childInfo(info);
  7443. if (childInfo.actionIfDiffer == break_stmt)
  7444. childInfo.actionIfDiffer = null_stmt;
  7445. //***childInfo??
  7446. doBuildAssignCompareRow(loopctx, info, left, right);
  7447. }
  7448. if (info.actionIfDiffer != return_stmt)
  7449. {
  7450. // if (cmp != 0) break;
  7451. BuildCtx donectx(loopctx);
  7452. donectx.addFilter(info.target.expr);
  7453. donectx.addQuotedLiteral("break;");
  7454. }
  7455. // i1.next();
  7456. buildIteratorNext(loopctx, leftIter, leftRow);
  7457. }
  7458. buildIteratorIsValid(subctx, leftIter, leftRow, isValid);
  7459. if (info.actionIfDiffer == return_stmt)
  7460. {
  7461. //if (i1.isValid) return +1;
  7462. BuildCtx moreLeftCtx(subctx);
  7463. moreLeftCtx.addFilter(isValid.expr);
  7464. if (info.isEqualityCompare())
  7465. {
  7466. OwnedHqlExpr returnValue = info.getEqualityReturnValue();
  7467. moreLeftCtx.addReturn(returnValue);
  7468. }
  7469. else
  7470. moreLeftCtx.addReturn(plusOne);
  7471. }
  7472. else
  7473. {
  7474. //if (cmp == 0 && i1.isValid) cmp = +1;
  7475. OwnedHqlExpr cmp = createBoolExpr(no_and, createBoolExpr(no_eq, LINK(info.target.expr), LINK(zeroTarget)), LINK(isValid.expr));
  7476. BuildCtx moreLeftCtx(subctx);
  7477. moreLeftCtx.addFilter(cmp);
  7478. buildExprAssign(moreLeftCtx, info.target, plusOne);
  7479. BuildCtx tailctx(ctx);
  7480. if (info.actionIfDiffer == break_stmt)
  7481. {
  7482. tailctx.addFilter(info.target.expr);
  7483. tailctx.addBreak();
  7484. }
  7485. }
  7486. }
  7487. void HqlCppTranslator::expandRowOrder(IHqlExpression * selector, IHqlExpression * record, HqlExprArray & values, bool isRow)
  7488. {
  7489. ForEachChild(idx, record)
  7490. {
  7491. IHqlExpression * field = record->queryChild(idx);
  7492. switch (field->getOperator())
  7493. {
  7494. case no_ifblock:
  7495. expandRowOrder(selector, field->queryChild(1), values, isRow);
  7496. break;
  7497. case no_record:
  7498. expandRowOrder(selector, field, values, isRow);
  7499. break;
  7500. case no_field:
  7501. {
  7502. OwnedHqlExpr selected;
  7503. if (isRow)
  7504. selected.setown(createNewSelectExpr(LINK(selector), LINK(field)));
  7505. else
  7506. selected.setown(createSelectExpr(LINK(selector), LINK(field)));
  7507. if (field->isDatarow())
  7508. expandRowOrder(selected, field->queryRecord(), values, false);
  7509. else
  7510. values.append(*LINK(selected));
  7511. break;
  7512. }
  7513. }
  7514. }
  7515. }
  7516. void HqlCppTranslator::expandSimpleOrder(IHqlExpression * left, IHqlExpression * right, HqlExprArray & leftValues, HqlExprArray & rightValues)
  7517. {
  7518. while ((left->getOperator() == no_negate) && (right->getOperator() == no_negate))
  7519. {
  7520. IHqlExpression * temp = right->queryChild(0);
  7521. right = left->queryChild(0);
  7522. left = temp;
  7523. }
  7524. if (left == right)
  7525. {
  7526. //Weird code is here just so I can force some strange exceptions in the regression suite.
  7527. IHqlExpression * cur = left;
  7528. if (cur->getOperator() == no_alias)
  7529. cur = cur->queryChild(0);
  7530. if (cur->getOperator() != no_nofold)
  7531. return;
  7532. }
  7533. if (left->isDatarow())
  7534. {
  7535. IHqlExpression * leftRecord = left->queryRecord();
  7536. IHqlExpression * rightRecord = right->queryRecord();
  7537. assertex(right->isDatarow());
  7538. if (leftRecord != rightRecord)
  7539. {
  7540. OwnedHqlExpr leftSerialRecord = getSerializedForm(leftRecord, diskAtom);
  7541. OwnedHqlExpr rightSerialRecord = getSerializedForm(rightRecord, diskAtom);
  7542. assertex(leftSerialRecord == rightSerialRecord);
  7543. }
  7544. expandRowOrder(left, leftRecord, leftValues, !isActiveRow(left) && (left->getOperator() != no_select));
  7545. expandRowOrder(right, rightRecord, rightValues, !isActiveRow(right) && (right->getOperator() != no_select));
  7546. }
  7547. else
  7548. {
  7549. leftValues.append(*LINK(left));
  7550. rightValues.append(*LINK(right));
  7551. }
  7552. }
  7553. void HqlCppTranslator::expandOrder(IHqlExpression * expr, HqlExprArray & leftValues, HqlExprArray & rightValues, SharedHqlExpr & defaultValue)
  7554. {
  7555. OwnedHqlExpr left = normalizeListCasts(expr->queryChild(0));
  7556. OwnedHqlExpr right = normalizeListCasts(expr->queryChild(1));
  7557. if ((isFixedLengthList(left) || isNullList(left)) && (isFixedLengthList(right) || isNullList(right)))
  7558. {
  7559. unsigned maxLeft = left->numChildren();
  7560. unsigned maxRight = right->numChildren();
  7561. unsigned max = std::min(maxLeft, maxRight);
  7562. for (unsigned i=0; i < max; i++)
  7563. expandSimpleOrder(left->queryChild(i), right->queryChild(i), leftValues, rightValues);
  7564. if (maxLeft != maxRight)
  7565. defaultValue.setown(createConstant(signedType->castFrom(true, ((maxLeft > maxRight) ? +1 : -1))));
  7566. }
  7567. else
  7568. expandSimpleOrder(left, right, leftValues, rightValues);
  7569. }
  7570. IHqlExpression * HqlCppTranslator::querySimpleOrderSelector(IHqlExpression * expr)
  7571. {
  7572. if (expr->getOperator() != no_select)
  7573. return NULL;
  7574. return queryDatasetCursor(expr);
  7575. }
  7576. static unsigned getMemcmpSize(IHqlExpression * left, IHqlExpression * right, bool isEqualityCompare)
  7577. {
  7578. ITypeInfo * leftType = left->queryType();
  7579. if (!leftType)
  7580. return 0;
  7581. if (!isSameBasicType(leftType, right->queryType()))
  7582. return 0;
  7583. unsigned size = leftType->getSize();
  7584. switch (leftType->getTypeCode())
  7585. {
  7586. case type_bigendianint:
  7587. case type_boolean:
  7588. return size;
  7589. case type_littleendianint:
  7590. if ((size == 1) && !leftType->isSigned())
  7591. return 1;
  7592. if (isEqualityCompare)
  7593. return size;
  7594. break;
  7595. case type_string:
  7596. case type_data:
  7597. case type_qstring:
  7598. if (size != UNKNOWN_LENGTH)
  7599. return size;
  7600. break;
  7601. }
  7602. return 0;
  7603. }
  7604. void HqlCppTranslator::optimizeOrderValues(HqlExprArray & leftValues, HqlExprArray & rightValues, bool isEqualityCompare)
  7605. {
  7606. unsigned max = leftValues.ordinality();
  7607. if (max <= 1)
  7608. return;
  7609. for (unsigned i=0; i < max-1; i++)
  7610. {
  7611. IHqlExpression * curFirstLeft = &leftValues.item(i);
  7612. IHqlExpression * curFirstRight = &rightValues.item(i);
  7613. IHqlExpression * leftSel = querySimpleOrderSelector(curFirstLeft);
  7614. IHqlExpression * rightSel = querySimpleOrderSelector(curFirstRight);
  7615. if (!leftSel || !rightSel)
  7616. continue;
  7617. unsigned compareSize = getMemcmpSize(curFirstLeft, curFirstRight, isEqualityCompare);
  7618. if (!compareSize)
  7619. continue;
  7620. IHqlExpression * nextLeft = &leftValues.item(i+1);
  7621. IHqlExpression * nextRight = &rightValues.item(i+1);
  7622. if (querySimpleOrderSelector(nextLeft) != leftSel || querySimpleOrderSelector(nextRight) != rightSel ||
  7623. (getMemcmpSize(nextLeft, nextRight, isEqualityCompare) == 0))
  7624. continue;
  7625. //Worth iterating the selectors...
  7626. RecordSelectIterator leftIter(leftSel->queryRecord(), leftSel);
  7627. ForEach(leftIter)
  7628. if (leftIter.query() == curFirstLeft)
  7629. break;
  7630. if (!leftIter.isValid() || leftIter.isInsideIfBlock())
  7631. continue;
  7632. RecordSelectIterator rightIter(rightSel->queryRecord(), rightSel);
  7633. ForEach(rightIter)
  7634. if (rightIter.query() == curFirstRight)
  7635. break;
  7636. if (!rightIter.isValid() || rightIter.isInsideIfBlock())
  7637. continue;
  7638. unsigned j; // linux wants it declared outside of 'for'
  7639. for (j=i+1; j < max; j++)
  7640. {
  7641. if (!leftIter.next() || leftIter.isInsideIfBlock())
  7642. break;
  7643. if (!rightIter.next() || rightIter.isInsideIfBlock())
  7644. break;
  7645. IHqlExpression * nextLeft = &leftValues.item(j);
  7646. IHqlExpression * nextRight = &rightValues.item(j);
  7647. if (leftIter.query() != nextLeft || rightIter.query() != nextRight)
  7648. break;
  7649. unsigned thisSize = getMemcmpSize(nextLeft, nextRight, isEqualityCompare);
  7650. if (!thisSize)
  7651. break;
  7652. compareSize += thisSize;
  7653. }
  7654. if (j != i+1)
  7655. {
  7656. IHqlExpression * newLeft = createValue(no_typetransfer, makeStringType(compareSize), LINK(curFirstLeft));
  7657. IHqlExpression * newRight = createValue(no_typetransfer, makeStringType(compareSize), LINK(curFirstRight));
  7658. leftValues.replace(*newLeft, i);
  7659. rightValues.replace(*newRight, i);
  7660. leftValues.removen(i+1, j-(i+1));
  7661. rightValues.removen(i+1, j-(i+1));
  7662. max -= (j - (i+1));
  7663. }
  7664. }
  7665. }
  7666. inline IHqlExpression * createSignedConstant(__int64 value)
  7667. {
  7668. return createConstant(signedType->castFrom(true, value));
  7669. }
  7670. static IHqlExpression * convertAllToInteger(IHqlExpression * allExpr)
  7671. {
  7672. IValue * allValue = allExpr->queryValue();
  7673. if (allValue)
  7674. return createSignedConstant(allValue->getBoolValue() ? 1 : 0);
  7675. return createValue(no_if, LINK(signedType), LINK(allExpr), createSignedConstant(1), createSignedConstant(0));
  7676. }
  7677. void HqlCppTranslator::doBuildAssignCompareElement(BuildCtx & ctx, EvaluateCompareInfo & info, IHqlExpression * left, IHqlExpression * right, bool isFirst, bool isLast)
  7678. {
  7679. if (left->getOperator() == no_if && right->getOperator() == no_if && left->queryChild(0) == right->queryChild(0))
  7680. {
  7681. BuildCtx subctx(ctx);
  7682. IHqlStmt * filter = buildFilterViaExpr(subctx, left->queryChild(0));
  7683. doBuildAssignCompareElement(subctx, info, left->queryChild(1), right->queryChild(1), isFirst, false);
  7684. if ((isFirst && (info.actionIfDiffer != return_stmt)) || left->queryChild(2) != right->queryChild(2))
  7685. {
  7686. subctx.selectElse(filter);
  7687. doBuildAssignCompareElement(subctx, info, left->queryChild(2), right->queryChild(2), isFirst, false);
  7688. }
  7689. return;
  7690. }
  7691. if (left == right)
  7692. {
  7693. //Can happen from conditions expanded above
  7694. if (isFirst)
  7695. {
  7696. if (info.actionIfDiffer != return_stmt)
  7697. buildExprAssign(ctx, info.target, queryZero());
  7698. }
  7699. return;
  7700. }
  7701. ITypeInfo * leftType = left->queryType();
  7702. type_t tc;
  7703. if (leftType)
  7704. tc = leftType->getTypeCode();
  7705. else
  7706. tc = type_set;
  7707. CHqlBoundExpr lhs,rhs;
  7708. bool useMemCmp = false;
  7709. switch (tc)
  7710. {
  7711. case type_dictionary:
  7712. {
  7713. //You can't iterate dictionaries, so convert to datasets first. A bit of a silly comparison anyway
  7714. OwnedHqlExpr dsLeft = createDataset(no_datasetfromdictionary, LINK(left));
  7715. OwnedHqlExpr dsRight = createDataset(no_datasetfromdictionary, LINK(right));
  7716. doBuildAssignCompareTable(ctx, info, dsLeft, dsRight);
  7717. return;
  7718. }
  7719. case type_table:
  7720. case type_groupedtable:
  7721. doBuildAssignCompareTable(ctx, info, left, right);
  7722. return;
  7723. case type_row:
  7724. doBuildAssignCompareRow(ctx, info, left, right);
  7725. return;
  7726. case type_bigendianint:
  7727. {
  7728. //MORE: Compare big endian integers with a memcmp
  7729. if (hasAddress(ctx, left) && hasAddress(ctx, right) && isSameBasicType(leftType, right->queryType()))
  7730. {
  7731. buildAddress(ctx, left, lhs);
  7732. buildAddress(ctx, right, rhs);
  7733. useMemCmp = true;
  7734. break;
  7735. }
  7736. Owned<ITypeInfo> type = makeIntType(leftType->getSize(), leftType->isSigned());
  7737. OwnedHqlExpr intLeft = createValue(no_implicitcast, type.getLink(), LINK(left));
  7738. OwnedHqlExpr intRight = createValue(no_implicitcast, type.getLink(), LINK(right));
  7739. buildCachedExpr(ctx, intLeft, lhs);
  7740. buildCachedExpr(ctx, intRight, rhs);
  7741. break;
  7742. }
  7743. case type_string:
  7744. case type_data:
  7745. case type_qstring:
  7746. {
  7747. OwnedHqlExpr simpleLeft = getSimplifyCompareArg(left);
  7748. OwnedHqlExpr simpleRight = getSimplifyCompareArg(right);
  7749. buildCachedExpr(ctx, simpleLeft, lhs);
  7750. buildCachedExpr(ctx, simpleRight, rhs);
  7751. break;
  7752. }
  7753. default:
  7754. buildCachedExpr(ctx, left, lhs);
  7755. buildCachedExpr(ctx, right, rhs);
  7756. break;
  7757. }
  7758. ITypeInfo * realType = lhs.queryType()->queryPromotedType();
  7759. tc = realType->getTypeCode();
  7760. IHqlExpression * op = NULL;
  7761. switch (tc)
  7762. {
  7763. //MORE: Should common up with comparison code...
  7764. case type_string:
  7765. case type_data:
  7766. case type_qstring:
  7767. {
  7768. HqlExprArray args;
  7769. IIdAtom * func;
  7770. if (lhs.length || rhs.length || needVarStringCompare(realType, rhs.queryType()->queryPromotedType()))
  7771. {
  7772. //MORE: This does not cope with different padding characters...
  7773. func = queryStrCompareFunc(realType);
  7774. args.append(*getBoundLength(lhs));
  7775. args.append(*getElementPointer(lhs.expr));
  7776. args.append(*getBoundLength(rhs));
  7777. args.append(*getElementPointer(rhs.expr));
  7778. }
  7779. else
  7780. {
  7781. func = memcmpId;
  7782. args.append(*getElementPointer(lhs.expr));
  7783. args.append(*getElementPointer(rhs.expr));
  7784. args.append(*getSizetConstant(realType->getSize()));
  7785. }
  7786. op = bindTranslatedFunctionCall(func, args);
  7787. break;
  7788. }
  7789. case type_unicode:
  7790. {
  7791. HqlExprArray args;
  7792. assertex(haveCommonLocale(leftType, right->queryType()));
  7793. char const * locale = str(getCommonLocale(leftType, right->queryType()));
  7794. args.append(*getBoundLength(lhs));
  7795. args.append(*getElementPointer(lhs.expr));
  7796. args.append(*getBoundLength(rhs));
  7797. args.append(*getElementPointer(rhs.expr));
  7798. args.append(*createConstant(locale));
  7799. op = bindTranslatedFunctionCall(compareUnicodeUnicodeId, args);
  7800. break;
  7801. }
  7802. case type_varunicode:
  7803. {
  7804. HqlExprArray args;
  7805. assertex(haveCommonLocale(leftType, right->queryType()));
  7806. char const * locale = str(getCommonLocale(leftType, right->queryType()));
  7807. args.append(*getElementPointer(lhs.expr));
  7808. args.append(*getElementPointer(rhs.expr));
  7809. args.append(*createConstant(locale));
  7810. op = bindTranslatedFunctionCall(compareVUnicodeVUnicodeId, args);
  7811. break;
  7812. }
  7813. case type_utf8:
  7814. {
  7815. HqlExprArray args;
  7816. assertex(haveCommonLocale(leftType, right->queryType()));
  7817. char const * locale = str(getCommonLocale(leftType, right->queryType()));
  7818. args.append(*getBoundLength(lhs));
  7819. args.append(*getElementPointer(lhs.expr));
  7820. args.append(*getBoundLength(rhs));
  7821. args.append(*getElementPointer(rhs.expr));
  7822. args.append(*createConstant(locale));
  7823. op = bindTranslatedFunctionCall(compareUtf8Utf8Id, args);
  7824. break;
  7825. }
  7826. case type_varstring:
  7827. {
  7828. HqlExprArray args;
  7829. args.append(*getElementPointer(lhs.expr));
  7830. args.append(*getElementPointer(rhs.expr));
  7831. op = bindTranslatedFunctionCall(compareVStrVStrId, args);
  7832. break;
  7833. }
  7834. case type_decimal:
  7835. {
  7836. HqlExprArray args;
  7837. if (!isPushed(lhs) && !isPushed(rhs) && (leftType->queryPromotedType() == right->queryType()->queryPromotedType()))
  7838. {
  7839. args.append(*getSizetConstant(leftType->queryPromotedType()->getSize()));
  7840. args.append(*getPointer(lhs.expr));
  7841. args.append(*getPointer(rhs.expr));
  7842. op = bindTranslatedFunctionCall(leftType->isSigned() ? DecCompareDecimalId : DecCompareUDecimalId, args);
  7843. }
  7844. else
  7845. {
  7846. bool pushedLhs = ensurePushed(ctx, lhs);
  7847. bool pushedRhs = ensurePushed(ctx, rhs);
  7848. //NB: Arguments could be pushed in opposite order 1 <=> x *2
  7849. if (pushedLhs && !pushedRhs)
  7850. op = bindTranslatedFunctionCall(DecDistinctRId, args);
  7851. else
  7852. op = bindTranslatedFunctionCall(DecDistinctId, args);
  7853. }
  7854. break;
  7855. }
  7856. case type_set:
  7857. case type_array:
  7858. {
  7859. //compare all
  7860. OwnedHqlExpr leftAll = lhs.getIsAll();
  7861. OwnedHqlExpr rightAll = rhs.getIsAll();
  7862. assertex(leftAll && rightAll);
  7863. if (leftAll != rightAll)
  7864. {
  7865. if (leftAll->queryValue() && rightAll->queryValue())
  7866. {
  7867. op = createConstant(leftAll->queryValue()->getIntValue() - rightAll->queryValue()->getIntValue());
  7868. break;
  7869. }
  7870. if (getIntValue(leftAll, false) || getIntValue(rightAll, false))
  7871. {
  7872. op = createValue(no_sub, LINK(signedType), convertAllToInteger(leftAll), convertAllToInteger(rightAll));
  7873. break;
  7874. }
  7875. }
  7876. if (lhs.expr != rhs.expr)
  7877. {
  7878. HqlExprArray args;
  7879. args.append(*getBoundLength(lhs));
  7880. args.append(*getElementPointer(lhs.expr));
  7881. args.append(*getBoundLength(rhs));
  7882. args.append(*getElementPointer(rhs.expr));
  7883. op = bindTranslatedFunctionCall(compareDataDataId, args);
  7884. }
  7885. if (leftAll != rightAll)
  7886. {
  7887. IHqlExpression * orderAll = createValue(no_sub, LINK(signedType), convertAllToInteger(leftAll), convertAllToInteger(rightAll));
  7888. if (op)
  7889. {
  7890. IHqlExpression * cond = NULL;
  7891. if (!getIntValue(leftAll, true))
  7892. {
  7893. if (!getIntValue(rightAll, true))
  7894. cond = NULL;
  7895. else
  7896. cond = LINK(rightAll);
  7897. }
  7898. else
  7899. {
  7900. if (!getIntValue(rightAll, true))
  7901. cond = LINK(leftAll);
  7902. else
  7903. cond = createBoolExpr(no_or, LINK(leftAll), LINK(rightAll));
  7904. }
  7905. if (cond)
  7906. op = createValue(no_if, LINK(signedType), cond, orderAll, op);
  7907. else
  7908. orderAll->Release();
  7909. }
  7910. else
  7911. op = orderAll;
  7912. }
  7913. if (!op)
  7914. op = getZero();
  7915. break;
  7916. }
  7917. case type_boolean:
  7918. case type_swapint:
  7919. case type_packedint:
  7920. case type_int:
  7921. if (!useMemCmp && !info.isEqualityCompare() && (realType->getSize() < signedType->getSize()))
  7922. {
  7923. op = createValue(no_sub, LINK(signedType),
  7924. createValue(no_implicitcast, LINK(signedType), lhs.expr.getLink()),
  7925. createValue(no_implicitcast, LINK(signedType), rhs.expr.getLink()));
  7926. break;
  7927. }
  7928. //fall through
  7929. default:
  7930. if (useMemCmp)
  7931. {
  7932. HqlExprArray args;
  7933. args.append(*lhs.expr.getLink());
  7934. args.append(*rhs.expr.getLink());
  7935. args.append(*getSizetConstant(leftType->getSize()));
  7936. op = bindTranslatedFunctionCall(memcmpId, args);
  7937. }
  7938. else
  7939. {
  7940. if (info.isEqualityCompare())
  7941. {
  7942. op = createBoolExpr(no_ne, LINK(lhs.expr), LINK(rhs.expr));
  7943. }
  7944. else
  7945. {
  7946. ensureSimpleExpr(ctx, lhs);
  7947. ensureSimpleExpr(ctx, rhs);
  7948. OwnedHqlExpr testlt = createBoolExpr(no_lt, lhs.expr.getLink(), rhs.expr.getLink());
  7949. OwnedHqlExpr retlt = createIntConstant(-1);
  7950. OwnedHqlExpr testgt = createBoolExpr(no_gt, lhs.expr.getLink(), rhs.expr.getLink());
  7951. OwnedHqlExpr retgt = createIntConstant(+1);
  7952. if (info.actionIfDiffer == return_stmt)
  7953. {
  7954. BuildCtx subctx1(ctx);
  7955. subctx1.addFilter(testlt);
  7956. subctx1.addReturn(retlt);
  7957. BuildCtx subctx2(ctx);
  7958. subctx2.addFilter(testgt);
  7959. subctx2.addReturn(retgt);
  7960. return;
  7961. }
  7962. else
  7963. {
  7964. // generate (a < b ? -1 : (a > b ? +1 : 0))
  7965. op = createValue(no_if, LINK(signedType),
  7966. LINK(testlt), LINK(retlt),
  7967. createValue(no_if, LINK(signedType), LINK(testgt), LINK(retgt), getZero()));
  7968. }
  7969. }
  7970. }
  7971. break;
  7972. }
  7973. OwnedHqlExpr safeReleaseOp = op;
  7974. assertex(op);
  7975. BuildCtx subctx(ctx);
  7976. if (info.isEqualityCompare() && (info.actionIfDiffer == return_stmt))
  7977. {
  7978. subctx.addFilter(op);
  7979. OwnedHqlExpr returnValue = info.getEqualityReturnValue();
  7980. subctx.addReturn(returnValue);
  7981. }
  7982. else
  7983. {
  7984. assignBound(subctx, info.target, op);
  7985. switch (info.actionIfDiffer)
  7986. {
  7987. case break_stmt:
  7988. subctx.addFilter(info.target.expr);
  7989. subctx.addBreak();
  7990. break;
  7991. case return_stmt:
  7992. if (!isLast || info.neverReturnMatch)
  7993. subctx.addFilter(info.target.expr);
  7994. else
  7995. info.alwaysReturns = true;
  7996. subctx.addReturn(info.target.expr);
  7997. break;
  7998. }
  7999. }
  8000. }
  8001. void HqlCppTranslator::doBuildAssignCompare(BuildCtx & ctx, EvaluateCompareInfo & info, HqlExprArray & leftValues, HqlExprArray & rightValues, bool isFirst, bool isOuter)
  8002. {
  8003. assertex(leftValues.ordinality() == rightValues.ordinality());
  8004. Owned<BuildCtx> subctx = new BuildCtx(ctx);
  8005. OwnedHqlExpr compare = createBoolExpr(no_not, LINK(info.target.expr));
  8006. unsigned idx;
  8007. unsigned max = leftValues.ordinality();
  8008. for (idx = 0; idx < max; idx++)
  8009. {
  8010. if ((idx & 7) == 0)
  8011. subctx.setown(new BuildCtx(ctx));
  8012. if (!info.actionIfDiffer && !isFirst)
  8013. subctx->addFilter(compare);
  8014. doBuildAssignCompareElement(*subctx, info, &leftValues.item(idx), &rightValues.item(idx), isFirst, isOuter && (idx == max-1));
  8015. isFirst = false;
  8016. }
  8017. }
  8018. void HqlCppTranslator::doBuildAssignOrder(BuildCtx & ctx, const CHqlBoundTarget & target, IHqlExpression * expr)
  8019. {
  8020. HqlExprArray leftValues, rightValues;
  8021. OwnedHqlExpr defaultValue;
  8022. expandOrder(expr, leftValues, rightValues, defaultValue);
  8023. optimizeOrderValues(leftValues, rightValues, false);
  8024. EvaluateCompareInfo info(no_order);
  8025. info.target.set(target);
  8026. doBuildAssignCompare(ctx, info, leftValues, rightValues, true, false);
  8027. unsigned maxLeft = leftValues.ordinality();
  8028. if (defaultValue)
  8029. {
  8030. if (maxLeft != 0)
  8031. {
  8032. OwnedHqlExpr compare = createBoolExpr(no_not, LINK(target.expr));
  8033. ctx.addFilter(compare);
  8034. }
  8035. buildExprAssign(ctx, target, defaultValue);
  8036. }
  8037. else if (maxLeft == 0)
  8038. buildExprAssign(ctx, target, queryZero());
  8039. }
  8040. void HqlCppTranslator::doBuildReturnCompare(BuildCtx & ctx, IHqlExpression * expr, node_operator op, bool isBoolEquality, bool neverReturnTrue)
  8041. {
  8042. HqlExprArray leftValues, rightValues;
  8043. OwnedHqlExpr defaultValue;
  8044. expandOrder(expr, leftValues, rightValues, defaultValue);
  8045. optimizeOrderValues(leftValues, rightValues, (op == no_eq));
  8046. EvaluateCompareInfo info(op);
  8047. info.actionIfDiffer = return_stmt;
  8048. info.isBoolEquality = isBoolEquality;
  8049. info.neverReturnMatch = (defaultValue != NULL) || neverReturnTrue;
  8050. createTempFor(ctx, expr, info.target);
  8051. doBuildAssignCompare(ctx, info, leftValues, rightValues, true, true);
  8052. assertex(!(neverReturnTrue && info.alwaysReturns));
  8053. if (!info.alwaysReturns && !neverReturnTrue)
  8054. {
  8055. if (info.isBoolEquality)
  8056. {
  8057. OwnedHqlExpr returnValue = createConstant(defaultValue == NULL);
  8058. buildReturn(ctx, returnValue);
  8059. }
  8060. else
  8061. {
  8062. if (defaultValue)
  8063. buildReturn(ctx, defaultValue);
  8064. else
  8065. buildReturn(ctx, queryZero());
  8066. }
  8067. }
  8068. }
  8069. //---------------------------------------------------------------------------
  8070. //-- no_hash --
  8071. class HashCodeCreator
  8072. {
  8073. public:
  8074. HashCodeCreator(HqlCppTranslator & _translator, const CHqlBoundTarget & _target, node_operator _hashKind, bool _optimizeInternal)
  8075. : translator(_translator), target(_target), hashKind(_hashKind), optimizeInternal(_optimizeInternal)
  8076. {
  8077. prevFunc = NULL;
  8078. }
  8079. //Combine calls to the hash function on adjacent memory to minimise the number of calls
  8080. //and the generated code size.
  8081. void buildHash(BuildCtx & ctx, IIdAtom * func, IHqlExpression * length, IHqlExpression * ptr)
  8082. {
  8083. if ((func == hash32DataId) || (func == hash64DataId))
  8084. {
  8085. ptr = stripTranslatedCasts(ptr);
  8086. if (prevFunc)
  8087. {
  8088. if ((prevFunc == func) && rightFollowsLeft(prevPtr, prevLength, ptr))
  8089. {
  8090. prevLength.setown(peepholeAddExpr(prevLength, length));
  8091. return;
  8092. }
  8093. flush(ctx);
  8094. }
  8095. prevFunc = func;
  8096. prevLength.set(length);
  8097. prevPtr.set(ptr);
  8098. return;
  8099. }
  8100. flush(ctx);
  8101. buildCall(ctx, func, length, ptr);
  8102. }
  8103. void beginCondition(BuildCtx & ctx)
  8104. {
  8105. ensureInitialAssigned(ctx);
  8106. flush(ctx);
  8107. }
  8108. void endCondition(BuildCtx & ctx)
  8109. {
  8110. flush(ctx);
  8111. }
  8112. void finish(BuildCtx & ctx)
  8113. {
  8114. flush(ctx);
  8115. ensureInitialAssigned(ctx);
  8116. }
  8117. void setInitialValue(IHqlExpression * expr)
  8118. {
  8119. initialValue.set(expr);
  8120. }
  8121. inline node_operator kind() const { return hashKind; }
  8122. inline bool optimize() const { return optimizeInternal; }
  8123. protected:
  8124. void buildCall(BuildCtx & ctx, IIdAtom * func, IHqlExpression * length, IHqlExpression * ptr)
  8125. {
  8126. if (func == hash32DataId)
  8127. {
  8128. unsigned fixedSize = (unsigned)getIntValue(length, 0);
  8129. switch (fixedSize)
  8130. {
  8131. case 1: func = hash32Data1Id; break;
  8132. case 2: func = hash32Data2Id; break;
  8133. case 3: func = hash32Data3Id; break;
  8134. case 4: func = hash32Data4Id; break;
  8135. case 5: func = hash32Data5Id; break;
  8136. case 6: func = hash32Data6Id; break;
  8137. case 7: func = hash32Data7Id; break;
  8138. case 8: func = hash32Data8Id; break;
  8139. }
  8140. if (func != hash32DataId)
  8141. length = NULL;
  8142. }
  8143. HqlExprArray args;
  8144. if (length)
  8145. args.append(*LINK(length));
  8146. args.append(*LINK(ptr));
  8147. if (initialValue)
  8148. args.append(*initialValue.getClear());
  8149. else
  8150. args.append(*LINK(target.expr));
  8151. CHqlBoundExpr boundHash;
  8152. boundHash.expr.setown(translator.bindTranslatedFunctionCall(func, args));
  8153. translator.assign(ctx, target, boundHash);
  8154. }
  8155. void ensureInitialAssigned(BuildCtx & ctx)
  8156. {
  8157. if (initialValue)
  8158. {
  8159. translator.assignBound(ctx, target, initialValue);
  8160. initialValue.clear();
  8161. }
  8162. }
  8163. void flush(BuildCtx & ctx)
  8164. {
  8165. if (prevFunc)
  8166. {
  8167. buildCall(ctx, prevFunc, prevLength, prevPtr);
  8168. prevFunc = NULL;
  8169. }
  8170. }
  8171. protected:
  8172. HqlCppTranslator & translator;
  8173. const CHqlBoundTarget & target;
  8174. LinkedHqlExpr initialValue;
  8175. node_operator hashKind;
  8176. bool optimizeInternal;
  8177. IIdAtom * prevFunc;
  8178. OwnedHqlExpr prevLength;
  8179. OwnedHqlExpr prevPtr;
  8180. };
  8181. void HqlCppTranslator::doBuildAssignHashCrc(BuildCtx & ctx, const CHqlBoundTarget & target, IHqlExpression * expr)
  8182. {
  8183. IHqlExpression * child = expr->queryChild(0);
  8184. LinkedHqlExpr initialValue = queryZero();
  8185. node_operator op = expr->getOperator();
  8186. if (op == no_hash32)
  8187. initialValue.setown(createConstant(createIntValue(HASH32_INIT, 4, false)));
  8188. else if (op == no_hash64)
  8189. initialValue.setown(createConstant(createIntValue(HASH64_INIT, 8, false)));
  8190. HashCodeCreator creator(*this, target, op, expr->hasAttribute(internalAtom));
  8191. creator.setInitialValue(initialValue);
  8192. if (child->getOperator() != no_sortlist)
  8193. doBuildAssignHashElement(ctx, creator, child);
  8194. else
  8195. {
  8196. unsigned max = child->numChildren();
  8197. unsigned idx;
  8198. for (idx = 0; idx < max; idx++)
  8199. doBuildAssignHashElement(ctx, creator, child->queryChild(idx));
  8200. }
  8201. creator.finish(ctx);
  8202. }
  8203. void HqlCppTranslator::doBuildAssignHashElement(BuildCtx & ctx, HashCodeCreator & creator, IHqlExpression * elem, IHqlExpression * record)
  8204. {
  8205. bool useNewSelector = elem->isDatarow() && ((elem->getOperator() != no_select) || isNewSelector(elem));
  8206. ForEachChild(i, record)
  8207. {
  8208. IHqlExpression * cur = record->queryChild(i);
  8209. switch (cur->getOperator())
  8210. {
  8211. case no_field:
  8212. {
  8213. OwnedHqlExpr selected = useNewSelector ? createNewSelectExpr(LINK(elem), LINK(cur)) : createSelectExpr(LINK(elem), LINK(cur));
  8214. doBuildAssignHashElement(ctx, creator, selected);
  8215. break;
  8216. }
  8217. case no_record:
  8218. doBuildAssignHashElement(ctx, creator, elem, cur);
  8219. break;
  8220. case no_ifblock:
  8221. doBuildAssignHashElement(ctx, creator, elem, cur->queryChild(1));
  8222. break;
  8223. }
  8224. }
  8225. }
  8226. void HqlCppTranslator::doBuildAssignHashElement(BuildCtx & ctx, HashCodeCreator & creator, IHqlExpression * elem)
  8227. {
  8228. if (creator.optimize())
  8229. {
  8230. switch (elem->getOperator())
  8231. {
  8232. case no_if:
  8233. {
  8234. BuildCtx subctx(ctx);
  8235. creator.beginCondition(subctx);
  8236. IHqlStmt * cond = buildFilterViaExpr(subctx, elem->queryChild(0));
  8237. doBuildAssignHashElement(subctx, creator, elem->queryChild(1));
  8238. creator.endCondition(subctx);
  8239. IHqlExpression * elseValue = elem->queryChild(2);
  8240. if (elseValue && elseValue->getOperator() != no_constant)
  8241. {
  8242. subctx.selectElse(cond);
  8243. creator.beginCondition(subctx);
  8244. doBuildAssignHashElement(subctx, creator, elseValue);
  8245. creator.endCondition(subctx);
  8246. }
  8247. return;
  8248. }
  8249. case no_constant:
  8250. return;
  8251. }
  8252. }
  8253. Linked<ITypeInfo> type = elem->queryType()->queryPromotedType(); // skip alien data types, to logical type.
  8254. if (type->getTypeCode() == type_row)
  8255. {
  8256. doBuildAssignHashElement(ctx, creator, elem, elem->queryRecord());
  8257. return;
  8258. }
  8259. IIdAtom * func=NULL;
  8260. switch (creator.kind())
  8261. {
  8262. case no_hash: func = hashDataId; break;
  8263. case no_hash32: func = hash32DataId; break;
  8264. case no_hash64: func = hash64DataId; break;
  8265. case no_crc: func = crcDataId; break;
  8266. }
  8267. CHqlBoundExpr bound;
  8268. OwnedHqlExpr length;
  8269. OwnedHqlExpr ptr;
  8270. bool alreadyTrimmedRight = (elem->getOperator() == no_trim) && (elem->hasAttribute(rightAtom) || !elem->hasAttribute(leftAtom));
  8271. //If this hash is generated internally (e.g., for a dedup) and fixed length, then can simplify the hash calculation
  8272. bool canOptimizeHash = (creator.optimize() && isFixedSize(type));
  8273. bool optimizeTrim = alreadyTrimmedRight || canOptimizeHash;
  8274. switch (type->getTypeCode())
  8275. {
  8276. case type_string:
  8277. {
  8278. if (!optimizeTrim)
  8279. {
  8280. OwnedHqlExpr trimmed = createValue(no_trim, getStretchedType(UNKNOWN_LENGTH, type), LINK(elem));
  8281. buildCachedExpr(ctx, trimmed, bound);
  8282. }
  8283. else
  8284. buildCachedExpr(ctx, elem, bound);
  8285. length.setown(getBoundLength(bound));
  8286. ptr.setown(getElementPointer(bound.expr));
  8287. }
  8288. break;
  8289. case type_unicode:
  8290. {
  8291. if (!optimizeTrim)
  8292. {
  8293. OwnedHqlExpr trimmed = createValue(no_trim, getStretchedType(UNKNOWN_LENGTH, type), LINK(elem));
  8294. buildCachedExpr(ctx, trimmed, bound);
  8295. }
  8296. else
  8297. buildCachedExpr(ctx, elem, bound);
  8298. length.setown(getBoundLength(bound));
  8299. ptr.setown(getElementPointer(bound.expr));
  8300. switch (creator.kind())
  8301. {
  8302. case no_hash: func = hashUnicodeId; break;
  8303. case no_hash32: func = hash32UnicodeId; break;
  8304. case no_hash64: func = hash64UnicodeId; break;
  8305. case no_crc: func = crcUnicodeId; break;
  8306. }
  8307. }
  8308. break;
  8309. case type_utf8:
  8310. {
  8311. if (!optimizeTrim)
  8312. {
  8313. OwnedHqlExpr trimmed = createValue(no_trim, getStretchedType(UNKNOWN_LENGTH, type), LINK(elem));
  8314. buildCachedExpr(ctx, trimmed, bound);
  8315. }
  8316. else
  8317. buildCachedExpr(ctx, elem, bound);
  8318. length.setown(getBoundLength(bound));
  8319. ptr.setown(getElementPointer(bound.expr));
  8320. switch (creator.kind())
  8321. {
  8322. case no_hash: func = hashUtf8Id; break;
  8323. case no_hash32: func = hash32Utf8Id; break;
  8324. case no_hash64: func = hash64Utf8Id; break;
  8325. case no_crc: func = crcUtf8Id; break;
  8326. }
  8327. }
  8328. break;
  8329. case type_data:
  8330. {
  8331. buildCachedExpr(ctx, elem, bound);
  8332. length.setown(getBoundLength(bound));
  8333. ptr.setown(getElementPointer(bound.expr));
  8334. break;
  8335. }
  8336. case type_qstring:
  8337. {
  8338. LinkedHqlExpr exprToHash = elem;
  8339. if (!canOptimizeHash)
  8340. {
  8341. //Always convert to a string so the hash is compatible with a string.
  8342. OwnedHqlExpr cast = ensureExprType(elem, unknownStringType);
  8343. if (alreadyTrimmedRight)
  8344. {
  8345. exprToHash.set(cast);
  8346. }
  8347. else
  8348. {
  8349. OwnedHqlExpr trimmed = createValue(no_trim, LINK(unknownStringType), LINK(cast));
  8350. exprToHash.setown(foldHqlExpression(trimmed));
  8351. }
  8352. }
  8353. buildCachedExpr(ctx, exprToHash, bound);
  8354. length.setown(getBoundSize(bound));
  8355. ptr.setown(getElementPointer(bound.expr));
  8356. break;
  8357. }
  8358. case type_varstring:
  8359. buildCachedExpr(ctx, elem, bound);
  8360. ptr.setown(getElementPointer(bound.expr));
  8361. switch (creator.kind())
  8362. {
  8363. case no_hash: func = hashVStrId; break;
  8364. case no_hash32: func = hash32VStrId; break;
  8365. case no_hash64: func = hash64VStrId; break;
  8366. case no_crc: func = crcVStrId; break;
  8367. }
  8368. break;
  8369. case type_varunicode:
  8370. buildCachedExpr(ctx, elem, bound);
  8371. ptr.setown(getElementPointer(bound.expr));
  8372. switch (creator.kind())
  8373. {
  8374. case no_hash: func = hashVUnicodeId; break;
  8375. case no_hash32: func = hash32VUnicodeId; break;
  8376. case no_hash64: func = hash64VUnicodeId; break;
  8377. case no_crc: func = crcVUnicodeId; break;
  8378. }
  8379. break;
  8380. case type_boolean:
  8381. case type_int:
  8382. case type_swapint:
  8383. case type_real:
  8384. if (creator.optimize() && hasAddress(ctx, elem))
  8385. {
  8386. buildAddress(ctx, elem, bound);
  8387. length.setown(getSizetConstant(type->getSize()));
  8388. ptr.setown(LINK(bound.expr));
  8389. }
  8390. else
  8391. {
  8392. if (!creator.optimize())
  8393. type.setown(makeIntType(8, true));
  8394. OwnedHqlExpr castElem = ensureExprType(elem, type);
  8395. buildTempExpr(ctx, castElem, bound);
  8396. length.setown(getSizetConstant(type->getSize()));
  8397. ptr.setown(getPointer(bound.expr));
  8398. }
  8399. break;
  8400. case type_row:
  8401. throwUnexpected();
  8402. break;
  8403. case type_dictionary:
  8404. case type_groupedtable:
  8405. case type_table:
  8406. //MORE: Should be handle this differently, with an iterator for the link counted rows case?
  8407. //Not sure if that is a good idea - we need to be certain we get the same values with
  8408. //LCR rows enabled and disabled. But this won't be very efficient with lcr rows.
  8409. //fallthrough
  8410. if (creator.optimize() && hasOutOfLineRows(elem->queryType()))
  8411. {
  8412. creator.beginCondition(ctx);
  8413. BuildCtx iterctx(ctx);
  8414. buildDatasetIterate(iterctx, elem, false);
  8415. doBuildAssignHashElement(iterctx, creator, elem->queryNormalizedSelector(), elem->queryRecord());
  8416. creator.endCondition(iterctx);
  8417. return;
  8418. }
  8419. else
  8420. {
  8421. OwnedHqlExpr serialized = ::ensureSerialized(elem, diskAtom);
  8422. buildDataset(ctx, serialized, bound, FormatBlockedDataset);
  8423. length.setown(getBoundSize(bound));
  8424. ptr.setown(getPointer(bound.expr));
  8425. }
  8426. break;
  8427. default:
  8428. buildTempExpr(ctx, elem, bound, FormatBlockedDataset);
  8429. length.setown(getBoundSize(bound));
  8430. ptr.setown(getPointer(bound.expr));
  8431. break;
  8432. }
  8433. creator.buildHash(ctx, func, length, ptr);
  8434. }
  8435. //---------------------------------------------------------------------------
  8436. //-- no_hash --
  8437. void HqlCppTranslator::doBuildAssignHashMd5(BuildCtx & ctx, const CHqlBoundTarget & target, IHqlExpression * expr)
  8438. {
  8439. IHqlExpression * child = expr->queryChild(0);
  8440. Owned<ITypeInfo> stateType = makeDataType(sizeof(md5_state_s));
  8441. //initialize the state object
  8442. CHqlBoundTarget stateTemp;
  8443. CHqlBoundExpr state;
  8444. createTempFor(ctx, stateType, stateTemp, typemod_none, FormatNatural);
  8445. state.setFromTarget(stateTemp);
  8446. OwnedHqlExpr stateExpr = state.getTranslatedExpr();
  8447. HqlExprArray args;
  8448. args.append(*LINK(stateExpr));
  8449. OwnedHqlExpr callInit = bindFunctionCall(hashMd5InitId, args);
  8450. buildStmt(ctx, callInit);
  8451. //Now hash each of the elements in turn.
  8452. if (child->getOperator() != no_sortlist)
  8453. doBuildHashMd5Element(ctx, child, state);
  8454. else
  8455. {
  8456. unsigned max = child->numChildren();
  8457. for (unsigned idx = 0; idx < max; idx++)
  8458. doBuildHashMd5Element(ctx, child->queryChild(idx), state);
  8459. }
  8460. //finalise the md5, and get the result.
  8461. args.append(*LINK(stateExpr));
  8462. OwnedHqlExpr callFinish = bindFunctionCall(hashMd5FinishId, args);
  8463. buildExprAssign(ctx, target, callFinish);
  8464. }
  8465. void HqlCppTranslator::doBuildHashMd5Element(BuildCtx & ctx, IHqlExpression * elem, CHqlBoundExpr & state)
  8466. {
  8467. CHqlBoundExpr bound;
  8468. Linked<ITypeInfo> type = elem->queryType()->queryPromotedType(); // skip alien data types, to logical type.
  8469. HqlExprArray args;
  8470. switch (type->getTypeCode())
  8471. {
  8472. case type_string:
  8473. case type_unicode:
  8474. case type_data:
  8475. case type_qstring:
  8476. case type_varstring:
  8477. case type_varunicode:
  8478. case type_utf8:
  8479. buildExpr(ctx, elem, bound);
  8480. args.append(*getBoundSize(bound));
  8481. args.append(*getElementPointer(bound.expr));
  8482. break;
  8483. case type_int:
  8484. case type_swapint:
  8485. case type_packedint:
  8486. {
  8487. type.setown(makeIntType(8, true));
  8488. OwnedHqlExpr castElem = ensureExprType(elem, type);
  8489. buildTempExpr(ctx, castElem, bound);
  8490. args.append(*getSizetConstant(type->getSize()));
  8491. args.append(*getPointer(bound.expr));
  8492. break;
  8493. }
  8494. default:
  8495. buildTempExpr(ctx, elem, bound);
  8496. args.append(*getSizetConstant(type->getSize()));
  8497. args.append(*getPointer(bound.expr));
  8498. break;
  8499. }
  8500. args.append(*getBoundSize(state));
  8501. args.append(*LINK(state.expr));
  8502. OwnedHqlExpr call = bindTranslatedFunctionCall(hashMd5DataId, args);
  8503. ctx.addExpr(call);
  8504. }
  8505. //---------------------------------------------------------------------------
  8506. void HqlCppTranslator::doBuildExprTransfer(BuildCtx & ctx, IHqlExpression * expr, CHqlBoundExpr & tgt)
  8507. {
  8508. CHqlBoundExpr bound;
  8509. //Ensure the bound result has an address
  8510. IHqlExpression * src = expr->queryChild(0);
  8511. bool gotAddress = false;
  8512. if (src->isDataset())
  8513. {
  8514. buildDataset(ctx, src, bound, FormatBlockedDataset);
  8515. bound.expr.setown(getPointer(bound.expr));
  8516. }
  8517. else if (src->isDatarow())
  8518. {
  8519. Owned<IReferenceSelector> ref = buildNewRow(ctx, src);
  8520. ref->get(ctx, bound);
  8521. bound.expr.setown(getPointer(bound.expr));
  8522. }
  8523. else if (isTypePassedByAddress(src->queryType()))
  8524. buildCachedExpr(ctx, src, bound);
  8525. else if (hasAddress(ctx, src))
  8526. {
  8527. buildAddress(ctx, src, bound);
  8528. gotAddress = true;
  8529. }
  8530. else
  8531. buildTempExpr(ctx, src, bound);
  8532. OwnedITypeInfo from = bound.expr->getType();
  8533. ITypeInfo * to = expr->queryType();
  8534. //Must calculate the size of the bound value before we start messing about with stripping casts etc.
  8535. OwnedHqlExpr size;
  8536. if (to->getSize() == UNKNOWN_LENGTH)
  8537. {
  8538. if (from->getSize() == UNKNOWN_LENGTH)
  8539. size.setown(getBoundSize(bound));
  8540. else
  8541. size.setown(getSizetConstant(from->getSize()));
  8542. }
  8543. if (!isTypePassedByAddress(from) && !gotAddress)
  8544. bound.expr.setown(getAddress(bound.expr));
  8545. //strip unnecessary casts...
  8546. while (bound.expr->getOperator() == no_implicitcast)
  8547. bound.expr.set(bound.expr->queryChild(0));
  8548. if (isTypePassedByAddress(to))
  8549. {
  8550. to->Link();
  8551. if (!to->isReference())
  8552. to = makeReferenceModifier(to);
  8553. tgt.expr.setown(createValue(no_implicitcast, to, LINK(bound.expr)));
  8554. if (to->getSize() == UNKNOWN_LENGTH)
  8555. {
  8556. switch (to->getTypeCode())
  8557. {
  8558. case type_unicode:
  8559. if (size->isConstant())
  8560. tgt.length.setown(getSizetConstant((size32_t)getIntValue(size)/sizeof(UChar)));
  8561. else
  8562. tgt.length.setown(createValue(no_div, LINK(sizetType), LINK(size), getSizetConstant(2)));
  8563. break;
  8564. case type_qstring:
  8565. if (size->isConstant())
  8566. tgt.length.setown(getSizetConstant(rtlQStrLength((size32_t)getIntValue(size))));
  8567. else
  8568. tgt.length.setown(createValue(no_div, LINK(sizetType), multiplyValue(size, 4), getSizetConstant(3)));
  8569. break;
  8570. case type_varstring:
  8571. case type_varunicode:
  8572. break;
  8573. default:
  8574. tgt.length.set(size);
  8575. break;
  8576. }
  8577. }
  8578. }
  8579. else
  8580. {
  8581. tgt.length.clear();
  8582. tgt.expr.set(bound.expr);
  8583. if (hasWrapperModifier(tgt.expr->queryType()))
  8584. tgt.expr.setown(createValue(no_implicitcast, makeReferenceModifier(LINK(queryUnqualifiedType(from))), LINK(tgt.expr)));
  8585. tgt.expr.setown(createValue(no_implicitcast, makePointerType(LINK(to)), tgt.expr.getClear()));
  8586. tgt.expr.setown(createValue(no_deref, LINK(to), tgt.expr.getClear()));
  8587. }
  8588. }
  8589. //---------------------------------------------------------------------------
  8590. //-- no_ordered
  8591. void HqlCppTranslator::doBuildExprOrdered(BuildCtx & ctx, IHqlExpression * expr, CHqlBoundExpr & tgt)
  8592. {
  8593. if (ctx.getMatchExpr(expr, tgt))
  8594. return;
  8595. bool ascending = true;
  8596. OwnedHqlExpr list = simplifyFixedLengthList(expr->queryChild(0));
  8597. IHqlExpression * attr = expr->queryChild(1);
  8598. if (attr && attr->isAttribute() && (attr->queryName() == descAtom))
  8599. ascending = false;
  8600. if (list->numChildren() == 0)
  8601. throwError(HQLERR_RankOnNull);
  8602. //create the list that is going to be sorted
  8603. CHqlBoundExpr boundList;
  8604. OwnedHqlExpr optimalList = getOptimialListFormat(list);
  8605. buildExpr(ctx, optimalList, boundList);
  8606. //create a compare function....
  8607. ITypeInfo * elementType = boundList.expr->queryType()->queryChildType();
  8608. unsigned elementSize = elementType->getSize();
  8609. if (elementSize == UNKNOWN_LENGTH)
  8610. throwError(HQLERR_OrderOnVarlengthStrings);
  8611. StringBuffer tempName;
  8612. getUniqueId(tempName.append('v'));
  8613. IHqlExpression * tempCompare = createVariable(tempName.str(), makeVoidType());
  8614. StringBuffer s;
  8615. s.clear().append("extern int ").append(tempName).append("(const void * left, const void * right);");
  8616. if (options.spanMultipleCpp)
  8617. {
  8618. BuildCtx protoctx(*code, mainprototypesAtom);
  8619. protoctx.addQuoted(s);
  8620. }
  8621. else
  8622. {
  8623. BuildCtx protoctx(*code, prototypeAtom);
  8624. protoctx.addQuoted(s);
  8625. }
  8626. BuildCtx declareCtx(*code, declareAtom);
  8627. s.clear().append("int ").append(tempName).append("(const void * left, const void * right)");
  8628. MemberFunction compareFunc(*this, declareCtx, s, MFdynamicproto);
  8629. Owned<ITypeInfo> argType;
  8630. if (isTypePassedByAddress(elementType) && !hasReferenceModifier(elementType))
  8631. argType.setown(makeReferenceModifier(LINK(elementType)));
  8632. else
  8633. argType.setown(makePointerType(LINK(elementType)));
  8634. OwnedHqlExpr leftAddr = createVariable("left", LINK(argType));
  8635. OwnedHqlExpr rightAddr = createVariable("right", LINK(argType));
  8636. IHqlExpression * left = convertAddressToValue(leftAddr, elementType);
  8637. IHqlExpression * right = convertAddressToValue(rightAddr, elementType);
  8638. if (elementType->isReference())
  8639. elementSize = sizeof(char * *);
  8640. left = createTranslatedOwned(left);
  8641. right = createTranslatedOwned(right);
  8642. OwnedHqlExpr compare;
  8643. if (ascending)
  8644. compare.setown(createValue(no_order, LINK(signedType), left, right));
  8645. else
  8646. compare.setown(createValue(no_order, LINK(signedType), right, left));
  8647. CHqlBoundExpr boundCompare;
  8648. buildExpr(compareFunc.ctx, compare, boundCompare);
  8649. compareFunc.ctx.setNextDestructor();
  8650. compareFunc.ctx.addReturn(boundCompare.expr);
  8651. //Allocate an array to store the orders
  8652. unsigned max = list->numChildren();
  8653. Owned<ITypeInfo> t = makeArrayType(LINK(unsignedType), max);
  8654. IHqlExpression * table = ctx.getTempDeclare(t, NULL);
  8655. ctx.associateExpr(expr, table);
  8656. //Generate the call to the function that calculates the orders
  8657. IHqlExpression * castCompare = createValue(no_implicitcast, makePointerType(makeVoidType()), tempCompare);
  8658. HqlExprArray args;
  8659. args.append(*getPointer(table));
  8660. args.append(*getPointer(boundList.expr));
  8661. args.append(*createConstant(unsignedType->castFrom(false, max)));
  8662. args.append(*getSizetConstant(elementSize));
  8663. args.append(*castCompare);
  8664. callProcedure(ctx, createOrderId, args);
  8665. tgt.expr.setown(table);
  8666. }
  8667. //---------------------------------------------------------------------------
  8668. //-- no_rank
  8669. void checkRankRange(IHqlExpression * index, IHqlExpression * list)
  8670. {
  8671. IValue * indexValue = index->queryValue();
  8672. if (indexValue)
  8673. {
  8674. unsigned max = list->numChildren();
  8675. unsigned idx = (unsigned)indexValue->getIntValue();
  8676. //MORE: Should be a warning.....
  8677. if ((idx < 1) || (idx > max))
  8678. throwError(HQLERR_RankOutOfRange);
  8679. }
  8680. //MORE: Could dynamically allocate the array indexes...
  8681. if (list->getOperator() == no_getresult)
  8682. {
  8683. StringBuffer s;
  8684. IHqlExpression * sequence = queryAttributeChild(list, sequenceAtom, 0);
  8685. IHqlExpression * name = queryAttributeChild(list, namedAtom, 0);
  8686. getStoredDescription(s, sequence, name, true);
  8687. throwError1(HQLERR_RankOnNonList, s.str());
  8688. }
  8689. if (list->getOperator() != no_list)
  8690. {
  8691. StringBuffer s;
  8692. getExprECL(list, s);
  8693. throwError1(HQLERR_RankOnNonList, s.str());
  8694. }
  8695. }
  8696. void HqlCppTranslator::createOrderList(BuildCtx & ctx, IHqlExpression * expr, IHqlExpression * ascdesc, CHqlBoundExpr & tgt)
  8697. {
  8698. ITypeInfo * orderedType = makeArrayType(LINK(unsignedType), expr->numChildren());
  8699. OwnedHqlExpr ordered = createValue(no_ordered, orderedType, LINK(expr), LINK(ascdesc));
  8700. buildExpr(ctx, ordered, tgt);
  8701. }
  8702. void HqlCppTranslator::doBuildExprRank(BuildCtx & ctx, IHqlExpression * expr, CHqlBoundExpr & tgt)
  8703. {
  8704. IHqlExpression * index = expr->queryChild(0);
  8705. IHqlExpression * list = expr->queryChild(1);
  8706. if (list->getOperator() == no_alias_scope)
  8707. list = list->queryChild(0);
  8708. checkRankRange(index, list);
  8709. CHqlBoundExpr bound, boundIndex;
  8710. createOrderList(ctx, list, expr->queryChild(2), bound);
  8711. buildExpr(ctx, index, boundIndex);
  8712. HqlExprArray args;
  8713. args.append(*boundIndex.expr.getClear());
  8714. args.append(*createConstant(unsignedType->castFrom(false, list->numChildren())));
  8715. args.append(*getPointer(bound.expr));
  8716. tgt.expr.setown(bindTranslatedFunctionCall(rankFromOrderId, args));
  8717. }
  8718. //---------------------------------------------------------------------------
  8719. //-- no_ranked
  8720. void HqlCppTranslator::doBuildExprRanked(BuildCtx & ctx, IHqlExpression * expr, CHqlBoundExpr & tgt)
  8721. {
  8722. IHqlExpression * index = expr->queryChild(0);
  8723. IHqlExpression * list = expr->queryChild(1);
  8724. if (list->getOperator() == no_alias_scope)
  8725. list = list->queryChild(0);
  8726. checkRankRange(index, list);
  8727. CHqlBoundExpr bound, boundIndex;
  8728. createOrderList(ctx, list, expr->queryChild(2), bound);
  8729. buildExpr(ctx, index, boundIndex);
  8730. HqlExprArray args;
  8731. args.append(*boundIndex.expr.getClear());
  8732. args.append(*createConstant(unsignedType->castFrom(false, list->numChildren())));
  8733. args.append(*getPointer(bound.expr));
  8734. tgt.expr.setown(bindTranslatedFunctionCall(rankedFromOrderId, args));
  8735. }
  8736. //---------------------------------------------------------------------------
  8737. //-- no_fail
  8738. void HqlCppTranslator::doBuildStmtFail(BuildCtx & ctx, IHqlExpression * expr)
  8739. {
  8740. HqlExprArray args;
  8741. args.append(*getFailCode(expr));
  8742. args.append(*getFailMessage(expr, false));
  8743. IIdAtom * func = expr->hasAttribute(defaultAtom) ? sysFailId : _failId;
  8744. OwnedHqlExpr call = bindFunctionCall(func, args);
  8745. buildStmt(ctx, call);
  8746. }
  8747. void HqlCppTranslator::doBuildExprFailCode(BuildCtx & ctx, IHqlExpression * expr, CHqlBoundExpr & tgt)
  8748. {
  8749. OwnedHqlExpr activeFailMarker = createAttribute(activeFailureAtom);
  8750. HqlExprAssociation * matchedMarker = ctx.queryMatchExpr(activeFailMarker);
  8751. if (!matchedMarker && !ctx.queryMatchExpr(globalContextMarkerExpr))
  8752. {
  8753. if (!buildExprInCorrectContext(ctx, expr, tgt, false))
  8754. throwError1(HQLERR_FailXUsedOutsideFailContext, getOpString(expr->getOperator()));
  8755. return;
  8756. }
  8757. HqlExprArray args;
  8758. if (matchedMarker)
  8759. {
  8760. args.append(*LINK(matchedMarker->queryExpr()));
  8761. tgt.expr.setown(bindTranslatedFunctionCall(queryLocalFailCodeId, args));
  8762. }
  8763. else
  8764. {
  8765. tgt.expr.setown(bindTranslatedFunctionCall(queryFailCodeId, args));
  8766. }
  8767. }
  8768. void HqlCppTranslator::doBuildExprMatchedInJoin(BuildCtx & ctx, IHqlExpression * expr, CHqlBoundExpr & tgt)
  8769. {
  8770. IHqlExpression *selExpr = expr->queryChild(0);
  8771. assertex(selExpr);
  8772. node_operator op = selExpr->getOperator();
  8773. if (op==no_rows) //denormalize group
  8774. {
  8775. selExpr = selExpr->queryChild(0);
  8776. assertex(selExpr);
  8777. op = selExpr->getOperator();
  8778. if (op!=no_right)
  8779. throwError(HQLERR_InvalidMatchedPatternInJoin);
  8780. }
  8781. OwnedHqlExpr markerExpr = createValue(no_matched_injoin, makeBoolType(), LINK(selExpr));
  8782. if (!ctx.getMatchExpr(markerExpr, tgt))
  8783. {
  8784. if (!buildExprInCorrectContext(ctx, expr, tgt, false))
  8785. throwError(HQLERR_InvalidMatchedPatternInJoin); //to get this far they must be matching on a dataset that is not the current left or right
  8786. return;
  8787. }
  8788. }
  8789. void HqlCppTranslator::doBuildAssignFailMessage(BuildCtx & ctx, const CHqlBoundTarget & target, IHqlExpression * expr)
  8790. {
  8791. OwnedHqlExpr activeFailMarker = createAttribute(activeFailureAtom);
  8792. HqlExprAssociation * matchedMarker = ctx.queryMatchExpr(activeFailMarker);
  8793. if (!matchedMarker && !ctx.queryMatchExpr(globalContextMarkerExpr))
  8794. {
  8795. CHqlBoundExpr match;
  8796. if (!buildExprInCorrectContext(ctx, expr, match, false))
  8797. throwError1(HQLERR_FailXUsedOutsideFailContext, getOpString(expr->getOperator()));
  8798. assign(ctx, target, match);
  8799. return;
  8800. }
  8801. IIdAtom * func = getFailMessageId;
  8802. HqlExprArray args;
  8803. if (matchedMarker)
  8804. {
  8805. func = getLocalFailMessageId;
  8806. args.append(*createTranslated(matchedMarker->queryExpr()));
  8807. }
  8808. LinkedHqlExpr tag = expr->queryChild(0);
  8809. if (!tag)
  8810. tag.setown(createQuoted("0", makeConstantModifier(makeReferenceModifier(makeVarStringType(UNKNOWN_LENGTH, 0, 0)))));
  8811. args.append(*LINK(tag));
  8812. OwnedHqlExpr call = bindFunctionCall(func, args);
  8813. buildExprAssign(ctx, target, call);
  8814. }
  8815. void HqlCppTranslator::doBuildAssignEventName(BuildCtx & ctx, const CHqlBoundTarget & target, IHqlExpression * expr)
  8816. {
  8817. HqlExprArray args;
  8818. OwnedHqlExpr call = bindFunctionCall(getEventNameId, args);
  8819. buildExprAssign(ctx, target, call);
  8820. }
  8821. void HqlCppTranslator::doBuildAssignEventExtra(BuildCtx & ctx, const CHqlBoundTarget & target, IHqlExpression * expr)
  8822. {
  8823. LinkedHqlExpr tag = expr->queryChild(0);
  8824. if (!tag)
  8825. tag.setown(createQuoted("0", makeConstantModifier(makeReferenceModifier(makeVarStringType(UNKNOWN_LENGTH, 0, 0)))));
  8826. HqlExprArray args;
  8827. args.append(*LINK(tag));
  8828. OwnedHqlExpr call = bindFunctionCall(getEventExtraId, args);
  8829. buildExprAssign(ctx, target, call);
  8830. }
  8831. //---------------------------------------------------------------------------
  8832. //-- system call e.g. EXP(), LOG()...
  8833. void HqlCppTranslator::doBuildExprSysFunc(BuildCtx & ctx, IHqlExpression * expr, CHqlBoundExpr & tgt, IIdAtom * funcName, byte dbz)
  8834. {
  8835. HqlExprArray args;
  8836. ForEachChild(i, expr)
  8837. {
  8838. IHqlExpression * cur = expr->queryChild(i);
  8839. if (!cur->isAttribute())
  8840. args.append(*LINK(cur));
  8841. }
  8842. if (dbz)
  8843. args.append(*createConstant(dbz));
  8844. OwnedHqlExpr call = bindFunctionCall(funcName, args);
  8845. buildExpr(ctx, call, tgt);
  8846. }
  8847. void HqlCppTranslator::doBuildExprOffsetOf(BuildCtx & ctx, IHqlExpression * expr, CHqlBoundExpr & tgt)
  8848. {
  8849. if (ctx.getMatchExpr(expr, tgt))
  8850. return;
  8851. IHqlExpression * arg = expr->queryChild(0);
  8852. Owned<IReferenceSelector> selector = buildActiveReference(ctx, arg);
  8853. selector->getOffset(ctx, tgt);
  8854. //cache non-constant values in a temporary variable...
  8855. if (!isSimpleLength(tgt.expr))
  8856. {
  8857. IHqlExpression * temp = ctx.getTempDeclare(expr->queryType(), tgt.expr);
  8858. tgt.expr.setown(temp);
  8859. ctx.associateExpr(expr, tgt);
  8860. }
  8861. }
  8862. //---------------------------------------------------------------------------
  8863. //-- no_subselect --
  8864. void HqlCppTranslator::doBuildAssignSubString(BuildCtx & ctx, const CHqlBoundTarget & target, IHqlExpression * expr)
  8865. {
  8866. if (expr->queryChild(1)->getOperator() == no_rangecommon)
  8867. throwError(HQLERR_StarRangeOnlyInJoinCondition);
  8868. ITypeInfo * resultType = expr->queryType();
  8869. ITypeInfo * targetType = target.queryType();
  8870. type_t rtc = resultType->getTypeCode();
  8871. type_t ttc = targetType->getTypeCode();
  8872. SubStringInfo info(expr);
  8873. CHqlBoundExpr newBound;
  8874. bool doneAssign = false;
  8875. if (info.special)
  8876. doBuildExprSpecialSubString(ctx, info, newBound);
  8877. else if (info.infiniteString)
  8878. doBuildExprInfiniteSubString(ctx, info, newBound);
  8879. if (!newBound.expr)
  8880. {
  8881. IIdAtom * func = NULL;
  8882. type_t stc = info.src->queryType()->getTypeCode();
  8883. if (target.isFixedSize())
  8884. {
  8885. switch (ttc)
  8886. {
  8887. case type_qstring:
  8888. if (stc == type_qstring)
  8889. func = subQStrFTId;
  8890. break;
  8891. case type_data:
  8892. switch (stc)
  8893. {
  8894. case type_data:
  8895. case type_string:
  8896. case type_varstring:
  8897. func = subDataFTId;
  8898. break;
  8899. }
  8900. break;
  8901. case type_string:
  8902. switch (stc)
  8903. {
  8904. case type_data:
  8905. func = subDataFTId;
  8906. break;
  8907. case type_string:
  8908. case type_varstring:
  8909. if (resultType->queryCharset() == info.src->queryType()->queryCharset())
  8910. func = subStrFTId;
  8911. break;
  8912. }
  8913. break;
  8914. }
  8915. if (!func && (queryUnqualifiedType(resultType) != queryUnqualifiedType(targetType)))
  8916. {
  8917. CHqlBoundExpr bound;
  8918. buildTempExpr(ctx, expr, bound);
  8919. assign(ctx, target, bound);
  8920. return;
  8921. }
  8922. }
  8923. CHqlBoundExpr boundSrc;
  8924. buildCachedExpr(ctx, info.src, boundSrc);
  8925. info.bindToFrom(*this, ctx);
  8926. if (!info.boundFrom.expr)
  8927. info.boundFrom.expr.setown(getSizetConstant(1));
  8928. //Some hacks to force the parameters/return values to the same type. It could be solved more cleanly,
  8929. //but with more functions by calling different functions instead.
  8930. CHqlBoundTarget tempTarget;
  8931. tempTarget.set(target);
  8932. switch (rtc)
  8933. {
  8934. case type_string:
  8935. if (resultType->queryCollation()->queryName() != asciiAtom)
  8936. {
  8937. unsigned sourceLen = boundSrc.queryType()->getStringLen();
  8938. boundSrc.expr.setown(createValue(no_typetransfer, makeStringType(sourceLen, NULL, NULL), LINK(boundSrc.expr)));
  8939. OwnedITypeInfo newTargetType = makeStringType(targetType->getStringLen(), NULL, NULL);
  8940. tempTarget.expr.setown(createValue(no_typetransfer, cloneModifiers(targetType, newTargetType), LINK(tempTarget.expr)));
  8941. }
  8942. break;
  8943. }
  8944. HqlExprArray args;
  8945. args.append(*boundSrc.getTranslatedExpr());
  8946. args.append(*info.boundFrom.getTranslatedExpr());
  8947. if (func)
  8948. {
  8949. args.add(*createTranslated(tempTarget.expr), 0);
  8950. if (info.boundTo.expr)
  8951. args.append(*info.boundTo.getTranslatedExpr());
  8952. else
  8953. args.append(*createConstant(unsignedType->castFrom(false, 0x7fffffff)));
  8954. OwnedHqlExpr call = bindFunctionCall(func, args);
  8955. buildStmt(ctx, call);
  8956. }
  8957. else
  8958. {
  8959. if (info.boundTo.expr)
  8960. {
  8961. args.append(*info.boundTo.getTranslatedExpr());
  8962. switch (rtc)
  8963. {
  8964. case type_qstring:
  8965. func = subQStrFTXId;
  8966. break;
  8967. case type_data:
  8968. func = subDataFTXId;
  8969. break;
  8970. case type_unicode:
  8971. case type_varunicode:
  8972. func = unicodeSubStrFTXId;
  8973. break;
  8974. case type_utf8:
  8975. func = utf8SubStrFTXId;
  8976. break;
  8977. default:
  8978. func = subStrFTXId;
  8979. break;
  8980. }
  8981. }
  8982. else
  8983. {
  8984. switch (rtc)
  8985. {
  8986. case type_qstring:
  8987. func = subQStrFXId;
  8988. break;
  8989. case type_data:
  8990. func = subDataFXId;
  8991. break;
  8992. case type_unicode:
  8993. case type_varunicode:
  8994. func = unicodeSubStrFXId;
  8995. break;
  8996. case type_utf8:
  8997. func = utf8SubStrFXId;
  8998. break;
  8999. default:
  9000. func = subStrFXId;
  9001. break;
  9002. }
  9003. }
  9004. OwnedHqlExpr call = bindFunctionCall(func, args);
  9005. buildExprAssign(ctx, tempTarget, call);
  9006. }
  9007. doneAssign = true;
  9008. }
  9009. if (!doneAssign)
  9010. assign(ctx, target, newBound);
  9011. }
  9012. bool HqlCppTranslator::doBuildExprSpecialSubString(BuildCtx & ctx, SubStringInfo & info, CHqlBoundExpr & tgt)
  9013. {
  9014. unsigned size = info.srcType->getStringLen();
  9015. unsigned fromIndex = info.fixedStart;
  9016. unsigned toIndex = info.fixedEnd;
  9017. //If substring is larger than the source use the default processing.
  9018. if (toIndex <= size)
  9019. {
  9020. CHqlBoundExpr boundSrc;
  9021. buildCachedExpr(ctx, info.src, boundSrc);
  9022. boundSrc.expr.setown(getIndexedElementPointer(boundSrc.expr, fromIndex-1));
  9023. unsigned newLength = fromIndex <= toIndex ? toIndex-(fromIndex-1) : 0;
  9024. ITypeInfo * newType = makeReferenceModifier(getStretchedType(newLength, info.srcType));
  9025. tgt.expr.setown(createValue(no_typetransfer, newType, boundSrc.expr.getClear()));
  9026. if (info.expr->queryType()->getStringLen() != newLength)
  9027. tgt.length.setown(getSizetConstant(newLength));
  9028. return true;
  9029. }
  9030. return false;
  9031. }
  9032. bool HqlCppTranslator::doBuildExprInfiniteSubString(BuildCtx & ctx, SubStringInfo & info, CHqlBoundExpr & tgt)
  9033. {
  9034. CHqlBoundExpr boundSrc;
  9035. info.bindToFrom(*this, ctx);
  9036. buildCachedExpr(ctx, info.src, boundSrc);
  9037. IHqlExpression * from = info.from;
  9038. if (info.fixedStart == 1)
  9039. from = NULL;
  9040. IHqlExpression * start = from ? adjustValue(info.boundFrom.expr, -1) : NULL;
  9041. tgt.expr.setown(getIndexedElementPointer(boundSrc.expr, start));
  9042. //ensure type is no longer infinite length, so same optimization does not happen again...
  9043. ITypeInfo * newType = makeReferenceModifier(getStretchedType(UNKNOWN_LENGTH, tgt.expr->queryType()));
  9044. tgt.expr.setown(createValue(no_typetransfer, newType, tgt.expr.getLink()));
  9045. OwnedHqlExpr length;
  9046. if (start && !isZero(start))
  9047. length.setown(createValue(no_sub, info.boundTo.expr.getLink(), LINK(start)));
  9048. else
  9049. length.setown(info.boundTo.expr.getLink());
  9050. tgt.length.setown(ensureExprType(length, sizetType));
  9051. ::Release(start);
  9052. return true;
  9053. }
  9054. void HqlCppTranslator::doBuildExprAnySubString(BuildCtx & ctx, SubStringInfo & info, CHqlBoundExpr & tgt)
  9055. {
  9056. CHqlBoundExpr boundSource;
  9057. buildCachedExpr(ctx, info.src, boundSource);
  9058. info.bindToFrom(*this, ctx);
  9059. OwnedHqlExpr from;
  9060. if (info.from)
  9061. {
  9062. OwnedHqlExpr start = adjustValue(info.boundFrom.expr, -1);
  9063. if (!isZero(start))
  9064. {
  9065. HqlExprArray args;
  9066. args.append(*LINK(start));
  9067. args.append(*getBoundLength(boundSource));
  9068. OwnedHqlExpr call = bindTranslatedFunctionCall(rtlMinId, args);
  9069. call.setown(createTranslated(call));
  9070. CHqlBoundExpr fromVar;
  9071. buildTempExpr(ctx, call, fromVar);
  9072. from.set(fromVar.expr);
  9073. }
  9074. }
  9075. OwnedHqlExpr to;
  9076. if (info.to)
  9077. {
  9078. OwnedHqlExpr toExpr = LINK(info.boundTo.expr);
  9079. if (from)
  9080. {
  9081. HqlExprArray args;
  9082. args.append(*LINK(toExpr));
  9083. args.append(*LINK(from));
  9084. toExpr.setown(bindTranslatedFunctionCall(rtlMaxId, args));
  9085. }
  9086. HqlExprArray args;
  9087. args.append(*LINK(toExpr));
  9088. args.append(*getBoundLength(boundSource));
  9089. to.setown(bindTranslatedFunctionCall(rtlMinId, args));
  9090. }
  9091. else
  9092. to.setown(getBoundLength(boundSource));
  9093. boundSource.expr.setown(getIndexedElementPointer(boundSource.expr, from));
  9094. ITypeInfo * newType = makeReferenceModifier(info.expr->getType());
  9095. tgt.expr.setown(createValue(no_typetransfer, newType, boundSource.expr.getClear()));
  9096. if (from && !isZero(from))
  9097. tgt.length.setown(createValue(no_sub, LINK(sizetType), LINK(to), LINK(from)));
  9098. else
  9099. tgt.length.set(to);
  9100. }
  9101. void HqlCppTranslator::doBuildExprSubString(BuildCtx & ctx, IHqlExpression * expr, CHqlBoundExpr & tgt)
  9102. {
  9103. if (expr->queryChild(1)->getOperator() == no_rangecommon)
  9104. throwError(HQLERR_StarRangeOnlyInJoinCondition);
  9105. /* Optimize string[start..end] into a type transfer where appropriate */
  9106. SubStringInfo info(expr);
  9107. if (info.special)
  9108. if (doBuildExprSpecialSubString(ctx, info, tgt))
  9109. return;
  9110. if (info.infiniteString)
  9111. if (doBuildExprInfiniteSubString(ctx, info, tgt))
  9112. return;
  9113. if (expr->hasAttribute(quickAtom))
  9114. {
  9115. doBuildExprAnySubString(ctx, info, tgt);
  9116. return;
  9117. }
  9118. buildTempExpr(ctx, expr, tgt);
  9119. }
  9120. //---------------------------------------------------------------------------
  9121. //-- no_trim --
  9122. void HqlCppTranslator::doBuildAssignTrim(BuildCtx & ctx, const CHqlBoundTarget & target, IHqlExpression * expr)
  9123. {
  9124. IHqlExpression * str = expr->queryChild(0);
  9125. IIdAtom * func;
  9126. bool hasAll = expr->hasAttribute(allAtom);
  9127. bool hasLeft = expr->hasAttribute(leftAtom);
  9128. bool hasRight = expr->hasAttribute(rightAtom) || !(hasAll || hasLeft);
  9129. bool hasWS = expr->hasAttribute(whitespaceAtom);
  9130. if (str->queryType()->getTypeCode() == type_varstring)
  9131. {
  9132. if(hasWS)
  9133. func = trimVWSId;
  9134. else if(hasAll)
  9135. func = trimVAllId;
  9136. else if(hasLeft && hasRight)
  9137. func = trimVBothId;
  9138. else if(hasLeft)
  9139. func = trimVLeftId;
  9140. else
  9141. func = trimVRightId;
  9142. }
  9143. else if(str->queryType()->getTypeCode() == type_unicode)
  9144. {
  9145. if(hasWS)
  9146. func = trimUnicodeWSId;
  9147. else if(hasAll)
  9148. func = trimUnicodeAllId;
  9149. else if(hasLeft && hasRight)
  9150. func = trimUnicodeBothId;
  9151. else if(hasLeft)
  9152. func = trimUnicodeLeftId;
  9153. else
  9154. func = trimUnicodeRightId;
  9155. }
  9156. else if(str->queryType()->getTypeCode() == type_varunicode)
  9157. {
  9158. if(hasWS)
  9159. func = trimVUnicodeWSId;
  9160. else if(hasAll)
  9161. func = trimVUnicodeAllId;
  9162. else if(hasLeft && hasRight)
  9163. func = trimVUnicodeBothId;
  9164. else if(hasLeft)
  9165. func = trimVUnicodeLeftId;
  9166. else
  9167. func = trimVUnicodeRightId;
  9168. }
  9169. else if(str->queryType()->getTypeCode() == type_utf8)
  9170. {
  9171. if(hasWS)
  9172. func = trimUtf8WSId;
  9173. else if(hasAll)
  9174. func = trimUtf8AllId;
  9175. else if(hasLeft && hasRight)
  9176. func = trimUtf8BothId;
  9177. else if(hasLeft)
  9178. func = trimUtf8LeftId;
  9179. else
  9180. func = trimUtf8RightId;
  9181. }
  9182. else
  9183. {
  9184. if(hasWS)
  9185. func = trimWSId;
  9186. else if(hasAll)
  9187. func = trimAllId;
  9188. else if(hasLeft && hasRight)
  9189. func = trimBothId;
  9190. else if(hasLeft)
  9191. func = trimLeftId;
  9192. else
  9193. func = trimRightId;
  9194. }
  9195. HqlExprArray args;
  9196. args.append(*LINK(str));
  9197. if (hasWS)
  9198. {
  9199. args.append(*LINK(queryBoolExpr(hasLeft)));
  9200. args.append(*LINK(queryBoolExpr(hasAll)));
  9201. args.append(*LINK(queryBoolExpr(hasRight)));
  9202. }
  9203. OwnedHqlExpr call = bindFunctionCall(func, args);
  9204. buildExprAssign(ctx, target, call);
  9205. }
  9206. void HqlCppTranslator::doBuildExprTrim(BuildCtx & ctx, IHqlExpression * expr, CHqlBoundExpr & tgt)
  9207. {
  9208. // MORE - support LEFT,RIGHT,ALL attributes
  9209. CHqlBoundExpr bound;
  9210. buildSimpleExpr(ctx, expr->queryChild(0), bound);
  9211. HqlExprArray args;
  9212. IIdAtom * func;
  9213. OwnedHqlExpr str = getElementPointer(bound.expr);
  9214. bool hasAll = expr->hasAttribute(allAtom);
  9215. bool hasLeft = expr->hasAttribute(leftAtom);
  9216. bool hasRight = expr->hasAttribute(rightAtom) || !(hasAll || hasLeft);
  9217. bool hasWS = expr->hasAttribute(whitespaceAtom);
  9218. type_t btc = bound.expr->queryType()->getTypeCode();
  9219. if(hasAll || hasLeft || hasWS || btc == type_varstring || btc == type_varunicode)
  9220. {
  9221. if (btc == type_varstring)
  9222. {
  9223. if(hasWS) {
  9224. func = trimVWSId;
  9225. }
  9226. else if(hasAll) {
  9227. func = trimVAllId;
  9228. }
  9229. else if(hasLeft && hasRight) {
  9230. func = trimVBothId;
  9231. }
  9232. else if (hasLeft) {
  9233. func = trimVLeftId;
  9234. }
  9235. else {
  9236. func = trimVRightId;
  9237. }
  9238. }
  9239. else if (btc == type_unicode)
  9240. {
  9241. if(hasWS) {
  9242. func = trimUnicodeWSId;
  9243. }
  9244. else if(hasAll) {
  9245. func = trimUnicodeAllId;
  9246. }
  9247. else if(hasLeft && hasRight) {
  9248. func = trimUnicodeBothId;
  9249. }
  9250. else {
  9251. func = trimUnicodeLeftId;
  9252. }
  9253. }
  9254. else if (btc == type_varunicode)
  9255. {
  9256. if(hasWS) {
  9257. func = trimVUnicodeWSId;
  9258. }
  9259. else if(hasAll) {
  9260. func = trimVUnicodeAllId;
  9261. }
  9262. else if(hasLeft && hasRight) {
  9263. func = trimVUnicodeBothId;
  9264. }
  9265. else if(hasLeft) {
  9266. func = trimVUnicodeLeftId;
  9267. }
  9268. else {
  9269. func = trimVUnicodeRightId;
  9270. }
  9271. }
  9272. else if (btc == type_utf8)
  9273. {
  9274. if(hasWS) {
  9275. func = trimUtf8WSId;
  9276. }
  9277. else if(hasAll) {
  9278. func = trimUtf8AllId;
  9279. }
  9280. else if(hasLeft && hasRight) {
  9281. func = trimUtf8BothId;
  9282. }
  9283. else {
  9284. func = trimUtf8LeftId;
  9285. }
  9286. }
  9287. else
  9288. {
  9289. if(hasWS) {
  9290. func = trimWSId;
  9291. }
  9292. else if(hasAll) {
  9293. func = trimAllId;
  9294. }
  9295. else if(hasLeft && hasRight) {
  9296. func = trimBothId;
  9297. }
  9298. else {
  9299. func = trimLeftId;
  9300. }
  9301. }
  9302. args.append(*bound.getTranslatedExpr());
  9303. if (hasWS)
  9304. {
  9305. args.append(*LINK(queryBoolExpr(hasLeft)));
  9306. args.append(*LINK(queryBoolExpr(hasAll)));
  9307. args.append(*LINK(queryBoolExpr(hasRight)));
  9308. }
  9309. OwnedHqlExpr call = bindFunctionCall(func, args);
  9310. buildExpr(ctx, call, tgt);
  9311. }
  9312. else {
  9313. if (btc == type_varstring)
  9314. {
  9315. args.append(*LINK(str));
  9316. func = trimVStrLenId;
  9317. }
  9318. else if (btc == type_unicode)
  9319. {
  9320. args.append(*getBoundLength(bound));
  9321. args.append(*LINK(str));
  9322. func = trimUnicodeStrLenId;
  9323. }
  9324. else if (btc == type_varunicode)
  9325. {
  9326. args.append(*LINK(str));
  9327. func = trimVUnicodeStrLenId;
  9328. }
  9329. else if (btc == type_utf8)
  9330. {
  9331. args.append(*getBoundLength(bound));
  9332. args.append(*LINK(str));
  9333. func = trimUtf8StrLenId;
  9334. }
  9335. else
  9336. {
  9337. args.append(*getBoundLength(bound));
  9338. args.append(*LINK(str));
  9339. func = trimStrLenId;
  9340. }
  9341. tgt.length.setown(bindTranslatedFunctionCall(func, args));
  9342. tgt.expr.set(str);
  9343. }
  9344. }
  9345. //---------------------------------------------------------------------------
  9346. void HqlCppTranslator::doBuildExprIsValid(BuildCtx & ctx, IHqlExpression * expr, CHqlBoundExpr & tgt)
  9347. {
  9348. IHqlExpression * const value = expr->queryChild(0);
  9349. HqlExprArray args;
  9350. ITypeInfo * type = value->queryType();
  9351. assertex(type);
  9352. if (type->getTypeCode() == type_alien)
  9353. {
  9354. IHqlAlienTypeInfo * alien = queryAlienType(type);
  9355. IHqlExpression * isValidFunction = alien->queryFunction(getIsValidId);
  9356. if (isValidFunction)
  9357. {
  9358. CHqlBoundExpr bound;
  9359. buildAddress(ctx, value, bound);
  9360. OwnedITypeInfo physicalType = alien->getPhysicalType();
  9361. if (!isTypePassedByAddress(physicalType))
  9362. bound.expr.setown(createValue(no_deref, makeReferenceModifier(LINK(physicalType)), LINK(bound.expr)));
  9363. HqlExprArray args;
  9364. args.append(*bound.getTranslatedExpr());
  9365. OwnedHqlExpr test = createBoundFunction(NULL, isValidFunction, args, NULL, true);
  9366. buildExpr(ctx, test, tgt);
  9367. return;
  9368. }
  9369. else
  9370. type = alien->queryLogicalType();
  9371. }
  9372. CHqlBoundExpr bound;
  9373. buildExpr(ctx, value, bound);
  9374. type_t tc = type->getTypeCode();
  9375. if ((tc == type_decimal) && (bound.expr->getOperator() == no_decimalstack))
  9376. {
  9377. tgt.expr.setown(bindTranslatedFunctionCall(DecValidTosId, args));
  9378. return;
  9379. }
  9380. ensureHasAddress(ctx, bound);
  9381. OwnedHqlExpr address = getPointer(bound.expr);
  9382. switch (tc)
  9383. {
  9384. case type_decimal:
  9385. args.append(*createConstant(type->isSigned()));
  9386. args.append(*getSizetConstant(type->getDigits()));
  9387. args.append(*address.getLink());
  9388. tgt.expr.setown(bindTranslatedFunctionCall(DecValidId, args));
  9389. break;
  9390. case type_real:
  9391. args.append(*getSizetConstant(type->getSize()));
  9392. args.append(*address.getLink());
  9393. tgt.expr.setown(bindTranslatedFunctionCall(validRealId, args));
  9394. break;
  9395. default:
  9396. tgt.expr.set(queryBoolExpr(true));
  9397. break;
  9398. }
  9399. }
  9400. IHqlExpression * HqlCppTranslator::getConstWuid(IHqlExpression * expr)
  9401. {
  9402. OwnedHqlExpr wuid = createConstant(wu()->queryWuid());
  9403. return ensureExprType(wuid, expr->queryType());
  9404. }
  9405. void HqlCppTranslator::doBuildAssignWuid(BuildCtx & ctx, const CHqlBoundTarget & target, IHqlExpression * expr)
  9406. {
  9407. HqlExprArray args;
  9408. OwnedHqlExpr call = bindFunctionCall(getWuidId, args);
  9409. buildExprAssign(ctx, target, call);
  9410. }
  9411. void HqlCppTranslator::doBuildExprWuid(BuildCtx & ctx, IHqlExpression * expr, CHqlBoundExpr & tgt)
  9412. {
  9413. HqlExprArray args;
  9414. OwnedHqlExpr call = bindFunctionCall(getWuidId, args);
  9415. buildTempExpr(ctx, call, tgt);
  9416. }
  9417. IHqlExpression * HqlCppTranslator::cvtGetEnvToCall(IHqlExpression * expr)
  9418. {
  9419. IHqlExpression * dft = queryRealChild(expr, 1);
  9420. HqlExprArray args;
  9421. args.append(*LINK(expr->queryChild(0)));
  9422. if (dft)
  9423. args.append(*LINK(dft));
  9424. else
  9425. args.append(*createConstant(createStringValue((const char *)NULL, 0U)));
  9426. return bindFunctionCall(getEnvId, args);
  9427. }
  9428. //---------------------------------------------------------------------------
  9429. void HqlCppTranslator::doBuildAssignToFromUnicode(BuildCtx & ctx, const CHqlBoundTarget & target, IHqlExpression * expr)
  9430. {
  9431. HqlExprArray args;
  9432. if(!target.isFixedSize())
  9433. {
  9434. args.append(*LINK(expr->queryChild(0)));
  9435. args.append(*foldHqlExpression(expr->queryChild(1)));
  9436. OwnedHqlExpr call = bindFunctionCall((expr->getOperator() == no_fromunicode) ? unicode2CodepageXId : codepage2UnicodeXId, args);
  9437. buildExprAssign(ctx, target, call);
  9438. }
  9439. else
  9440. {
  9441. args.append(*createTranslated(target.expr));
  9442. args.append(*LINK(expr->queryChild(0)));
  9443. args.append(*foldHqlExpression(expr->queryChild(1)));
  9444. OwnedHqlExpr call = bindFunctionCall((expr->getOperator() == no_fromunicode) ? unicode2CodepageId : codepage2UnicodeId, args);
  9445. buildStmt(ctx, call);
  9446. }
  9447. }
  9448. void HqlCppTranslator::doBuildExprToFromUnicode(BuildCtx & ctx, IHqlExpression * expr, CHqlBoundExpr & tgt)
  9449. {
  9450. HqlExprArray args;
  9451. args.append(*LINK(expr->queryChild(0)));
  9452. args.append(*foldHqlExpression(expr->queryChild(1)));
  9453. OwnedHqlExpr call = bindFunctionCall((expr->getOperator() == no_fromunicode) ? unicode2CodepageXId : codepage2UnicodeXId, args);
  9454. buildExpr(ctx, call, tgt);
  9455. }
  9456. //---------------------------------------------------------------------------
  9457. void HqlCppTranslator::doBuildExprKeyUnicode(BuildCtx & ctx, IHqlExpression * expr, CHqlBoundExpr & tgt)
  9458. {
  9459. HqlExprArray args;
  9460. args.append(*LINK(expr->queryChild(0)));
  9461. args.append(*LINK(expr->queryChild(1)));
  9462. args.append(*LINK(expr->queryChild(2)));
  9463. OwnedHqlExpr call = bindFunctionCall(keyUnicodeStrengthXId, args);
  9464. buildExpr(ctx, call, tgt);
  9465. }
  9466. //---------------------------------------------------------------------------
  9467. void HqlCppTranslator::doBuildAssignWhich(BuildCtx & ctx, const CHqlBoundTarget & target, IHqlExpression * expr)
  9468. {
  9469. BuildCtx whichCtx(ctx);
  9470. unsigned max = expr->numChildren();
  9471. unsigned idx;
  9472. bool invert = (expr->getOperator() == no_rejected);
  9473. for (idx = 0; idx < max; idx++)
  9474. {
  9475. IHqlExpression * cur = expr->queryChild(idx);
  9476. CHqlBoundExpr bound;
  9477. if (invert)
  9478. {
  9479. OwnedHqlExpr inverse = getInverse(cur);
  9480. buildExpr(whichCtx, inverse, bound);
  9481. }
  9482. else
  9483. buildExpr(whichCtx, cur, bound);
  9484. IHqlStmt * stmt = whichCtx.addFilter(bound.expr);
  9485. OwnedHqlExpr value = createConstant(target.queryType()->castFrom(false, idx+1));
  9486. assignBound(whichCtx, target, value);
  9487. whichCtx.selectElse(stmt);
  9488. }
  9489. assignBound(whichCtx, target, queryZero());
  9490. }
  9491. //---------------------------------------------------------------------------
  9492. void HqlCppTranslator::assignBound(BuildCtx & ctx, const CHqlBoundTarget & lhs, IHqlExpression * rhs)
  9493. {
  9494. CHqlBoundExpr bound;
  9495. bound.expr.set(rhs);
  9496. assign(ctx, lhs, bound);
  9497. }
  9498. void HqlCppTranslator::assignBoundToTemp(BuildCtx & ctx, IHqlExpression * lhs, IHqlExpression * rhs)
  9499. {
  9500. CHqlBoundExpr bound;
  9501. CHqlBoundTarget target;
  9502. bound.expr.set(rhs);
  9503. target.expr.set(lhs);
  9504. assign(ctx, target, bound);
  9505. }
  9506. void HqlCppTranslator::assign(BuildCtx & ctx, const CHqlBoundTarget & target, CHqlBoundExpr & rhs)
  9507. {
  9508. IHqlExpression * lhs = target.expr;
  9509. ITypeInfo * lType = lhs->queryType()->queryPromotedType();
  9510. if ((lType->getTypeCode() != type_row) && !target.isFixedSize())
  9511. {
  9512. assignCastUnknownLength(ctx, target, rhs);
  9513. return;
  9514. }
  9515. if (!isSameBasicType(lType, rhs.expr->queryType()->queryPromotedType()))
  9516. assignAndCast(ctx, target, rhs);
  9517. else
  9518. {
  9519. switch (lType->getTypeCode())
  9520. {
  9521. case type_decimal:
  9522. if (isPushed(rhs))
  9523. {
  9524. IIdAtom * funcName = lType->isSigned() ? DecPopDecimalId : DecPopUDecimalId;
  9525. HqlExprArray args;
  9526. args.append(*getPointer(lhs));
  9527. args.append(*getSizetConstant(lType->getSize()));
  9528. args.append(*getSizetConstant(lType->getPrecision()));
  9529. callProcedure(ctx, funcName, args);
  9530. return;
  9531. }
  9532. buildBlockCopy(ctx, lhs, rhs);
  9533. return;
  9534. case type_string:
  9535. {
  9536. if (lType->getSize() == 1 && !options.peephole)
  9537. {
  9538. OwnedHqlExpr l1 = getFirstCharacter(lhs);
  9539. OwnedHqlExpr r1 = getFirstCharacter(rhs.expr);
  9540. ctx.addAssign(l1, r1);
  9541. }
  9542. else
  9543. buildBlockCopy(ctx, lhs, rhs);
  9544. break;
  9545. }
  9546. //fall through...
  9547. case type_unicode:
  9548. case type_data:
  9549. case type_qstring:
  9550. case type_utf8:
  9551. {
  9552. buildBlockCopy(ctx, lhs, rhs);
  9553. break;
  9554. }
  9555. case type_varstring:
  9556. {
  9557. HqlExprArray args;
  9558. args.append(*getElementPointer(lhs));
  9559. args.append(*getElementPointer(rhs.expr));
  9560. callProcedure(ctx, strcpyId, args);
  9561. break;
  9562. }
  9563. case type_varunicode:
  9564. {
  9565. HqlExprArray args;
  9566. args.append(*getElementPointer(lhs));
  9567. args.append(*getElementPointer(rhs.expr));
  9568. callProcedure(ctx, unicodeStrcpyId, args);
  9569. break;
  9570. }
  9571. case type_row:
  9572. {
  9573. if (hasWrapperModifier(target.queryType()))
  9574. {
  9575. //I can't think of any situation where this isn't true....
  9576. assertex(hasLinkCountedModifier(rhs.expr));
  9577. ctx.addAssignLink(lhs, rhs.expr);
  9578. }
  9579. else
  9580. ctx.addAssign(lhs, rhs.expr);
  9581. break;
  9582. }
  9583. default:
  9584. ctx.addAssign(lhs, rhs.expr);
  9585. break;
  9586. }
  9587. }
  9588. }
  9589. void HqlCppTranslator::doStringTranslation(BuildCtx & ctx, ICharsetInfo * tgtset, ICharsetInfo * srcset, unsigned tgtlen, IHqlExpression * srclen, IHqlExpression * target, IHqlExpression * src)
  9590. {
  9591. HqlExprArray args;
  9592. ITranslationInfo * translator = queryDefaultTranslation(tgtset, srcset);
  9593. if (translator)
  9594. {
  9595. IIdAtom * func = createIdAtom(translator->queryRtlFunction());
  9596. args.append(*getSizetConstant(tgtlen));
  9597. args.append(*getElementPointer(target));
  9598. args.append(*LINK(srclen));
  9599. args.append(*getElementPointer(src));
  9600. callProcedure(ctx, func, args);
  9601. }
  9602. }
  9603. void HqlCppTranslator::assignSwapInt(BuildCtx & ctx, ITypeInfo * to, const CHqlBoundTarget & target, CHqlBoundExpr & pure)
  9604. {
  9605. switch (pure.expr->getOperator())
  9606. {
  9607. case no_deref:
  9608. case no_variable:
  9609. break;
  9610. default:
  9611. {
  9612. OwnedHqlExpr translated = pure.getTranslatedExpr();
  9613. pure.clear();
  9614. buildTempExpr(ctx, translated, pure);
  9615. break;
  9616. }
  9617. }
  9618. ITypeInfo * from = pure.expr->queryType();
  9619. unsigned copySize = to->getSize();
  9620. assertex(copySize == from->getSize());
  9621. IHqlExpression * address = getRawAddress(pure.expr);
  9622. switch (copySize)
  9623. {
  9624. case 1:
  9625. ctx.addAssign(target.expr, pure.expr);
  9626. break;
  9627. default:
  9628. {
  9629. HqlExprArray args;
  9630. args.append(*address);
  9631. OwnedHqlExpr call = bindTranslatedFunctionCall(reverseIntId[copySize][to->isSigned()], args);
  9632. ctx.addAssign(target.expr, call);
  9633. break;
  9634. }
  9635. }
  9636. }
  9637. void HqlCppTranslator::throwCannotCast(ITypeInfo * from, ITypeInfo * to)
  9638. {
  9639. StringBuffer fromText, toText;
  9640. getFriendlyTypeStr(from, fromText);
  9641. getFriendlyTypeStr(to, toText);
  9642. throwError2(HQLERR_CastXNotImplemented, fromText.str(), toText.str());
  9643. }
  9644. void HqlCppTranslator::assignAndCast(BuildCtx & ctx, const CHqlBoundTarget & target, CHqlBoundExpr & pure)
  9645. {
  9646. if (!target.isFixedSize())
  9647. {
  9648. assignCastUnknownLength(ctx, target, pure);
  9649. return;
  9650. }
  9651. ITypeInfo * to = target.queryType()->queryPromotedType();
  9652. if ((pure.expr->getOperator() == no_constant) && options.foldConstantCast &&
  9653. ((options.inlineStringThreshold == 0) || (to->getSize() <= options.inlineStringThreshold)))
  9654. {
  9655. OwnedHqlExpr cast = getCastExpr(to, pure.expr);
  9656. if (cast)
  9657. {
  9658. assignBound(ctx, target, cast);
  9659. return;
  9660. }
  9661. }
  9662. //NB: Does not include variable length return types....
  9663. ITypeInfo * from = (pure.expr->queryType()->queryPromotedType());
  9664. type_t toType = to->getTypeCode();
  9665. type_t fromType = from->getTypeCode();
  9666. unsigned toSize = to->getSize();
  9667. IHqlExpression * targetVar = target.expr;
  9668. HqlExprArray args;
  9669. assertex(targetVar);
  9670. assertex(toSize != UNKNOWN_LENGTH);
  9671. switch(toType)
  9672. {
  9673. case type_qstring:
  9674. switch (fromType)
  9675. {
  9676. case type_qstring:
  9677. {
  9678. unsigned srcsize = from->getSize();
  9679. if (!pure.length && (srcsize == toSize))
  9680. {
  9681. //memcpy(tgt, src, srclen)
  9682. args.append(*getElementPointer(targetVar));
  9683. args.append(*getElementPointer(pure.expr));
  9684. args.append(*getSizetConstant(toSize));
  9685. callProcedure(ctx, memcpyId, args);
  9686. }
  9687. else
  9688. {
  9689. args.append(*getSizetConstant(to->getStringLen()));
  9690. args.append(*getElementPointer(targetVar));
  9691. args.append(*getBoundLength(pure));
  9692. args.append(*getElementPointer(pure.expr));
  9693. callProcedure(ctx, qstrToQStrId, args);
  9694. }
  9695. break;
  9696. }
  9697. case type_data:
  9698. case type_varstring:
  9699. case type_string:
  9700. {
  9701. if(!queryDefaultTranslation(to->queryCharset(), from->queryCharset()))
  9702. {
  9703. args.append(*getSizetConstant(to->getStringLen()));
  9704. args.append(*getElementPointer(targetVar));
  9705. args.append(*getBoundLength(pure));
  9706. args.append(*getElementPointer(pure.expr));
  9707. callProcedure(ctx, strToQStrId, args);
  9708. break;
  9709. }
  9710. //fall through
  9711. }
  9712. default:
  9713. //Need to go via a temporary string.
  9714. OwnedHqlExpr temp = pure.getTranslatedExpr();
  9715. buildExprAssignViaString(ctx, target, temp, to);
  9716. return;
  9717. }
  9718. break;
  9719. case type_data:
  9720. case type_string:
  9721. case type_varstring:
  9722. {
  9723. unsigned srclen = from->getSize();
  9724. ICharsetInfo * srcset = NULL;
  9725. ICharsetInfo * tgtset = to->queryCharset();
  9726. IIdAtom * func = NULL;
  9727. switch (fromType)
  9728. {
  9729. case type_data:
  9730. case type_string:
  9731. case type_varstring:
  9732. {
  9733. srcset = from->queryCharset();
  9734. OwnedHqlExpr boundLen = getBoundLength(pure);
  9735. if ((srcset == tgtset) || (toType == type_data) || (fromType == type_data))
  9736. {
  9737. bool doDefault = true;
  9738. if (boundLen->queryValue())
  9739. {
  9740. unsigned srclen = (unsigned)boundLen->queryValue()->getIntValue();
  9741. if (srclen >= toSize && toType != type_varstring)
  9742. {
  9743. if (srclen > toSize)
  9744. srclen = toSize;
  9745. //memcpy(tgt, src, srclen)
  9746. args.append(*getElementPointer(targetVar));
  9747. args.append(*getElementPointer(pure.expr));
  9748. args.append(*getSizetConstant(srclen));
  9749. callProcedure(ctx, memcpyId, args);
  9750. doDefault = false;
  9751. }
  9752. }
  9753. if (doDefault)
  9754. {
  9755. if (fromType == type_varstring)
  9756. {
  9757. switch (toType)
  9758. {
  9759. case type_varstring: func = vstr2VStrId; break;
  9760. case type_string: func = vstr2StrId; break;
  9761. case type_data: func = vstr2DataId; break;
  9762. default: UNIMPLEMENTED;
  9763. }
  9764. if ((toSize < srclen) || (srclen==UNKNOWN_LENGTH) || (toType != type_varstring))
  9765. {
  9766. args.append(*getSizetConstant(toSize));
  9767. args.append(*getElementPointer(targetVar));
  9768. args.append(*getElementPointer(pure.expr));
  9769. callProcedure(ctx, func, args);
  9770. }
  9771. else
  9772. {
  9773. //strcpy(tgt, src);
  9774. args.append(*getElementPointer(targetVar));
  9775. args.append(*getElementPointer(pure.expr));
  9776. callProcedure(ctx, strcpyId, args);
  9777. }
  9778. }
  9779. else
  9780. {
  9781. switch (toType)
  9782. {
  9783. case type_data:
  9784. func = str2DataId;
  9785. break;
  9786. case type_varstring:
  9787. func = str2VStrId;
  9788. break;
  9789. case type_string:
  9790. func = (srcset->queryName() == ebcdicAtom) ? estr2EStrId : str2StrId;
  9791. break;
  9792. }
  9793. args.append(*getSizetConstant(toSize));
  9794. args.append(*getElementPointer(targetVar));
  9795. args.append(*LINK(boundLen));
  9796. args.append(*getElementPointer(pure.expr));
  9797. callProcedure(ctx, func, args);
  9798. }
  9799. }
  9800. }
  9801. else
  9802. {
  9803. if ((from->getSize() == INFINITE_LENGTH) && !pure.length)
  9804. throwError(HQLERR_CastInfiniteString);
  9805. IHqlExpression * srclen;
  9806. if (toType == type_varstring)
  9807. {
  9808. srclen = getSizetConstant(toSize);
  9809. args.append(*srclen);
  9810. args.append(*getElementPointer(targetVar));
  9811. args.append(*LINK(boundLen));
  9812. args.append(*getElementPointer(pure.expr));
  9813. callProcedure(ctx, estr2VStrId, args);
  9814. }
  9815. else
  9816. doStringTranslation(ctx, tgtset, srcset, toSize, boundLen, targetVar, pure.expr);
  9817. }
  9818. }
  9819. break;
  9820. case type_qstring:
  9821. if (queryDefaultTranslation(tgtset, from->queryCharset()))
  9822. {
  9823. OwnedHqlExpr temp = pure.getTranslatedExpr();
  9824. buildExprAssignViaString(ctx, target, temp, to);
  9825. }
  9826. else
  9827. {
  9828. switch (toType)
  9829. {
  9830. case type_varstring: func = qstr2VStrId; break;
  9831. case type_string: func = qstr2StrId; break;
  9832. case type_data: func = qstr2DataId; break;
  9833. }
  9834. args.append(*getSizetConstant(toSize));
  9835. args.append(*getElementPointer(targetVar));
  9836. args.append(*getBoundLength(pure));
  9837. args.append(*getElementPointer(pure.expr));
  9838. callProcedure(ctx, func, args);
  9839. }
  9840. break;
  9841. case type_unicode:
  9842. case type_varunicode:
  9843. {
  9844. switch(toType)
  9845. {
  9846. case type_data:
  9847. func = (fromType == type_varunicode) ? vunicode2DataId : unicode2DataId;
  9848. break;
  9849. case type_string:
  9850. func = (fromType == type_varunicode) ? vunicode2CodepageId : unicode2CodepageId;
  9851. break;
  9852. case type_varstring:
  9853. func = (fromType == type_varunicode) ? vunicode2VCodepageId : unicode2VCodepageId;
  9854. break;
  9855. }
  9856. args.append(*getSizetConstant(toSize));
  9857. args.append(*getElementPointer(targetVar));
  9858. if(fromType != type_varunicode)
  9859. args.append(*getBoundLength(pure));
  9860. args.append(*getElementPointer(pure.expr));
  9861. if(toType != type_data)
  9862. args.append(*createConstant(to->queryCharset()->queryCodepageName()));
  9863. callProcedure(ctx, func, args);
  9864. }
  9865. break;
  9866. case type_utf8:
  9867. {
  9868. switch(toType)
  9869. {
  9870. case type_data:
  9871. func = utf82DataId;
  9872. break;
  9873. case type_string:
  9874. func = utf82CodepageId;
  9875. break;
  9876. case type_varstring:
  9877. OwnedHqlExpr temp = pure.getTranslatedExpr();
  9878. buildExprAssignViaString(ctx, target, temp, to);
  9879. return;
  9880. }
  9881. args.append(*getSizetConstant(toSize));
  9882. args.append(*getElementPointer(targetVar));
  9883. args.append(*getBoundLength(pure));
  9884. args.append(*getElementPointer(pure.expr));
  9885. if(toType != type_data)
  9886. args.append(*createConstant(to->queryCharset()->queryCodepageName()));
  9887. callProcedure(ctx, func, args);
  9888. }
  9889. break;
  9890. case type_swapint:
  9891. {
  9892. CHqlBoundExpr recast;
  9893. ITypeInfo * tempType = makeIntType(srclen, from->isSigned());
  9894. OwnedHqlExpr translated = createValue(no_implicitcast, tempType, pure.getTranslatedExpr());
  9895. buildExpr(ctx, translated, recast);
  9896. assignAndCast(ctx, target, recast);
  9897. return;
  9898. }
  9899. case type_int:
  9900. case type_packedint:
  9901. {
  9902. //l2an4(toSize, tgt, expr);
  9903. IIdAtom * funcName;
  9904. if (from->isSigned())
  9905. {
  9906. if (toType != type_varstring)
  9907. funcName = (srclen > 4 ? ls82anId : ls42anId);
  9908. else
  9909. funcName = (srclen > 4 ? ls82vnId : ls42vnId);
  9910. }
  9911. else
  9912. {
  9913. if (toType != type_varstring)
  9914. funcName = (srclen > 4 ? l82anId : l42anId);
  9915. else
  9916. funcName = (srclen > 4 ? l82vnId : l42vnId);
  9917. }
  9918. IHqlExpression * strlen = getSizetConstant(toSize);
  9919. args.append(*strlen);
  9920. args.append(*getElementPointer(targetVar));
  9921. args.append(*LINK(pure.expr));
  9922. callProcedure(ctx, funcName, args);
  9923. if (toType != type_data)
  9924. {
  9925. Owned<ICharsetInfo> charset = getCharset(asciiAtom);
  9926. doStringTranslation(ctx, tgtset, charset, toSize, strlen, targetVar, targetVar);
  9927. }
  9928. break;
  9929. }
  9930. case type_void:
  9931. if (pure.expr->getOperator() != no_decimalstack)
  9932. {
  9933. throwCannotCast(from, to);
  9934. break;
  9935. }
  9936. //fall through
  9937. case type_decimal:
  9938. {
  9939. ensurePushed(ctx, pure);
  9940. args.append(*getSizetConstant(toSize));
  9941. OwnedHqlExpr sp = getElementPointer(targetVar);
  9942. args.append(*ensureIndexable(sp));
  9943. switch (toType)
  9944. {
  9945. case type_string: func = DecPopStringId; break;
  9946. case type_data: func = DecPopStringId; break;
  9947. case type_varstring: func = DecPopVStringId; break;
  9948. }
  9949. callProcedure(ctx, func, args);
  9950. break;
  9951. }
  9952. case type_enumerated:
  9953. throwCannotCast(from, to);
  9954. break;
  9955. case type_boolean:
  9956. {
  9957. func = (toType == type_varstring) ? bool2VStrId : (toType == type_data) ? bool2DataId : bool2StrId;
  9958. args.append(*getSizetConstant(toSize));
  9959. args.append(*getElementPointer(targetVar));
  9960. args.append(*pure.expr.getLink());
  9961. callProcedure(ctx, func, args);
  9962. break;
  9963. }
  9964. case type_real:
  9965. {
  9966. IHqlExpression * strlen = getSizetConstant(toSize);
  9967. args.append(*strlen);
  9968. args.append(*getElementPointer(targetVar));
  9969. args.append(*pure.expr.getLink());
  9970. func = (toType == type_varstring) ? f2vnId : f2anId;
  9971. callProcedure(ctx, func, args);
  9972. if (toType != type_data)
  9973. {
  9974. Owned<ICharsetInfo> charset = getCharset(asciiAtom);
  9975. doStringTranslation(ctx, tgtset, charset, toSize, strlen, targetVar, targetVar);
  9976. }
  9977. }
  9978. break;
  9979. default:
  9980. throwCannotCast(from, to);
  9981. break;
  9982. }
  9983. }
  9984. break;
  9985. case type_unicode:
  9986. case type_varunicode:
  9987. switch (fromType)
  9988. {
  9989. case type_unicode:
  9990. case type_varunicode:
  9991. case type_data:
  9992. case type_string:
  9993. case type_varstring:
  9994. case type_utf8:
  9995. {
  9996. IIdAtom * func;
  9997. switch(fromType)
  9998. {
  9999. case type_unicode:
  10000. func = (toType == type_varunicode) ? unicode2VUnicodeId : unicode2UnicodeId;
  10001. break;
  10002. case type_varunicode:
  10003. func = (toType == type_varunicode) ? vunicode2VUnicodeId : vunicode2UnicodeId;
  10004. break;
  10005. case type_data:
  10006. pure.expr.setown(createValue(no_implicitcast, makeReferenceModifier(makeStringType(from->getStringLen(), NULL)), LINK(pure.expr)));
  10007. func = (toType == type_varunicode) ? codepage2VUnicodeId : codepage2UnicodeId;
  10008. break;
  10009. case type_string:
  10010. func = (toType == type_varunicode) ? codepage2VUnicodeId : codepage2UnicodeId;
  10011. break;
  10012. case type_varstring:
  10013. func = (toType == type_varunicode) ? vcodepage2VUnicodeId : vcodepage2UnicodeId;
  10014. break;
  10015. case type_utf8:
  10016. if (toType == type_varunicode)
  10017. {
  10018. OwnedHqlExpr temp = pure.getTranslatedExpr();
  10019. OwnedITypeInfo type = makeUnicodeType(UNKNOWN_LENGTH, NULL);
  10020. buildExprAssignViaType(ctx, target, temp, type);
  10021. return;
  10022. }
  10023. func = utf82UnicodeId;
  10024. break;
  10025. default:
  10026. throwUnexpected();
  10027. }
  10028. args.append(*getSizetConstant(toSize/2));
  10029. args.append(*getElementPointer(targetVar));
  10030. if((fromType != type_varunicode) && (fromType != type_varstring))
  10031. args.append(*getBoundLength(pure));
  10032. args.append(*getElementPointer(pure.expr));
  10033. if((fromType == type_data) || (fromType == type_string) || (fromType == type_varstring))
  10034. args.append(*createConstant(from->queryCharset()->queryCodepageName()));
  10035. callProcedure(ctx, func, args);
  10036. break;
  10037. }
  10038. default:
  10039. OwnedHqlExpr temp = pure.getTranslatedExpr();
  10040. buildExprAssignViaString(ctx, target, temp, to);
  10041. return;
  10042. }
  10043. break;
  10044. case type_utf8:
  10045. switch (fromType)
  10046. {
  10047. case type_unicode:
  10048. case type_varunicode:
  10049. case type_data:
  10050. case type_string:
  10051. case type_varstring:
  10052. case type_utf8:
  10053. {
  10054. IIdAtom * func;
  10055. switch(fromType)
  10056. {
  10057. case type_unicode:
  10058. case type_varunicode:
  10059. func = unicodeToUtf8Id;
  10060. break;
  10061. case type_utf8:
  10062. func = utf8ToUtf8Id;
  10063. break;
  10064. case type_data:
  10065. case type_string:
  10066. case type_varstring:
  10067. func = codepageToUtf8Id;
  10068. break;
  10069. default:
  10070. throwUnexpected();
  10071. }
  10072. args.append(*getSizetConstant(toSize/4));
  10073. args.append(*getElementPointer(targetVar));
  10074. args.append(*getBoundLength(pure));
  10075. args.append(*getElementPointer(pure.expr));
  10076. if((fromType == type_data) || (fromType == type_string) || (fromType == type_varstring))
  10077. args.append(*createConstant(from->queryCharset()->queryCodepageName()));
  10078. callProcedure(ctx, func, args);
  10079. break;
  10080. }
  10081. default:
  10082. OwnedHqlExpr temp = pure.getTranslatedExpr();
  10083. buildExprAssignViaString(ctx, target, temp, to);
  10084. return;
  10085. }
  10086. break;
  10087. case type_decimal:
  10088. {
  10089. CHqlBoundExpr cast;
  10090. doBuildExprCast(ctx, to, pure, cast);
  10091. ensurePushed(ctx, cast);
  10092. IIdAtom * funcName = to->isSigned() ? DecPopDecimalId : DecPopUDecimalId;
  10093. args.append(*getPointer(target.expr));
  10094. args.append(*getSizetConstant(to->getSize()));
  10095. args.append(*getSizetConstant(to->getPrecision()));
  10096. callProcedure(ctx, funcName, args);
  10097. }
  10098. break;
  10099. case type_swapint:
  10100. {
  10101. unsigned fromSize = from->getSize();
  10102. if (fromType == type_int)
  10103. {
  10104. if (fromSize != toSize)
  10105. {
  10106. Owned<ITypeInfo> tempType = makeIntType(toSize, from->isSigned());
  10107. pure.expr.setown(ensureExprType(pure.expr, tempType));
  10108. }
  10109. if (toSize != 1)
  10110. {
  10111. assignSwapInt(ctx, to, target, pure);
  10112. return;
  10113. }
  10114. }
  10115. CHqlBoundExpr cast;
  10116. doBuildExprCast(ctx, to, pure, cast);
  10117. ctx.addAssign(target.expr, cast.expr);
  10118. }
  10119. break;
  10120. case type_int:
  10121. case type_packedint:
  10122. {
  10123. unsigned fromSize = from->getSize();
  10124. if ((fromType == type_swapint) && !((fromSize == 1) && (toSize == 1)))
  10125. {
  10126. if (fromSize != toSize)
  10127. {
  10128. CHqlBoundExpr tempInt;
  10129. OwnedITypeInfo tempType = makeIntType(fromSize, from->isSigned());
  10130. doBuildCastViaTemp(ctx, tempType, pure, tempInt);
  10131. CHqlBoundExpr cast;
  10132. doBuildExprCast(ctx, to, tempInt, cast);
  10133. ctx.addAssign(target.expr, cast.expr);
  10134. }
  10135. else
  10136. assignSwapInt(ctx, to, target, pure);
  10137. return;
  10138. }
  10139. }
  10140. //fall through
  10141. case type_boolean:
  10142. case type_real:
  10143. case type_row:
  10144. case type_pointer:
  10145. {
  10146. CHqlBoundExpr cast;
  10147. doBuildExprCast(ctx, to, pure, cast);
  10148. ctx.addAssign(target.expr, cast.expr);
  10149. }
  10150. break;
  10151. default:
  10152. throwCannotCast(from, to);
  10153. break;
  10154. }
  10155. }
  10156. void HqlCppTranslator::assignCastUnknownLength(BuildCtx & ctx, const CHqlBoundTarget & target, CHqlBoundExpr & pure)
  10157. {
  10158. assertex(!target.isFixedSize());
  10159. // must be dynamically allocated return type
  10160. ITypeInfo * to = target.queryType();
  10161. ITypeInfo * from = pure.expr->queryType();
  10162. type_t toType = to->getTypeCode();
  10163. type_t fromType = from->getTypeCode();
  10164. IHqlExpression * codepageParam = 0;
  10165. HqlExprArray args;
  10166. IIdAtom * funcName = NULL;
  10167. // assertex(target.length && target.pointer || to->getTypeCode() == type_varstring || to->getTypeCode() == type_varunicode);
  10168. switch (toType)
  10169. {
  10170. case type_qstring:
  10171. {
  10172. switch (fromType)
  10173. {
  10174. case type_qstring:
  10175. funcName = qstrToQStrXId;
  10176. break;
  10177. case type_string:
  10178. case type_data:
  10179. case type_varstring:
  10180. if(!queryDefaultTranslation(to->queryCharset(), from->queryCharset()))
  10181. {
  10182. funcName = strToQStrXId;
  10183. break;
  10184. }
  10185. //fall through
  10186. default:
  10187. CHqlBoundExpr recast;
  10188. ITypeInfo * type = makeStringType(to->getStringLen(), NULL, NULL);
  10189. OwnedHqlExpr translated = createValue(no_implicitcast, type, pure.getTranslatedExpr());
  10190. buildExpr(ctx, translated, recast);
  10191. assignCastUnknownLength(ctx, target, recast);
  10192. return;
  10193. }
  10194. break;
  10195. }
  10196. case type_string:
  10197. case type_data:
  10198. {
  10199. unsigned srclen = from->getSize();
  10200. switch (fromType)
  10201. {
  10202. case type_data:
  10203. case type_string:
  10204. case type_varstring:
  10205. {
  10206. ICharsetInfo * srcset = from->queryCharset();
  10207. ICharsetInfo * tgtset = to->queryCharset();
  10208. if (to->getTypeCode() == type_data)
  10209. funcName = str2DataXId;
  10210. else if ((srcset == tgtset) || (from->getTypeCode() == type_data))
  10211. {
  10212. funcName = str2StrXId;
  10213. }
  10214. else
  10215. {
  10216. if ((from->getSize() == INFINITE_LENGTH) && !pure.length)
  10217. throwError(HQLERR_CastInfiniteString);
  10218. ITranslationInfo * translator = queryDefaultTranslation(tgtset, srcset);
  10219. if (translator)
  10220. funcName = createIdAtom(translator->queryVarRtlFunction());
  10221. else
  10222. funcName = str2StrXId;
  10223. }
  10224. }
  10225. break;
  10226. case type_qstring:
  10227. if(!queryDefaultTranslation(from->queryCharset(), to->queryCharset()))
  10228. {
  10229. funcName = (toType == type_data) ? qstr2DataXId : qstr2StrXId;
  10230. break;
  10231. }
  10232. else
  10233. {
  10234. CHqlBoundExpr recast;
  10235. ITypeInfo * type = makeStringType(to->getStringLen(), NULL, NULL);
  10236. OwnedHqlExpr translated = createValue(no_implicitcast, type, pure.getTranslatedExpr());
  10237. buildExpr(ctx, translated, recast);
  10238. assignCastUnknownLength(ctx, target, recast);
  10239. return;
  10240. }
  10241. case type_unicode:
  10242. {
  10243. if(toType == type_data)
  10244. funcName = unicode2DataXId;
  10245. else
  10246. {
  10247. funcName = unicode2CodepageXId;
  10248. codepageParam = createConstant(to->queryCharset()->queryCodepageName());
  10249. }
  10250. break;
  10251. }
  10252. case type_varunicode:
  10253. {
  10254. if(toType == type_data)
  10255. funcName = vunicode2DataXId;
  10256. else
  10257. {
  10258. funcName = vunicode2CodepageXId;
  10259. codepageParam = createConstant(to->queryCharset()->queryCodepageName());
  10260. }
  10261. break;
  10262. }
  10263. case type_utf8:
  10264. {
  10265. if(toType == type_data)
  10266. funcName = utf82DataXId;
  10267. else
  10268. {
  10269. funcName = utf82CodepageXId;
  10270. codepageParam = createConstant(to->queryCharset()->queryCodepageName());
  10271. }
  10272. break;
  10273. }
  10274. case type_swapint:
  10275. {
  10276. CHqlBoundExpr recast;
  10277. ITypeInfo * type = makeIntType(from->getSize(), from->isSigned());
  10278. OwnedHqlExpr translated = createValue(no_implicitcast, type, pure.getTranslatedExpr());
  10279. buildExpr(ctx, translated, recast);
  10280. assignCastUnknownLength(ctx, target, recast);
  10281. return;
  10282. }
  10283. case type_int:
  10284. case type_real:
  10285. case type_boolean:
  10286. case type_packedint:
  10287. {
  10288. Owned<ICharsetInfo> asciiCharset = getCharset(asciiAtom);
  10289. if (to->queryCharset() != asciiCharset)
  10290. {
  10291. //This should really be handled by the call processing.
  10292. CHqlBoundExpr recast;
  10293. ITypeInfo * type = makeStringType(to->getStringLen(), NULL, NULL);
  10294. OwnedHqlExpr translated = createValue(no_implicitcast, type, pure.getTranslatedExpr());
  10295. buildExpr(ctx, translated, recast);
  10296. assignCastUnknownLength(ctx, target, recast);
  10297. return;
  10298. }
  10299. if (fromType == type_real)
  10300. funcName = f2axId;
  10301. else if (fromType == type_boolean)
  10302. funcName = bool2StrXId;
  10303. else if (from->isSigned())
  10304. funcName = (srclen > 4 ? ls82axId : ls42axId);
  10305. else
  10306. funcName = (srclen > 4 ? l82axId : l42axId);
  10307. args.append(*pure.getTranslatedExpr());
  10308. OwnedHqlExpr call = bindFunctionCall(funcName, args);
  10309. buildExprAssign(ctx, target, call);
  10310. return;
  10311. }
  10312. case type_void:
  10313. if (pure.expr->getOperator() != no_decimalstack)
  10314. {
  10315. throwCannotCast(from, to);
  10316. break;
  10317. }
  10318. //fall through
  10319. case type_decimal:
  10320. {
  10321. ensurePushed(ctx, pure);
  10322. OwnedHqlExpr call = bindFunctionCall(DecPopStringXId, args);
  10323. buildExprAssign(ctx, target, call);
  10324. return;
  10325. }
  10326. default:
  10327. assertex(!"Unknown copy source type");
  10328. return;
  10329. }
  10330. break;
  10331. }
  10332. case type_varstring:
  10333. {
  10334. unsigned srclen = from->getSize();
  10335. switch (from->getTypeCode())
  10336. {
  10337. case type_data:
  10338. case type_string:
  10339. case type_varstring:
  10340. {
  10341. ICharsetInfo * srcset = from->queryCharset();
  10342. ICharsetInfo * tgtset = to->queryCharset();
  10343. if ((srcset == tgtset) || (to->getTypeCode() == type_data) || (from->getTypeCode() == type_data))
  10344. {
  10345. funcName = str2VStrXId;
  10346. }
  10347. else
  10348. {
  10349. funcName = estr2VStrXId;
  10350. }
  10351. }
  10352. break;
  10353. case type_unicode:
  10354. {
  10355. funcName = unicode2VCodepageXId;
  10356. codepageParam = createConstant(to->queryCharset()->queryCodepageName());
  10357. }
  10358. break;
  10359. case type_varunicode:
  10360. {
  10361. funcName = vunicode2VCodepageXId;
  10362. codepageParam = createConstant(to->queryCharset()->queryCodepageName());
  10363. }
  10364. break;
  10365. case type_qstring:
  10366. case type_utf8:
  10367. {
  10368. CHqlBoundExpr recast;
  10369. ITypeInfo * type = makeStringType(from->getStringLen(), NULL, NULL);
  10370. OwnedHqlExpr translated = createValue(no_implicitcast, type, pure.getTranslatedExpr());
  10371. buildExpr(ctx, translated, recast);
  10372. assignCastUnknownLength(ctx, target, recast);
  10373. return;
  10374. }
  10375. case type_swapint:
  10376. {
  10377. CHqlBoundExpr recast;
  10378. ITypeInfo * type = makeIntType(from->getSize(), from->isSigned());
  10379. OwnedHqlExpr translated = createValue(no_implicitcast, type, pure.getTranslatedExpr());
  10380. buildExpr(ctx, translated, recast);
  10381. assignCastUnknownLength(ctx, target, recast);
  10382. return;
  10383. }
  10384. case type_int:
  10385. case type_packedint:
  10386. {
  10387. //l2an4(tgtlen, tgt, expr);
  10388. if (from->isSigned())
  10389. funcName = (srclen > 4 ? ls82vxId : ls42vxId);
  10390. else
  10391. funcName = (srclen > 4 ? l82vxId : l42vxId);
  10392. break;
  10393. }
  10394. case type_boolean:
  10395. {
  10396. funcName = bool2VStrXId;
  10397. break;
  10398. }
  10399. case type_real:
  10400. {
  10401. funcName = f2vxId;;
  10402. break;
  10403. }
  10404. case type_void:
  10405. if (pure.expr->getOperator() != no_decimalstack)
  10406. {
  10407. throwCannotCast(from, to);
  10408. break;
  10409. }
  10410. //fall through
  10411. case type_decimal:
  10412. {
  10413. ensurePushed(ctx, pure);
  10414. OwnedHqlExpr call = bindFunctionCall(DecPopVStringXId, args);
  10415. buildExprAssign(ctx, target, call);
  10416. return;
  10417. }
  10418. default:
  10419. assertex(!"Unknown copy source type");
  10420. return;
  10421. }
  10422. break;
  10423. }
  10424. case type_unicode:
  10425. {
  10426. switch (fromType)
  10427. {
  10428. case type_unicode:
  10429. funcName = unicode2UnicodeXId;
  10430. break;
  10431. case type_varunicode:
  10432. funcName = vunicode2UnicodeXId;
  10433. break;
  10434. case type_utf8:
  10435. funcName = utf82UnicodeXId;
  10436. break;
  10437. case type_data:
  10438. funcName = codepage2UnicodeXId;
  10439. codepageParam = createConstant(from->queryCharset()->queryCodepageName());
  10440. pure.expr.setown(createValue(no_implicitcast, makeStringType(from->getStringLen(), NULL, NULL), LINK(pure.expr)));
  10441. break;
  10442. case type_string:
  10443. funcName = codepage2UnicodeXId;
  10444. codepageParam = createConstant(from->queryCharset()->queryCodepageName());
  10445. pure.expr.setown(createValue(no_typetransfer, makeStringType(from->getStringLen(), NULL, NULL), LINK(pure.expr)));
  10446. break;
  10447. case type_varstring:
  10448. funcName = vcodepage2UnicodeXId;
  10449. codepageParam = createConstant(from->queryCharset()->queryCodepageName());
  10450. pure.expr.setown(createValue(no_typetransfer, makeVarStringType(from->getStringLen(), NULL, NULL), LINK(pure.expr)));
  10451. break;
  10452. default:
  10453. CHqlBoundExpr recast;
  10454. ITypeInfo * type = makeStringType(to->getStringLen(), NULL, NULL);
  10455. OwnedHqlExpr translated = createValue(no_implicitcast, type, pure.getTranslatedExpr());
  10456. buildExpr(ctx, translated, recast);
  10457. assignCastUnknownLength(ctx, target, recast);
  10458. return;
  10459. }
  10460. break;
  10461. }
  10462. case type_varunicode:
  10463. {
  10464. switch (fromType)
  10465. {
  10466. case type_unicode:
  10467. case type_utf8: // go via unicode
  10468. funcName = unicode2VUnicodeXId;
  10469. break;
  10470. case type_varunicode:
  10471. funcName = vunicode2VUnicodeXId;
  10472. break;
  10473. case type_string:
  10474. case type_data:
  10475. funcName = codepage2VUnicodeXId;
  10476. codepageParam = createConstant(from->queryCharset()->queryCodepageName());
  10477. pure.expr.setown(createValue(no_typetransfer, makeStringType(from->getStringLen(), NULL, NULL), LINK(pure.expr)));
  10478. break;
  10479. case type_varstring:
  10480. funcName = vcodepage2VUnicodeXId;
  10481. codepageParam = createConstant(from->queryCharset()->queryCodepageName());
  10482. pure.expr.setown(createValue(no_typetransfer, makeVarStringType(from->getStringLen(), NULL, NULL), LINK(pure.expr)));
  10483. break;
  10484. default:
  10485. CHqlBoundExpr recast;
  10486. ITypeInfo * type = makeStringType(to->getStringLen(), NULL, NULL);
  10487. OwnedHqlExpr translated = createValue(no_implicitcast, type, pure.getTranslatedExpr());
  10488. buildExpr(ctx, translated, recast);
  10489. assignCastUnknownLength(ctx, target, recast);
  10490. return;
  10491. }
  10492. break;
  10493. }
  10494. case type_utf8:
  10495. {
  10496. switch (fromType)
  10497. {
  10498. case type_unicode:
  10499. case type_varunicode:
  10500. funcName = unicodeToUtf8XId;
  10501. break;
  10502. case type_utf8:
  10503. funcName = utf8ToUtf8XId;
  10504. break;
  10505. case type_string:
  10506. case type_data:
  10507. funcName = codepageToUtf8XId;
  10508. codepageParam = createConstant(from->queryCharset()->queryCodepageName());
  10509. pure.expr.setown(createValue(no_typetransfer, makeStringType(from->getStringLen(), NULL, NULL), LINK(pure.expr)));
  10510. break;
  10511. default:
  10512. CHqlBoundExpr recast;
  10513. ITypeInfo * type = makeStringType(to->getStringLen(), NULL, NULL);
  10514. OwnedHqlExpr translated = createValue(no_implicitcast, type, pure.getTranslatedExpr());
  10515. buildExpr(ctx, translated, recast);
  10516. assignCastUnknownLength(ctx, target, recast);
  10517. return;
  10518. }
  10519. break;
  10520. }
  10521. case type_set:
  10522. if (isSameBasicType(to->queryChildType(), from->queryChildType()))
  10523. {
  10524. if (!target.isAll)
  10525. {
  10526. //Ugly. Create a dummy isAll field to assign to..
  10527. assertex(!pure.isAll || matchesBoolean(pure.isAll, false));
  10528. CHqlBoundTarget tempTarget;
  10529. tempTarget.set(target);
  10530. tempTarget.isAll.setown(ctx.getTempDeclare(queryBoolType(), NULL));
  10531. assignCastUnknownLength(ctx, tempTarget, pure);
  10532. return;
  10533. }
  10534. funcName = set2SetXId;
  10535. }
  10536. else
  10537. {
  10538. OwnedHqlExpr values = pure.getTranslatedExpr();
  10539. buildSetAssignViaBuilder(ctx, target, values);
  10540. return;
  10541. }
  10542. break;
  10543. case type_dictionary:
  10544. case type_table:
  10545. case type_groupedtable:
  10546. {
  10547. OwnedHqlExpr src = pure.getTranslatedExpr();
  10548. buildDatasetAssign(ctx, target, src);
  10549. return;
  10550. }
  10551. default:
  10552. assertex(!"Unexpected target type for variable length");
  10553. break;
  10554. }
  10555. args.append(*pure.getTranslatedExpr());
  10556. if(codepageParam)
  10557. args.append(*codepageParam);
  10558. OwnedHqlExpr call = bindFunctionCall(funcName, args);
  10559. buildExprAssign(ctx, target, call);
  10560. }
  10561. void HqlCppTranslator::expandFunctions(bool expandInline)
  10562. {
  10563. if (expandInline)
  10564. {
  10565. BuildCtx ctx(*code, prototypeAtom);
  10566. ForEachItemIn(idx, code->helpers)
  10567. {
  10568. IHqlExpression & cur = (IHqlExpression &)code->helpers.item(idx);
  10569. StringBuffer init;
  10570. if (getAttribute(cur.queryChild(0), initfunctionAtom, init))
  10571. {
  10572. StringBuffer initproto("extern \"C\" void SERVICE_API ");
  10573. initproto.append(init).append("(const char *);");
  10574. ctx.addQuoted(initproto);
  10575. }
  10576. expandFunctionPrototype(ctx, &cur);
  10577. }
  10578. }
  10579. else
  10580. {
  10581. CIArray includes;
  10582. BuildCtx ctx(*code, includeAtom);
  10583. ForEachItemIn(idx, code->helpers)
  10584. {
  10585. //StringBuffer include;
  10586. //IHqlExpression & cur = (IHqlExpression &)code->helpers.item(idx);
  10587. // getLibraryName(cur, include);
  10588. //MORE!! Get the include name...
  10589. }
  10590. }
  10591. }
  10592. void HqlCppTranslator::bindAndPush(BuildCtx & ctx, IHqlExpression * value)
  10593. {
  10594. CHqlBoundExpr bound;
  10595. buildExpr(ctx, value, bound);
  10596. ensurePushed(ctx, bound);
  10597. }
  10598. bool HqlCppTranslator::ensurePushed(BuildCtx & ctx, const CHqlBoundExpr & pure)
  10599. {
  10600. if (!isPushed(pure))
  10601. {
  10602. //Temporary solution - create a critical block whenever the decimals are used.
  10603. OwnedHqlExpr marker = createAttribute(decimalAtom);
  10604. if (!ctx.queryMatchExpr(marker))
  10605. {
  10606. //If a group with no {} is added, we might get a name clash => make it unique
  10607. StringBuffer s;
  10608. getUniqueId(s.append("BcdCriticalBlock bcd")).append(";");
  10609. ctx.addQuoted(s);
  10610. ctx.associateExpr(marker, NULL);
  10611. }
  10612. ITypeInfo * type = pure.expr->queryType();
  10613. HqlExprArray args;
  10614. IIdAtom * funcName = NULL;
  10615. switch (type->getTypeCode())
  10616. {
  10617. case type_data:
  10618. case type_string:
  10619. case type_varstring:
  10620. funcName = DecPushStringId;
  10621. if (type->queryCharset()->queryName() == ebcdicAtom)
  10622. {
  10623. CHqlBoundExpr temp;
  10624. OwnedHqlExpr translated = pure.getTranslatedExpr();
  10625. OwnedHqlExpr cast = createValue(no_cast, getAsciiType(type), translated.getClear());
  10626. buildExpr(ctx, cast, temp);
  10627. args.append(*getBoundLength(temp));
  10628. args.append(*getElementPointer(temp.expr));
  10629. }
  10630. else
  10631. {
  10632. args.append(*getBoundLength(pure));
  10633. args.append(*getElementPointer(pure.expr));
  10634. }
  10635. break;
  10636. case type_qstring:
  10637. funcName = DecPushQStringId;
  10638. args.append(*getBoundLength(pure));
  10639. args.append(*getElementPointer(pure.expr));
  10640. break;
  10641. case type_unicode:
  10642. case type_varunicode:
  10643. funcName = DecPushUnicodeId;
  10644. args.append(*getBoundLength(pure));
  10645. args.append(*getElementPointer(pure.expr));
  10646. break;
  10647. case type_utf8:
  10648. funcName = DecPushUtf8Id;
  10649. args.append(*getBoundLength(pure));
  10650. args.append(*getElementPointer(pure.expr));
  10651. break;
  10652. case type_decimal:
  10653. funcName = type->isSigned() ? DecPushDecimalId : DecPushUDecimalId;
  10654. args.append(*getPointer(pure.expr));
  10655. args.append(*getSizetConstant(type->getSize()));
  10656. args.append(*getSizetConstant(type->getPrecision()));
  10657. break;
  10658. case type_swapint:
  10659. {
  10660. CHqlBoundExpr copyPure;
  10661. copyPure.set(pure);
  10662. //cast via intermediate int.
  10663. OwnedITypeInfo tempType = makeIntType(type->getSize(), type->isSigned());
  10664. CHqlBoundExpr boundCast;
  10665. doBuildExprCast(ctx, tempType, copyPure, boundCast);
  10666. funcName = type->isSigned() ? DecPushInt64Id : DecPushUInt64Id;
  10667. args.append(*boundCast.expr.getLink());
  10668. break;
  10669. }
  10670. //fall through
  10671. case type_int:
  10672. case type_packedint:
  10673. //more signed/unsigned and optimize the length...
  10674. funcName = type->isSigned() ? DecPushInt64Id : DecPushUInt64Id;
  10675. args.append(*pure.expr.getLink());
  10676. break;
  10677. case type_enumerated:
  10678. throwError2(HQLERR_CastXNotImplemented, "map", "decimal");
  10679. break;
  10680. case type_boolean:
  10681. funcName = DecPushLongId;
  10682. args.append(*pure.expr.getLink());
  10683. break;
  10684. case type_real:
  10685. funcName = DecPushRealId;
  10686. args.append(*pure.expr.getLink());
  10687. break;
  10688. default:
  10689. throwError2(HQLERR_CastXNotImplemented, "unknown", "varstring");
  10690. break;
  10691. }
  10692. if (funcName)
  10693. callProcedure(ctx, funcName, args);
  10694. return true;
  10695. }
  10696. return false;
  10697. }
  10698. static StringBuffer & appendCapital(StringBuffer & s, const char * name)
  10699. {
  10700. if (name && name[0])
  10701. {
  10702. s.append((char)toupper(*name));
  10703. s.append(name+1);
  10704. }
  10705. return s;
  10706. }
  10707. bool HqlCppTranslator::expandFunctionPrototype(StringBuffer & s, IHqlExpression * funcdef)
  10708. {
  10709. return generateFunctionPrototype(s, funcdef, options.targetCompiler);
  10710. }
  10711. void HqlCppTranslator::expandFunctionPrototype(BuildCtx & ctx, IHqlExpression * funcdef)
  10712. {
  10713. StringBuffer s;
  10714. if (expandFunctionPrototype(s, funcdef))
  10715. {
  10716. s.append(";");
  10717. IHqlExpression *body = funcdef->queryChild(0);
  10718. IHqlExpression *namespaceAttr = body->queryAttribute(namespaceAtom);
  10719. if (namespaceAttr)
  10720. {
  10721. StringBuffer ns;
  10722. getStringValue(ns, namespaceAttr->queryChild(0));
  10723. ns.insert(0, "namespace ").appendf(" { %s }", s.str());
  10724. ctx.addQuoted(ns);
  10725. }
  10726. else
  10727. ctx.addQuoted(s);
  10728. }
  10729. }
  10730. //Replace no_param with whatever they will have been bound to
  10731. static IHqlExpression *createActualFromFormal(IHqlExpression *param)
  10732. {
  10733. StringBuffer temp;
  10734. ITypeInfo *paramType = param->queryType();
  10735. CHqlBoundExpr bound;
  10736. //Case is significant if these parameters are use for BEGINC++ sections
  10737. IIdAtom * paramName = param->queryId();
  10738. const char * paramNameText = str(lower(paramName));
  10739. Linked<ITypeInfo> type = paramType;
  10740. switch (paramType->getTypeCode())
  10741. {
  10742. case type_set:
  10743. {
  10744. appendCapital(temp.clear().append("isAll"), paramNameText);
  10745. bound.isAll.setown(createVariable(temp.str(), makeBoolType()));
  10746. }
  10747. //fall through
  10748. case type_string:
  10749. case type_qstring:
  10750. case type_data:
  10751. case type_unicode:
  10752. case type_utf8:
  10753. case type_dictionary:
  10754. case type_table:
  10755. case type_groupedtable:
  10756. if (paramType->getSize() == UNKNOWN_LENGTH)
  10757. {
  10758. if (hasOutOfLineModifier(paramType) || hasLinkCountedModifier(paramType))
  10759. {
  10760. appendCapital(temp.clear().append("count"), paramNameText);
  10761. bound.count.setown(createVariable(temp.str(), LINK(sizetType)));
  10762. }
  10763. else
  10764. {
  10765. appendCapital(temp.clear().append("len"), paramNameText);
  10766. bound.length.setown(createVariable(temp.str(), LINK(sizetType)));
  10767. }
  10768. }
  10769. type.setown(makeReferenceModifier(LINK(type)));
  10770. break;
  10771. }
  10772. bound.expr.setown(createVariable(paramNameText, LINK(type)));
  10773. return bound.getTranslatedExpr();
  10774. }
  10775. IHqlExpression * replaceInlineParameters(IHqlExpression * funcdef, IHqlExpression * expr)
  10776. {
  10777. IHqlExpression * body = funcdef->queryChild(0);
  10778. assertex(!body->hasAttribute(oldSetFormatAtom));
  10779. IHqlExpression * formals = funcdef->queryChild(1);
  10780. HqlMapTransformer simpleTransformer;
  10781. StringBuffer paramNameText, temp;
  10782. ForEachChild(i, formals)
  10783. {
  10784. IHqlExpression * param = formals->queryChild(i);
  10785. OwnedHqlExpr actual = createActualFromFormal(param);
  10786. simpleTransformer.setMapping(param, actual);
  10787. }
  10788. return simpleTransformer.transformRoot(expr);
  10789. }
  10790. void HqlCppTranslator::doBuildUserFunctionReturn(BuildCtx & ctx, ITypeInfo * type, IHqlExpression * value)
  10791. {
  10792. if (!options.spotCSE)
  10793. {
  10794. doBuildFunctionReturn(ctx, type, value);
  10795. return;
  10796. }
  10797. switch (value->getOperator())
  10798. {
  10799. case no_if:
  10800. if (false)///disable for the moment - look at changes in klogermann11 to see why, some v.good, some bad.
  10801. {
  10802. //optimize the way that cses are spotted to minimise unnecessary calculations
  10803. OwnedHqlExpr branches = createComma(LINK(value->queryChild(1)), LINK(value->queryChild(2)));
  10804. OwnedHqlExpr cond = LINK(value->queryChild(0));
  10805. spotScalarCSE(cond, branches, NULL, NULL, queryOptions().spotCseInIfDatasetConditions);
  10806. BuildCtx subctx(ctx);
  10807. IHqlStmt * stmt = buildFilterViaExpr(subctx, cond);
  10808. doBuildUserFunctionReturn(subctx, type, branches->queryChild(0));
  10809. subctx.selectElse(stmt);
  10810. doBuildUserFunctionReturn(subctx, type, branches->queryChild(1));
  10811. break;
  10812. }
  10813. default:
  10814. {
  10815. OwnedHqlExpr optimized = spotScalarCSE(value, NULL, queryOptions().spotCseInIfDatasetConditions);
  10816. if (value->isAction())
  10817. buildStmt(ctx, value);
  10818. else
  10819. doBuildFunctionReturn(ctx, type, optimized);
  10820. break;
  10821. }
  10822. }
  10823. }
  10824. void HqlCppTranslator::buildCppFunctionDefinition(BuildCtx & ctx, IHqlExpression * bodyCode, const char *proto)
  10825. {
  10826. processCppBodyDirectives(bodyCode);
  10827. IHqlExpression * location = queryLocation(bodyCode);
  10828. const char * locationFilename = location ? str(location->querySourcePath()) : NULL;
  10829. unsigned startLine = location ? location->getStartLine() : 0;
  10830. IHqlExpression * cppBody = bodyCode->queryChild(0);
  10831. if (cppBody->getOperator() == no_record)
  10832. cppBody = bodyCode->queryChild(1);
  10833. StringBuffer text;
  10834. cppBody->queryValue()->getUTF8Value(text);
  10835. //remove #option, and remove /r so we don't end up with mixed format end of lines.
  10836. text.setLength(cleanupEmbeddedCpp(text.length(), (char*)text.str()));
  10837. const char * start = text.str();
  10838. for (;;)
  10839. {
  10840. char next = *start;
  10841. if (next == '\n')
  10842. startLine++;
  10843. else if (next != '\r')
  10844. break;
  10845. start++;
  10846. }
  10847. const char * body = start;
  10848. const char * cppSeparatorText = "#body";
  10849. const char * separator = strstr(body, cppSeparatorText);
  10850. if (separator)
  10851. {
  10852. if (bodyCode->hasAttribute(inlineAtom))
  10853. throwError(HQLERR_BodyNotAllowedWithInline);
  10854. text.setCharAt(separator-text.str(), 0);
  10855. if (location)
  10856. ctx.addLine(locationFilename, startLine);
  10857. ctx.addQuoted(body);
  10858. if (location)
  10859. ctx.addLine();
  10860. body = separator + strlen(cppSeparatorText);
  10861. if (*body == '\r') body++;
  10862. if (*body == '\n') body++;
  10863. startLine += memcount(body-start, start, '\n');
  10864. }
  10865. bool addPragmas = options.embeddedWarningsAsErrors && !bodyCode->hasAttribute(inlineAtom);
  10866. if (addPragmas)
  10867. {
  10868. ctx.addQuoted("#if defined(__clang__) || (__GNUC__ > 4 || (__GNUC__ == 4 && __GNUC_MINOR__ >= 2))\n"
  10869. "#pragma GCC diagnostic error \"-Wall\"\n"
  10870. "#pragma GCC diagnostic error \"-Wextra\"\n"
  10871. "#pragma GCC diagnostic ignored \"-Wunused-parameter\"\n" // Generated prototype tends to include ctx that is often not used
  10872. "#endif\n");
  10873. }
  10874. MemberFunction userFunc(*this, ctx, proto, MFdynamicproto);
  10875. if (location)
  10876. userFunc.ctx.addLine(locationFilename, startLine);
  10877. userFunc.ctx.addQuoted(body);
  10878. if (location)
  10879. userFunc.ctx.addLine();
  10880. if (addPragmas)
  10881. {
  10882. ctx.addQuoted("\n#if defined(__clang__) || (__GNUC__ > 4 || (__GNUC__ == 4 && __GNUC_MINOR__ >= 2))\n"
  10883. "#pragma GCC diagnostic ignored \"-Wall\"\n"
  10884. "#pragma GCC diagnostic ignored \"-Wextra\"\n"
  10885. "#pragma GCC diagnostic ignored \"-Wunused-variable\"\n" // Some variants of gcc seem to be buggy - this SHOULD be covered by -Wall above but gcc4.8.4 needs it explicit
  10886. "#pragma GCC diagnostic ignored \"-Wparentheses\"\n" // Some variants of gcc seem to be buggy - this SHOULD be covered by -Wall above but gcc4.8.4 needs it explicit
  10887. "#endif\n");
  10888. }
  10889. }
  10890. void HqlCppTranslator::buildScriptFunctionDefinition(BuildCtx &ctx, IHqlExpression * funcdef, const char *proto)
  10891. {
  10892. ITypeInfo * returnType = funcdef->queryType()->queryChildType();
  10893. IHqlExpression * outofline = funcdef->queryChild(0);
  10894. IHqlExpression * formals = funcdef->queryChild(1);
  10895. assertex(outofline->getOperator() == no_outofline);
  10896. IHqlExpression * bodyCode = outofline->queryChild(0);
  10897. IHqlExpression *language = queryAttributeChild(bodyCode, languageAtom, 0);
  10898. bool isImport = bodyCode->hasAttribute(importAtom);
  10899. MemberFunction scriptFunc(*this, ctx, proto, MFdynamicproto);
  10900. BuildCtx & funcctx = scriptFunc.ctx;
  10901. funcctx.associateExpr(codeContextMarkerExpr, codeContextMarkerExpr);
  10902. funcctx.associateExpr(globalContextMarkerExpr, globalContextMarkerExpr);
  10903. HqlExprArray noargs;
  10904. OwnedHqlExpr getPlugin = bindFunctionCall(language, noargs);
  10905. OwnedHqlExpr pluginPtr = createQuoted("Owned<IEmbedContext> __plugin", getPlugin->getType());
  10906. buildAssignToTemp(funcctx, pluginPtr, getPlugin);
  10907. StringBuffer createParam;
  10908. createParam.append("Owned<IEmbedFunctionContext> __ctx = __plugin->createFunctionContextEx(ctx,");
  10909. if (functionBodyIsActivity(bodyCode))
  10910. createParam.append("activity,");
  10911. else
  10912. createParam.append("nullptr,");
  10913. createParam.append(isImport ? "EFimport" : "EFembed");
  10914. if (returnType->getTypeCode()==type_void)
  10915. createParam.append("|EFnoreturn");
  10916. IHqlExpression *optionsParam = nullptr;
  10917. IHqlExpression *queryParam = nullptr;
  10918. unsigned numRealParams = 0;
  10919. ForEachChild(formalIdx, formals)
  10920. {
  10921. IHqlExpression *formal = formals->queryChild(formalIdx);
  10922. if (formal->queryId()==__optionsId)
  10923. optionsParam = formal;
  10924. else if (formal->queryId()==__queryId)
  10925. queryParam = formal;
  10926. else
  10927. numRealParams++;
  10928. }
  10929. if (!numRealParams)
  10930. createParam.append("|EFnoparams");
  10931. if (optionsParam)
  10932. {
  10933. OwnedHqlExpr folded = createActualFromFormal(optionsParam);
  10934. CHqlBoundExpr bound;
  10935. buildExpr(funcctx, folded, bound);
  10936. createParam.append(",");
  10937. generateExprCpp(createParam, bound.expr);
  10938. }
  10939. else
  10940. createParam.append(",NULL");
  10941. createParam.append(");");
  10942. funcctx.addQuoted(createParam);
  10943. OwnedHqlExpr ctxVar = createVariable("__ctx", makeBoolType());
  10944. HqlExprArray scriptArgs;
  10945. scriptArgs.append(*LINK(ctxVar));
  10946. if (bodyCode->hasAttribute(projectedAtom))
  10947. {
  10948. assertex(!isImport);
  10949. // Generate the field list from the output record
  10950. StringBuffer fieldlist;
  10951. IHqlExpression *outRec = bodyCode->queryChild(1);
  10952. assertex(outRec->queryRecordType());
  10953. HqlExprArray fields;
  10954. getSimpleFields(fields, outRec);
  10955. ForEachItemIn(idx, fields)
  10956. {
  10957. IIdAtom *fieldName = fields.item(idx).queryId();
  10958. assertex(fieldName);
  10959. fieldlist.append(',').append(fieldName->queryStr());
  10960. }
  10961. assertex(fieldlist.length());
  10962. LinkedHqlExpr substSearch = queryAttributeChild(bodyCode, projectedAtom, 0);
  10963. assertex (substSearch);
  10964. IValue *substValue = substSearch->queryValue();
  10965. if (queryParam || !substValue)
  10966. {
  10967. HqlExprArray args;
  10968. if (queryParam)
  10969. args.append(*createActualFromFormal(queryParam));
  10970. else
  10971. args.append(*LINK(bodyCode->queryChild(0)));
  10972. args.append(*createConstant(createUtf8Value(fieldlist.length()-1, fieldlist.str()+1, makeUtf8Type(UNKNOWN_LENGTH, NULL))));
  10973. args.append(*LINK(substSearch));
  10974. scriptArgs.append(*bindFunctionCall(substituteEmbeddedScriptId, args,makeUtf8Type(UNKNOWN_LENGTH, NULL)));
  10975. }
  10976. else
  10977. {
  10978. IValue *query = bodyCode->queryChild(0)->queryValue();
  10979. assertex(query);
  10980. StringBuffer origBody;
  10981. query->getUTF8Value(origBody);
  10982. StringBuffer search;
  10983. substValue->getUTF8Value(search);
  10984. rtlDataAttr result;
  10985. unsigned resultLen;
  10986. rtlSubstituteEmbeddedScript(resultLen, result.refstr(), origBody.lengthUtf8(), origBody.str(), fieldlist.lengthUtf8()-1, fieldlist.str()+1, search.lengthUtf8(), search.str());
  10987. scriptArgs.append(*createConstant(createUtf8Value(resultLen, result.getstr(), makeUtf8Type(resultLen, NULL))));
  10988. }
  10989. }
  10990. else
  10991. {
  10992. if (queryParam)
  10993. {
  10994. OwnedHqlExpr query = createActualFromFormal(queryParam);
  10995. scriptArgs.append(*query.getClear());
  10996. }
  10997. else
  10998. scriptArgs.append(*LINK(bodyCode->queryChild(0)));
  10999. }
  11000. if (!bodyCode->hasAttribute(prebindAtom))
  11001. buildFunctionCall(funcctx, isImport ? importId : compileEmbeddedScriptId, scriptArgs);
  11002. ForEachChild(i, formals)
  11003. {
  11004. IHqlExpression * param = formals->queryChild(i);
  11005. if (param == optionsParam || param==queryParam)
  11006. continue;
  11007. HqlExprArray args;
  11008. args.append(*LINK(ctxVar));
  11009. ITypeInfo *paramType = param->queryType();
  11010. IIdAtom * paramId = param->queryId();
  11011. const char * paramNameText = str(paramId);
  11012. if (!options.preserveCaseExternalParameter)
  11013. paramNameText = str(lower(paramId));
  11014. args.append(*createConstant(paramNameText));
  11015. IIdAtom * bindFunc;
  11016. switch (paramType->getTypeCode())
  11017. {
  11018. case type_int:
  11019. if (paramType->getSize()<8)
  11020. {
  11021. bindFunc = paramType->isSigned() ? bindSignedSizeParamId : bindUnsignedSizeParamId;
  11022. args.append(*createIntConstant(paramType->getSize()));
  11023. }
  11024. else
  11025. bindFunc = paramType->isSigned() ? bindSignedParamId : bindUnsignedParamId;
  11026. break;
  11027. case type_varstring:
  11028. bindFunc = bindVStringParamId;
  11029. break;
  11030. case type_string:
  11031. bindFunc = bindStringParamId;
  11032. break;
  11033. case type_real:
  11034. if (paramType->getSize()==4)
  11035. bindFunc = bindFloatParamId;
  11036. else
  11037. bindFunc = bindRealParamId;
  11038. break;
  11039. case type_boolean:
  11040. bindFunc = bindBooleanParamId;
  11041. break;
  11042. case type_utf8:
  11043. bindFunc = bindUtf8ParamId;
  11044. break;
  11045. case type_unicode:
  11046. bindFunc = bindUnicodeParamId;
  11047. break;
  11048. case type_data:
  11049. bindFunc = bindDataParamId;
  11050. break;
  11051. case type_row:
  11052. bindFunc = bindRowParamId;
  11053. break;
  11054. case type_table:
  11055. case type_groupedtable:
  11056. bindFunc = bindDatasetParamId;
  11057. break;
  11058. case type_set:
  11059. {
  11060. bindFunc = bindSetParamId;
  11061. ITypeInfo *childType = paramType->queryChildType();
  11062. type_t typeCode = childType->getTypeCode();
  11063. if (childType->isInteger() && !childType->isSigned())
  11064. typeCode = type_unsigned;
  11065. args.append(*createIntConstant(typeCode));
  11066. args.append(*createIntConstant(childType->getSize()));
  11067. break;
  11068. }
  11069. default:
  11070. StringBuffer typeText;
  11071. getFriendlyTypeStr(paramType, typeText);
  11072. throwError1(HQLERR_EmbeddedTypeNotSupported_X, typeText.str());
  11073. return; // Cannot reach here, but previous throw is virtual, so the compiler cannot be sure it does not return
  11074. }
  11075. args.append(*createActualFromFormal(param));
  11076. buildFunctionCall(funcctx, bindFunc, args);
  11077. }
  11078. if (bodyCode->hasAttribute(prebindAtom))
  11079. buildFunctionCall(funcctx, isImport ? importId : compileEmbeddedScriptId, scriptArgs);
  11080. funcctx.addQuotedLiteral("__ctx->callFunction();");
  11081. IIdAtom * returnFunc;
  11082. HqlExprArray retargs;
  11083. Owned<ITypeInfo> newReturnType;
  11084. retargs.append(*LINK(ctxVar));
  11085. switch (returnType->getTypeCode())
  11086. {
  11087. case type_int:
  11088. returnFunc = returnType->isSigned() ? getSignedResultId : getUnsignedResultId;
  11089. break;
  11090. case type_varstring:
  11091. case type_string:
  11092. returnFunc = getStringResultId;
  11093. break;
  11094. case type_real:
  11095. returnFunc = getRealResultId;
  11096. break;
  11097. case type_boolean:
  11098. returnFunc = getBooleanResultId;
  11099. break;
  11100. case type_unicode:
  11101. returnFunc = getUnicodeResultId;
  11102. break;
  11103. case type_utf8:
  11104. returnFunc = getUTF8ResultId;
  11105. break;
  11106. case type_data:
  11107. returnFunc = getDataResultId;
  11108. break;
  11109. case type_set:
  11110. {
  11111. returnFunc = getSetResultId;
  11112. ITypeInfo *childType = returnType->queryChildType();
  11113. type_t typeCode = childType->getTypeCode();
  11114. if (childType->isInteger() && !childType->isSigned())
  11115. typeCode = type_unsigned;
  11116. retargs.append(*createIntConstant(typeCode));
  11117. retargs.append(*createIntConstant(returnType->queryChildType()->getSize()));
  11118. break;
  11119. }
  11120. case type_row:
  11121. returnFunc = getRowResultId;
  11122. newReturnType.set(returnType);
  11123. break;
  11124. case type_table:
  11125. returnFunc = getDatasetResultId;
  11126. newReturnType.set(returnType);
  11127. break;
  11128. case type_transform:
  11129. returnFunc = getTransformResultId;
  11130. newReturnType.set(returnType);
  11131. break;
  11132. case type_void:
  11133. return;
  11134. default:
  11135. StringBuffer typeText;
  11136. getFriendlyTypeStr(returnType, typeText);
  11137. throwError1(HQLERR_EmbeddedTypeNotSupported_X, typeText.str());
  11138. return; // Cannot reach here, but previous throw is virtual, so the compiler cannot be sure it does not return
  11139. }
  11140. OwnedHqlExpr call = bindFunctionCall(returnFunc, retargs, newReturnType);
  11141. doBuildUserFunctionReturn(funcctx, returnType, call);
  11142. }
  11143. void HqlCppTranslator::buildFunctionDefinition(IHqlExpression * funcdef)
  11144. {
  11145. IHqlExpression * outofline = funcdef->queryChild(0);
  11146. assertex(outofline->getOperator() == no_outofline);
  11147. IHqlExpression * bodyCode = outofline->queryChild(0);
  11148. StringBuffer proto;
  11149. BuildCtx funcctx(*code, helperAtom);
  11150. if (options.spanMultipleCpp)
  11151. {
  11152. const bool inChildActivity = true; // assume the worst
  11153. OwnedHqlExpr pass = getSizetConstant(beginFunctionGetCppIndex(0, inChildActivity));
  11154. funcctx.addGroupPass(pass);
  11155. }
  11156. expandFunctionPrototype(proto, funcdef);
  11157. if (bodyCode->getOperator() == no_embedbody)
  11158. {
  11159. if (bodyCode->hasAttribute(_disallowed_Atom))
  11160. throwError(HQLERR_EmbeddedCppNotAllowed);
  11161. IHqlExpression *languageAttr = bodyCode->queryAttribute(languageAtom);
  11162. if (languageAttr)
  11163. {
  11164. buildScriptFunctionDefinition(funcctx, funcdef, proto);
  11165. }
  11166. else
  11167. {
  11168. bool isInline = bodyCode->hasAttribute(inlineAtom);
  11169. if (isInline)
  11170. {
  11171. if (options.spanMultipleCpp)
  11172. {
  11173. BuildCtx funcctx2(*code, parentHelpersAtom);
  11174. buildCppFunctionDefinition(funcctx2, bodyCode, proto);
  11175. }
  11176. else
  11177. buildCppFunctionDefinition(funcctx, bodyCode, proto);
  11178. }
  11179. else
  11180. {
  11181. BuildCtx funcctx2(*code, userFunctionAtom);
  11182. if (options.spanMultipleCpp)
  11183. {
  11184. OwnedHqlExpr pass = getSizetConstant(beginFunctionGetCppIndex(0, false));
  11185. funcctx2.addGroupPass(pass);
  11186. }
  11187. buildCppFunctionDefinition(funcctx2, bodyCode, proto);
  11188. }
  11189. }
  11190. }
  11191. else
  11192. {
  11193. MemberFunction func(*this, funcctx, proto, MFdynamicproto);
  11194. //MORE: Need to work out how to handle functions that require the context.
  11195. //Need to create a class instead.
  11196. if (functionBodyUsesContext(outofline))
  11197. {
  11198. func.ctx.associateExpr(codeContextMarkerExpr, codeContextMarkerExpr);
  11199. func.ctx.associateExpr(globalContextMarkerExpr, globalContextMarkerExpr);
  11200. }
  11201. OwnedHqlExpr newCode = replaceInlineParameters(funcdef, bodyCode);
  11202. newCode.setown(foldHqlExpression(newCode));
  11203. ITypeInfo * returnType = funcdef->queryType()->queryChildType();
  11204. doBuildUserFunctionReturn(func.ctx, returnType, newCode);
  11205. }
  11206. }
  11207. //---------------------------------------------------------------------------
  11208. void HqlCppTranslator::doBuildPureSubExpr(BuildCtx & ctx, IHqlExpression * expr, CHqlBoundExpr & tgt)
  11209. {
  11210. unsigned max = expr->numChildren();
  11211. if (max == 0)
  11212. tgt.expr.set(expr);
  11213. else
  11214. {
  11215. HqlExprArray args;
  11216. unsigned idx = 0;
  11217. CHqlBoundExpr bound;
  11218. for (idx = 0; idx < max; idx++)
  11219. {
  11220. buildExpr(ctx, expr->queryChild(idx), bound);
  11221. args.append(*bound.expr.getClear());
  11222. }
  11223. tgt.expr.setown(expr->clone(args));
  11224. }
  11225. }
  11226. //---------------------------------------------------------------------------
  11227. IHqlExpression * HqlCppTranslator::getListLength(BuildCtx & ctx, IHqlExpression * expr)
  11228. {
  11229. CHqlBoundExpr bound;
  11230. buildExpr(ctx, expr, bound);
  11231. return getBoundLength(bound);
  11232. }
  11233. IHqlExpression * HqlCppTranslator::getBoundCount(const CHqlBoundExpr & bound)
  11234. {
  11235. if (bound.count)
  11236. return bound.count.getLink();
  11237. ITypeInfo * type = bound.expr->queryType();
  11238. switch (type->getTypeCode())
  11239. {
  11240. case type_array:
  11241. {
  11242. if (bound.length)
  11243. return convertBetweenCountAndSize(bound, true);
  11244. unsigned size = type->getSize();
  11245. if (size != UNKNOWN_LENGTH)
  11246. return getSizetConstant(size / type->queryChildType()->getSize());
  11247. UNIMPLEMENTED;
  11248. }
  11249. case type_dictionary:
  11250. case type_table:
  11251. case type_groupedtable:
  11252. case type_set:
  11253. if (bound.length)
  11254. return convertBetweenCountAndSize(bound, true);
  11255. UNIMPLEMENTED;
  11256. default:
  11257. UNIMPLEMENTED;
  11258. }
  11259. }
  11260. IHqlExpression * HqlCppTranslator::getBoundLength(const CHqlBoundExpr & bound)
  11261. {
  11262. if (bound.length)
  11263. return bound.length.getLink();
  11264. ITypeInfo * type = bound.expr->queryType();
  11265. if (bound.expr->queryValue())
  11266. return getSizetConstant(type->getStringLen());
  11267. switch (type->getTypeCode())
  11268. {
  11269. case type_varstring:
  11270. {
  11271. HqlExprArray args;
  11272. args.append(*getElementPointer(bound.expr));
  11273. return bindTranslatedFunctionCall(strlenId, args);
  11274. }
  11275. case type_varunicode:
  11276. {
  11277. HqlExprArray args;
  11278. args.append(*getElementPointer(bound.expr));
  11279. return bindTranslatedFunctionCall(unicodeStrlenId, args);
  11280. }
  11281. case type_set:
  11282. case type_array:
  11283. case type_dictionary:
  11284. case type_table:
  11285. case type_groupedtable:
  11286. assertex(!isArrayRowset(type));
  11287. if (bound.count)
  11288. return convertBetweenCountAndSize(bound, false);
  11289. UNIMPLEMENTED;
  11290. case type_utf8:
  11291. {
  11292. assertex(type->getSize() != UNKNOWN_LENGTH);
  11293. HqlExprArray args;
  11294. args.append(*getSizetConstant(type->getSize()));
  11295. args.append(*getElementPointer(bound.expr));
  11296. return bindTranslatedFunctionCall(utf8LengthId, args);
  11297. }
  11298. default:
  11299. return getSizetConstant(type->getStringLen());
  11300. }
  11301. }
  11302. IHqlExpression * HqlCppTranslator::getBoundSize(ITypeInfo * type, IHqlExpression * length, IHqlExpression * data)
  11303. {
  11304. type_t tc = type->getTypeCode();
  11305. switch (tc)
  11306. {
  11307. case type_qstring:
  11308. {
  11309. if (length->queryValue())
  11310. return getSizetConstant((size32_t)rtlQStrSize((size32_t)length->queryValue()->getIntValue()));
  11311. HqlExprArray args;
  11312. args.append(*LINK(length));
  11313. return bindTranslatedFunctionCall(qstrSizeId, args);
  11314. }
  11315. case type_varstring:
  11316. return adjustValue(length, 1);
  11317. case type_varunicode:
  11318. {
  11319. OwnedHqlExpr temp = adjustValue(length, 1);
  11320. return multiplyValue(temp, 2);
  11321. }
  11322. case type_unicode:
  11323. return multiplyValue(length, 2);
  11324. case type_utf8:
  11325. {
  11326. assertex(data);
  11327. if (data->queryValue())
  11328. return getSizetConstant(data->queryValue()->getSize());
  11329. HqlExprArray args;
  11330. args.append(*LINK(length));
  11331. args.append(*getElementPointer(data));
  11332. return bindTranslatedFunctionCall(utf8SizeId, args);
  11333. }
  11334. case type_array:
  11335. case type_set:
  11336. return LINK(length);
  11337. default:
  11338. return LINK(length);
  11339. }
  11340. }
  11341. IHqlExpression * HqlCppTranslator::getBoundSize(const CHqlBoundExpr & bound)
  11342. {
  11343. ITypeInfo * type = bound.expr->queryType();
  11344. if (bound.length)
  11345. return getBoundSize(type, bound.length, bound.expr);
  11346. type_t tc = type->getTypeCode();
  11347. if (tc == type_row)
  11348. {
  11349. if (hasReferenceModifier(type))
  11350. return getSizetConstant(sizeof(void*));
  11351. IHqlExpression * record = ::queryRecord(type);
  11352. if (isFixedSizeRecord(record))
  11353. return getSizetConstant(getMinRecordSize(record));
  11354. //call meta function mm.queryRecordSize(&row)
  11355. StringBuffer metaInstance, temp;
  11356. buildMetaForRecord(metaInstance, record);
  11357. temp.append(metaInstance).append(".getRecordSize(");
  11358. OwnedHqlExpr rowAddr = getPointer(bound.expr);
  11359. generateExprCpp(temp, rowAddr);
  11360. temp.append(")");
  11361. return createQuoted(temp.str(), LINK(sizetType));
  11362. }
  11363. if (type->getSize() != UNKNOWN_LENGTH)
  11364. return getSizetConstant(type->getSize());
  11365. OwnedHqlExpr length = getBoundLength(bound);
  11366. return getBoundSize(type, length, bound.expr);
  11367. }
  11368. IHqlExpression * HqlCppTranslator::getFirstCharacter(IHqlExpression * source)
  11369. {
  11370. if (source->getOperator() == no_constant)
  11371. {
  11372. StringBuffer temp;
  11373. source->queryValue()->getStringValue(temp);
  11374. return createUIntConstant((unsigned char)temp.charAt(0));
  11375. }
  11376. return createValue(no_index, makeCharType(), LINK(source), getZero());
  11377. }
  11378. IHqlExpression * HqlCppTranslator::getElementPointer(IHqlExpression * source)
  11379. {
  11380. ITypeInfo * srcType = source->queryType();
  11381. switch (srcType->getTypeCode())
  11382. {
  11383. case type_string:
  11384. case type_data:
  11385. case type_qstring:
  11386. case type_varstring:
  11387. case type_unicode:
  11388. case type_utf8:
  11389. case type_varunicode:
  11390. case type_set:
  11391. case type_array:
  11392. break;
  11393. default:
  11394. throwUnexpectedType(srcType);
  11395. }
  11396. if (source->getOperator() == no_constant)
  11397. return LINK(source);
  11398. OwnedHqlExpr pointer = getPointer(source);
  11399. return ensureIndexable(pointer);
  11400. }
  11401. /* All in params: NOT linked */
  11402. IHqlExpression * HqlCppTranslator::getIndexedElementPointer(IHqlExpression * source, IHqlExpression * index)
  11403. {
  11404. ITypeInfo * srcType = source->queryType();
  11405. switch (srcType->getTypeCode())
  11406. {
  11407. case type_string:
  11408. case type_data:
  11409. case type_qstring:
  11410. case type_varstring:
  11411. case type_unicode:
  11412. case type_utf8:
  11413. case type_varunicode:
  11414. break;
  11415. default:
  11416. throwUnexpectedType(srcType);
  11417. }
  11418. if (!index)
  11419. return getElementPointer(source);
  11420. IValue * value = index->queryValue();
  11421. if (value && value->getIntValue() == 0)
  11422. return getElementPointer(source);
  11423. ITypeInfo * refType = LINK(srcType);
  11424. if (!srcType->isReference())
  11425. refType = makeReferenceModifier(refType);
  11426. OwnedHqlExpr temp;
  11427. if (srcType->getTypeCode() == type_utf8)
  11428. {
  11429. HqlExprArray args;
  11430. args.append(*LINK(index));
  11431. args.append(*getElementPointer(source));
  11432. temp.setown(bindTranslatedFunctionCall(utf8SizeId, args));
  11433. index = temp;
  11434. }
  11435. //special case string indexing
  11436. if (source->getOperator() != no_constant)
  11437. {
  11438. if (!srcType->isReference() && !hasWrapperModifier(srcType))
  11439. return createValue(no_address, refType, createValue(no_index, makeCharType(), ensureIndexable(source), LINK(index)));
  11440. }
  11441. return createValue(no_add, refType, ensureIndexable(source), LINK(index));
  11442. }
  11443. IHqlExpression * HqlCppTranslator::getIndexedElementPointer(IHqlExpression * source, unsigned index)
  11444. {
  11445. if (!index)
  11446. return getElementPointer(source);
  11447. OwnedHqlExpr ival = getSizetConstant(index);
  11448. return getIndexedElementPointer(source, ival);
  11449. }
  11450. IHqlExpression * HqlCppTranslator::needFunction(IIdAtom * name)
  11451. {
  11452. assertex(name);
  11453. HqlDummyLookupContext dummyctx(errorProcessor);
  11454. return internalScope->lookupSymbol(name, LSFsharedOK, dummyctx);
  11455. }
  11456. bool HqlCppTranslator::childrenRequireTemp(BuildCtx & ctx, IHqlExpression * expr, bool includeChildren)
  11457. {
  11458. unsigned numArgs = expr->numChildren();
  11459. for (unsigned index = 0; index < numArgs; index++)
  11460. if (requiresTemp(ctx, expr->queryChild(index), includeChildren))
  11461. return true;
  11462. return false;
  11463. }
  11464. bool HqlCppTranslator::requiresTemp(BuildCtx & ctx, IHqlExpression * expr, bool includeChildren)
  11465. {
  11466. switch (expr->getOperator())
  11467. {
  11468. case no_attr:
  11469. case no_attr_link:
  11470. case no_attr_expr:
  11471. case no_quoted:
  11472. case no_variable:
  11473. case no_constant:
  11474. case no_translated:
  11475. case no_matchtext:
  11476. case no_matchunicode:
  11477. case no_matchlength:
  11478. case no_matchattr:
  11479. case no_matchrow:
  11480. case no_matchutf8:
  11481. case no_libraryinput:
  11482. return false;
  11483. case no_getresult:
  11484. case no_getgraphresult:
  11485. case no_workunit_dataset:
  11486. return false; // if in an activity, then will be in setContext, if not then don't really care
  11487. case no_preservemeta:
  11488. return requiresTemp(ctx, expr->queryChild(0), includeChildren);
  11489. case no_alias:
  11490. {
  11491. if (expr->isPure() && ctx.queryMatchExpr(expr->queryChild(0)))
  11492. return false;
  11493. if (!containsActiveDataset(expr)) // generates a earlier temp even if generating within the onCreate() function
  11494. return false;
  11495. return true;
  11496. }
  11497. case no_select:
  11498. {
  11499. bool isNew;
  11500. IHqlExpression * ds = querySelectorDataset(expr, isNew);
  11501. if (isNew)
  11502. {
  11503. if (!ds->isPure() || !ds->isDatarow())
  11504. return true;
  11505. if (!ctx.queryAssociation(ds, AssocRow, NULL))
  11506. return true;
  11507. }
  11508. return false;
  11509. }
  11510. case no_field:
  11511. throwUnexpected();
  11512. return false; // more, depends on whether conditional etc.
  11513. case no_sizeof:
  11514. case no_offsetof:
  11515. return false; /// auto creates one anyway.
  11516. case no_typetransfer:
  11517. switch (expr->queryChild(0)->queryType()->getTypeCode())
  11518. {
  11519. case type_qstring:
  11520. case type_string:
  11521. case type_data:
  11522. case type_varstring:
  11523. break;
  11524. default:
  11525. return true;
  11526. }
  11527. break;
  11528. case no_substring:
  11529. {
  11530. SubStringInfo info(expr);
  11531. if (!info.canGenerateInline() && !expr->hasAttribute(quickAtom))
  11532. return true;
  11533. break;
  11534. }
  11535. case no_call:
  11536. case no_externalcall:
  11537. {
  11538. ITypeInfo * type = expr->queryType();
  11539. switch (type->getTypeCode())
  11540. {
  11541. case type_string:
  11542. case type_data:
  11543. case type_qstring:
  11544. case type_varstring:
  11545. case type_unicode:
  11546. case type_varunicode:
  11547. case type_utf8:
  11548. return true;
  11549. }
  11550. if (isTimed(expr))
  11551. return true;
  11552. break;
  11553. }
  11554. case no_cast:
  11555. case no_implicitcast:
  11556. {
  11557. ITypeInfo * type = expr->queryType();
  11558. IHqlExpression * child = expr->queryChild(0);
  11559. switch (type->getTypeCode())
  11560. {
  11561. case type_string:
  11562. case type_data:
  11563. if (!canRemoveStringCast(type, child->queryType()))
  11564. return true;
  11565. break;
  11566. case type_varstring:
  11567. if ((type->getSize() != UNKNOWN_LENGTH) || (child->queryType()->getTypeCode() != type_varstring))
  11568. return true;
  11569. break;
  11570. }
  11571. }
  11572. break;
  11573. case no_eq:
  11574. case no_ne:
  11575. case no_lt:
  11576. case no_le:
  11577. case no_gt:
  11578. case no_ge:
  11579. {
  11580. if (!includeChildren)
  11581. return false;
  11582. unsigned numArgs = expr->numChildren();
  11583. for (unsigned index = 0; index < numArgs; index++)
  11584. {
  11585. OwnedHqlExpr cur = getSimplifyCompareArg(expr->queryChild(index));
  11586. //decimal comparisons can't be short circuited because they might cause a bcd stack overflow.
  11587. if (cur->queryType()->getTypeCode() == type_decimal)
  11588. return true;
  11589. if (requiresTemp(ctx, cur, true))
  11590. return true;
  11591. }
  11592. return false;
  11593. }
  11594. case no_mul:
  11595. case no_div:
  11596. case no_modulus:
  11597. case no_add:
  11598. case no_sub:
  11599. case no_and:
  11600. case no_or:
  11601. case no_xor:
  11602. case no_lshift:
  11603. case no_rshift:
  11604. case no_comma:
  11605. case no_compound:
  11606. case no_band:
  11607. case no_bor:
  11608. case no_bxor:
  11609. case no_pselect:
  11610. case no_index:
  11611. case no_postinc:
  11612. case no_postdec:
  11613. case no_negate:
  11614. case no_not:
  11615. case no_bnot:
  11616. case no_address:
  11617. case no_deref:
  11618. case no_preinc:
  11619. case no_predec:
  11620. case no_if:
  11621. case no_charlen:
  11622. break;
  11623. case no_between:
  11624. //Assume the worse for decimals.
  11625. if (expr->queryChild(0)->queryType()->getTypeCode() == type_decimal)
  11626. return true;
  11627. break;
  11628. case no_order:
  11629. case no_crc:
  11630. case no_hash:
  11631. case no_hash32:
  11632. case no_hash64:
  11633. case no_hashmd5:
  11634. case no_abs:
  11635. return true;
  11636. default:
  11637. return true;
  11638. }
  11639. if (includeChildren)
  11640. return childrenRequireTemp(ctx, expr, includeChildren);
  11641. return false;
  11642. }
  11643. bool HqlCppTranslator::requiresTempAfterFirst(BuildCtx & ctx, IHqlExpression * expr)
  11644. {
  11645. unsigned numArgs = expr->numChildren();
  11646. for (unsigned index = 1; index < numArgs; index++)
  11647. if (requiresTemp(ctx, expr->queryChild(index), true))
  11648. return true;
  11649. return false;
  11650. }
  11651. void HqlCppTranslator::useFunction(IHqlExpression * func)
  11652. {
  11653. code->useFunction(func);
  11654. }
  11655. void HqlCppTranslator::useLibrary(const char * libname)
  11656. {
  11657. code->useLibrary(libname);
  11658. }
  11659. //===========================================================================
  11660. static unique_id_t queryInstance = 0;
  11661. HqlQueryInstance::HqlQueryInstance()
  11662. {
  11663. instance = ++queryInstance;
  11664. }
  11665. StringBuffer & HqlQueryInstance::queryDllName(StringBuffer & out)
  11666. {
  11667. return out.append("query").append(instance);
  11668. }