hqlsource.cpp 270 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666166716681669167016711672167316741675167616771678167916801681168216831684168516861687168816891690169116921693169416951696169716981699170017011702170317041705170617071708170917101711171217131714171517161717171817191720172117221723172417251726172717281729173017311732173317341735173617371738173917401741174217431744174517461747174817491750175117521753175417551756175717581759176017611762176317641765176617671768176917701771177217731774177517761777177817791780178117821783178417851786178717881789179017911792179317941795179617971798179918001801180218031804180518061807180818091810181118121813181418151816181718181819182018211822182318241825182618271828182918301831183218331834183518361837183818391840184118421843184418451846184718481849185018511852185318541855185618571858185918601861186218631864186518661867186818691870187118721873187418751876187718781879188018811882188318841885188618871888188918901891189218931894189518961897189818991900190119021903190419051906190719081909191019111912191319141915191619171918191919201921192219231924192519261927192819291930193119321933193419351936193719381939194019411942194319441945194619471948194919501951195219531954195519561957195819591960196119621963196419651966196719681969197019711972197319741975197619771978197919801981198219831984198519861987198819891990199119921993199419951996199719981999200020012002200320042005200620072008200920102011201220132014201520162017201820192020202120222023202420252026202720282029203020312032203320342035203620372038203920402041204220432044204520462047204820492050205120522053205420552056205720582059206020612062206320642065206620672068206920702071207220732074207520762077207820792080208120822083208420852086208720882089209020912092209320942095209620972098209921002101210221032104210521062107210821092110211121122113211421152116211721182119212021212122212321242125212621272128212921302131213221332134213521362137213821392140214121422143214421452146214721482149215021512152215321542155215621572158215921602161216221632164216521662167216821692170217121722173217421752176217721782179218021812182218321842185218621872188218921902191219221932194219521962197219821992200220122022203220422052206220722082209221022112212221322142215221622172218221922202221222222232224222522262227222822292230223122322233223422352236223722382239224022412242224322442245224622472248224922502251225222532254225522562257225822592260226122622263226422652266226722682269227022712272227322742275227622772278227922802281228222832284228522862287228822892290229122922293229422952296229722982299230023012302230323042305230623072308230923102311231223132314231523162317231823192320232123222323232423252326232723282329233023312332233323342335233623372338233923402341234223432344234523462347234823492350235123522353235423552356235723582359236023612362236323642365236623672368236923702371237223732374237523762377237823792380238123822383238423852386238723882389239023912392239323942395239623972398239924002401240224032404240524062407240824092410241124122413241424152416241724182419242024212422242324242425242624272428242924302431243224332434243524362437243824392440244124422443244424452446244724482449245024512452245324542455245624572458245924602461246224632464246524662467246824692470247124722473247424752476247724782479248024812482248324842485248624872488248924902491249224932494249524962497249824992500250125022503250425052506250725082509251025112512251325142515251625172518251925202521252225232524252525262527252825292530253125322533253425352536253725382539254025412542254325442545254625472548254925502551255225532554255525562557255825592560256125622563256425652566256725682569257025712572257325742575257625772578257925802581258225832584258525862587258825892590259125922593259425952596259725982599260026012602260326042605260626072608260926102611261226132614261526162617261826192620262126222623262426252626262726282629263026312632263326342635263626372638263926402641264226432644264526462647264826492650265126522653265426552656265726582659266026612662266326642665266626672668266926702671267226732674267526762677267826792680268126822683268426852686268726882689269026912692269326942695269626972698269927002701270227032704270527062707270827092710271127122713271427152716271727182719272027212722272327242725272627272728272927302731273227332734273527362737273827392740274127422743274427452746274727482749275027512752275327542755275627572758275927602761276227632764276527662767276827692770277127722773277427752776277727782779278027812782278327842785278627872788278927902791279227932794279527962797279827992800280128022803280428052806280728082809281028112812281328142815281628172818281928202821282228232824282528262827282828292830283128322833283428352836283728382839284028412842284328442845284628472848284928502851285228532854285528562857285828592860286128622863286428652866286728682869287028712872287328742875287628772878287928802881288228832884288528862887288828892890289128922893289428952896289728982899290029012902290329042905290629072908290929102911291229132914291529162917291829192920292129222923292429252926292729282929293029312932293329342935293629372938293929402941294229432944294529462947294829492950295129522953295429552956295729582959296029612962296329642965296629672968296929702971297229732974297529762977297829792980298129822983298429852986298729882989299029912992299329942995299629972998299930003001300230033004300530063007300830093010301130123013301430153016301730183019302030213022302330243025302630273028302930303031303230333034303530363037303830393040304130423043304430453046304730483049305030513052305330543055305630573058305930603061306230633064306530663067306830693070307130723073307430753076307730783079308030813082308330843085308630873088308930903091309230933094309530963097309830993100310131023103310431053106310731083109311031113112311331143115311631173118311931203121312231233124312531263127312831293130313131323133313431353136313731383139314031413142314331443145314631473148314931503151315231533154315531563157315831593160316131623163316431653166316731683169317031713172317331743175317631773178317931803181318231833184318531863187318831893190319131923193319431953196319731983199320032013202320332043205320632073208320932103211321232133214321532163217321832193220322132223223322432253226322732283229323032313232323332343235323632373238323932403241324232433244324532463247324832493250325132523253325432553256325732583259326032613262326332643265326632673268326932703271327232733274327532763277327832793280328132823283328432853286328732883289329032913292329332943295329632973298329933003301330233033304330533063307330833093310331133123313331433153316331733183319332033213322332333243325332633273328332933303331333233333334333533363337333833393340334133423343334433453346334733483349335033513352335333543355335633573358335933603361336233633364336533663367336833693370337133723373337433753376337733783379338033813382338333843385338633873388338933903391339233933394339533963397339833993400340134023403340434053406340734083409341034113412341334143415341634173418341934203421342234233424342534263427342834293430343134323433343434353436343734383439344034413442344334443445344634473448344934503451345234533454345534563457345834593460346134623463346434653466346734683469347034713472347334743475347634773478347934803481348234833484348534863487348834893490349134923493349434953496349734983499350035013502350335043505350635073508350935103511351235133514351535163517351835193520352135223523352435253526352735283529353035313532353335343535353635373538353935403541354235433544354535463547354835493550355135523553355435553556355735583559356035613562356335643565356635673568356935703571357235733574357535763577357835793580358135823583358435853586358735883589359035913592359335943595359635973598359936003601360236033604360536063607360836093610361136123613361436153616361736183619362036213622362336243625362636273628362936303631363236333634363536363637363836393640364136423643364436453646364736483649365036513652365336543655365636573658365936603661366236633664366536663667366836693670367136723673367436753676367736783679368036813682368336843685368636873688368936903691369236933694369536963697369836993700370137023703370437053706370737083709371037113712371337143715371637173718371937203721372237233724372537263727372837293730373137323733373437353736373737383739374037413742374337443745374637473748374937503751375237533754375537563757375837593760376137623763376437653766376737683769377037713772377337743775377637773778377937803781378237833784378537863787378837893790379137923793379437953796379737983799380038013802380338043805380638073808380938103811381238133814381538163817381838193820382138223823382438253826382738283829383038313832383338343835383638373838383938403841384238433844384538463847384838493850385138523853385438553856385738583859386038613862386338643865386638673868386938703871387238733874387538763877387838793880388138823883388438853886388738883889389038913892389338943895389638973898389939003901390239033904390539063907390839093910391139123913391439153916391739183919392039213922392339243925392639273928392939303931393239333934393539363937393839393940394139423943394439453946394739483949395039513952395339543955395639573958395939603961396239633964396539663967396839693970397139723973397439753976397739783979398039813982398339843985398639873988398939903991399239933994399539963997399839994000400140024003400440054006400740084009401040114012401340144015401640174018401940204021402240234024402540264027402840294030403140324033403440354036403740384039404040414042404340444045404640474048404940504051405240534054405540564057405840594060406140624063406440654066406740684069407040714072407340744075407640774078407940804081408240834084408540864087408840894090409140924093409440954096409740984099410041014102410341044105410641074108410941104111411241134114411541164117411841194120412141224123412441254126412741284129413041314132413341344135413641374138413941404141414241434144414541464147414841494150415141524153415441554156415741584159416041614162416341644165416641674168416941704171417241734174417541764177417841794180418141824183418441854186418741884189419041914192419341944195419641974198419942004201420242034204420542064207420842094210421142124213421442154216421742184219422042214222422342244225422642274228422942304231423242334234423542364237423842394240424142424243424442454246424742484249425042514252425342544255425642574258425942604261426242634264426542664267426842694270427142724273427442754276427742784279428042814282428342844285428642874288428942904291429242934294429542964297429842994300430143024303430443054306430743084309431043114312431343144315431643174318431943204321432243234324432543264327432843294330433143324333433443354336433743384339434043414342434343444345434643474348434943504351435243534354435543564357435843594360436143624363436443654366436743684369437043714372437343744375437643774378437943804381438243834384438543864387438843894390439143924393439443954396439743984399440044014402440344044405440644074408440944104411441244134414441544164417441844194420442144224423442444254426442744284429443044314432443344344435443644374438443944404441444244434444444544464447444844494450445144524453445444554456445744584459446044614462446344644465446644674468446944704471447244734474447544764477447844794480448144824483448444854486448744884489449044914492449344944495449644974498449945004501450245034504450545064507450845094510451145124513451445154516451745184519452045214522452345244525452645274528452945304531453245334534453545364537453845394540454145424543454445454546454745484549455045514552455345544555455645574558455945604561456245634564456545664567456845694570457145724573457445754576457745784579458045814582458345844585458645874588458945904591459245934594459545964597459845994600460146024603460446054606460746084609461046114612461346144615461646174618461946204621462246234624462546264627462846294630463146324633463446354636463746384639464046414642464346444645464646474648464946504651465246534654465546564657465846594660466146624663466446654666466746684669467046714672467346744675467646774678467946804681468246834684468546864687468846894690469146924693469446954696469746984699470047014702470347044705470647074708470947104711471247134714471547164717471847194720472147224723472447254726472747284729473047314732473347344735473647374738473947404741474247434744474547464747474847494750475147524753475447554756475747584759476047614762476347644765476647674768476947704771477247734774477547764777477847794780478147824783478447854786478747884789479047914792479347944795479647974798479948004801480248034804480548064807480848094810481148124813481448154816481748184819482048214822482348244825482648274828482948304831483248334834483548364837483848394840484148424843484448454846484748484849485048514852485348544855485648574858485948604861486248634864486548664867486848694870487148724873487448754876487748784879488048814882488348844885488648874888488948904891489248934894489548964897489848994900490149024903490449054906490749084909491049114912491349144915491649174918491949204921492249234924492549264927492849294930493149324933493449354936493749384939494049414942494349444945494649474948494949504951495249534954495549564957495849594960496149624963496449654966496749684969497049714972497349744975497649774978497949804981498249834984498549864987498849894990499149924993499449954996499749984999500050015002500350045005500650075008500950105011501250135014501550165017501850195020502150225023502450255026502750285029503050315032503350345035503650375038503950405041504250435044504550465047504850495050505150525053505450555056505750585059506050615062506350645065506650675068506950705071507250735074507550765077507850795080508150825083508450855086508750885089509050915092509350945095509650975098509951005101510251035104510551065107510851095110511151125113511451155116511751185119512051215122512351245125512651275128512951305131513251335134513551365137513851395140514151425143514451455146514751485149515051515152515351545155515651575158515951605161516251635164516551665167516851695170517151725173517451755176517751785179518051815182518351845185518651875188518951905191519251935194519551965197519851995200520152025203520452055206520752085209521052115212521352145215521652175218521952205221522252235224522552265227522852295230523152325233523452355236523752385239524052415242524352445245524652475248524952505251525252535254525552565257525852595260526152625263526452655266526752685269527052715272527352745275527652775278527952805281528252835284528552865287528852895290529152925293529452955296529752985299530053015302530353045305530653075308530953105311531253135314531553165317531853195320532153225323532453255326532753285329533053315332533353345335533653375338533953405341534253435344534553465347534853495350535153525353535453555356535753585359536053615362536353645365536653675368536953705371537253735374537553765377537853795380538153825383538453855386538753885389539053915392539353945395539653975398539954005401540254035404540554065407540854095410541154125413541454155416541754185419542054215422542354245425542654275428542954305431543254335434543554365437543854395440544154425443544454455446544754485449545054515452545354545455545654575458545954605461546254635464546554665467546854695470547154725473547454755476547754785479548054815482548354845485548654875488548954905491549254935494549554965497549854995500550155025503550455055506550755085509551055115512551355145515551655175518551955205521552255235524552555265527552855295530553155325533553455355536553755385539554055415542554355445545554655475548554955505551555255535554555555565557555855595560556155625563556455655566556755685569557055715572557355745575557655775578557955805581558255835584558555865587558855895590559155925593559455955596559755985599560056015602560356045605560656075608560956105611561256135614561556165617561856195620562156225623562456255626562756285629563056315632563356345635563656375638563956405641564256435644564556465647564856495650565156525653565456555656565756585659566056615662566356645665566656675668566956705671567256735674567556765677567856795680568156825683568456855686568756885689569056915692569356945695569656975698569957005701570257035704570557065707570857095710571157125713571457155716571757185719572057215722572357245725572657275728572957305731573257335734573557365737573857395740574157425743574457455746574757485749575057515752575357545755575657575758575957605761576257635764576557665767576857695770577157725773577457755776577757785779578057815782578357845785578657875788578957905791579257935794579557965797579857995800580158025803580458055806580758085809581058115812581358145815581658175818581958205821582258235824582558265827582858295830583158325833583458355836583758385839584058415842584358445845584658475848584958505851585258535854585558565857585858595860586158625863586458655866586758685869587058715872587358745875587658775878587958805881588258835884588558865887588858895890589158925893589458955896589758985899590059015902590359045905590659075908590959105911591259135914591559165917591859195920592159225923592459255926592759285929593059315932593359345935593659375938593959405941594259435944594559465947594859495950595159525953595459555956595759585959596059615962596359645965596659675968596959705971597259735974597559765977597859795980598159825983598459855986598759885989599059915992599359945995599659975998599960006001600260036004600560066007600860096010601160126013601460156016601760186019602060216022602360246025602660276028602960306031603260336034603560366037603860396040604160426043604460456046604760486049605060516052605360546055605660576058605960606061606260636064606560666067606860696070607160726073607460756076607760786079608060816082608360846085608660876088608960906091609260936094609560966097609860996100610161026103610461056106610761086109611061116112611361146115611661176118611961206121612261236124612561266127612861296130613161326133613461356136613761386139614061416142614361446145614661476148614961506151615261536154615561566157615861596160616161626163616461656166616761686169617061716172617361746175617661776178617961806181618261836184618561866187618861896190619161926193619461956196619761986199620062016202620362046205620662076208620962106211621262136214621562166217621862196220622162226223622462256226622762286229623062316232623362346235623662376238623962406241624262436244624562466247624862496250625162526253625462556256625762586259626062616262626362646265626662676268626962706271627262736274627562766277627862796280628162826283628462856286628762886289629062916292629362946295629662976298629963006301630263036304630563066307630863096310631163126313631463156316631763186319632063216322632363246325632663276328632963306331633263336334633563366337633863396340634163426343634463456346634763486349635063516352635363546355635663576358635963606361636263636364636563666367636863696370637163726373637463756376637763786379638063816382638363846385638663876388638963906391639263936394639563966397639863996400640164026403640464056406640764086409641064116412641364146415641664176418641964206421642264236424642564266427642864296430643164326433643464356436643764386439644064416442644364446445644664476448644964506451645264536454645564566457645864596460646164626463646464656466646764686469647064716472647364746475647664776478647964806481648264836484648564866487648864896490649164926493649464956496649764986499650065016502650365046505650665076508650965106511651265136514651565166517651865196520652165226523652465256526652765286529653065316532653365346535653665376538653965406541654265436544654565466547654865496550655165526553655465556556655765586559656065616562656365646565656665676568656965706571657265736574657565766577657865796580658165826583658465856586658765886589659065916592659365946595659665976598659966006601660266036604660566066607660866096610661166126613661466156616661766186619662066216622662366246625662666276628662966306631663266336634663566366637663866396640664166426643664466456646664766486649665066516652665366546655665666576658665966606661666266636664666566666667666866696670667166726673667466756676667766786679668066816682668366846685668666876688668966906691669266936694669566966697669866996700670167026703670467056706670767086709671067116712671367146715671667176718671967206721672267236724672567266727672867296730673167326733673467356736673767386739674067416742674367446745674667476748674967506751675267536754675567566757675867596760676167626763676467656766676767686769677067716772677367746775677667776778677967806781678267836784678567866787678867896790679167926793679467956796679767986799680068016802680368046805680668076808680968106811681268136814681568166817681868196820682168226823682468256826682768286829683068316832683368346835683668376838683968406841684268436844684568466847684868496850685168526853685468556856685768586859686068616862686368646865686668676868686968706871687268736874687568766877687868796880688168826883688468856886688768886889689068916892689368946895689668976898689969006901690269036904690569066907690869096910691169126913691469156916691769186919692069216922692369246925692669276928692969306931693269336934693569366937693869396940694169426943694469456946694769486949695069516952695369546955695669576958695969606961696269636964696569666967696869696970697169726973697469756976697769786979698069816982698369846985698669876988698969906991699269936994699569966997699869997000700170027003700470057006700770087009701070117012701370147015701670177018701970207021702270237024702570267027702870297030703170327033703470357036703770387039704070417042704370447045704670477048704970507051705270537054705570567057705870597060706170627063706470657066706770687069707070717072707370747075707670777078707970807081708270837084708570867087708870897090709170927093709470957096709770987099710071017102710371047105710671077108710971107111711271137114711571167117711871197120712171227123712471257126712771287129713071317132713371347135713671377138713971407141714271437144714571467147714871497150715171527153715471557156715771587159716071617162716371647165716671677168716971707171717271737174717571767177717871797180718171827183718471857186718771887189719071917192719371947195719671977198719972007201720272037204720572067207720872097210721172127213721472157216721772187219722072217222722372247225722672277228722972307231723272337234723572367237723872397240724172427243724472457246724772487249725072517252725372547255725672577258725972607261726272637264726572667267726872697270727172727273727472757276727772787279728072817282728372847285728672877288728972907291729272937294729572967297729872997300730173027303730473057306730773087309731073117312731373147315731673177318731973207321732273237324732573267327732873297330733173327333733473357336733773387339734073417342734373447345734673477348734973507351735273537354735573567357735873597360736173627363
  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 "jliball.hpp"
  14. #include "hql.hpp"
  15. #include "platform.h"
  16. #include "jlib.hpp"
  17. #include "jmisc.hpp"
  18. #include "jstream.ipp"
  19. #include "jdebug.hpp"
  20. #include "eclrtl_imp.hpp"
  21. #include "hql.hpp"
  22. #include "hqlattr.hpp"
  23. #include "hqlmeta.hpp"
  24. #include "hqlthql.hpp"
  25. #include "hqlhtcpp.ipp"
  26. #include "hqlttcpp.ipp"
  27. #include "hqlutil.hpp"
  28. #include "hqlthql.hpp"
  29. #include "hqlwcpp.hpp"
  30. #include "hqlcpputil.hpp"
  31. #include "hqltcppc.ipp"
  32. #include "hqlopt.hpp"
  33. #include "hqlfold.hpp"
  34. #include "hqlcerrors.hpp"
  35. #include "hqlcatom.hpp"
  36. #include "hqltrans.ipp"
  37. #include "hqlpmap.hpp"
  38. #include "hqlttcpp.ipp"
  39. #include "hqlsource.ipp"
  40. #include "hqlcse.ipp"
  41. #include "hqliter.ipp"
  42. #include "thorcommon.hpp"
  43. #include "hqlinline.hpp"
  44. //#define FLATTEN_DATASETS
  45. //#define HACK_TO_IGNORE_TABLE
  46. //#define TraceExprPrintLog(x, expr) PrintLog(x ": %s", expr->toString(StringBuffer()).str());
  47. #define TraceExprPrintLog(x, expr)
  48. //#define TraceTableFields
  49. inline bool needToSerializeRecord(node_operator mode)
  50. {
  51. return (mode == no_thor || mode == no_flat);
  52. }
  53. inline bool needToSerializeRecord(IHqlExpression * mode)
  54. {
  55. return needToSerializeRecord(mode->getOperator());
  56. }
  57. //---------------------------------------------------------------------------
  58. void HqlCppTranslator::addGlobalOnWarning(IHqlExpression * setMetaExpr)
  59. {
  60. globalOnWarnings->addOnWarning(setMetaExpr);
  61. }
  62. unsigned HqlCppTranslator::getSourceAggregateOptimizeFlags() const
  63. {
  64. return getOptimizeFlags()|HOOfold|HOOinsidecompound;
  65. }
  66. void HqlCppTranslator::doBuildExprFilepos(BuildCtx & ctx, IHqlExpression * expr, CHqlBoundExpr & tgt)
  67. {
  68. if (buildExprInCorrectContext(ctx, expr, tgt, false))
  69. return;
  70. throwError(HQLERR_CouldNotResolveFileposition); // internal error: fileposition should have been available.
  71. }
  72. void HqlCppTranslator::doBuildExprFileLogicalName(BuildCtx & ctx, IHqlExpression * expr, CHqlBoundExpr & tgt)
  73. {
  74. if (buildExprInCorrectContext(ctx, expr, tgt, false))
  75. return;
  76. throwError(HQLERR_CouldNotResolveFileposition); // internal error: fileposition should have been available.
  77. }
  78. //---------------------------------------------------------------------------
  79. node_operator getDatasetKind(IHqlExpression * dataset)
  80. {
  81. dataset = queryPhysicalRootTable(dataset);
  82. if (dataset)
  83. {
  84. IHqlExpression * mode = dataset->queryChild(2);
  85. if (mode)
  86. return mode->getOperator();
  87. }
  88. return no_none;
  89. }
  90. unsigned getProjectCount(IHqlExpression * expr)
  91. {
  92. unsigned projectCount = 0;
  93. while (expr->getOperator() != no_table)
  94. {
  95. switch (expr->getOperator())
  96. {
  97. case no_hqlproject:
  98. case no_usertable:
  99. case no_newusertable:
  100. case no_selectfields:
  101. projectCount++;
  102. break;
  103. }
  104. expr = expr->queryChild(0);
  105. }
  106. return projectCount;
  107. }
  108. IHqlExpression * queryFetch(IHqlExpression * expr)
  109. {
  110. loop
  111. {
  112. switch (expr->getOperator())
  113. {
  114. case no_fetch:
  115. return expr;
  116. case no_filter:
  117. case no_compound_fetch:
  118. case no_limit:
  119. case no_keyedlimit:
  120. case no_choosen:
  121. break;
  122. default:
  123. UNIMPLEMENTED;
  124. }
  125. expr = expr->queryChild(0);
  126. }
  127. }
  128. bool isSimpleSource(IHqlExpression * expr)
  129. {
  130. loop
  131. {
  132. switch (expr->getOperator())
  133. {
  134. case no_keyindex:
  135. case no_newkeyindex:
  136. case no_table:
  137. case no_temptable:
  138. case no_inlinetable:
  139. case no_workunit_dataset:
  140. case no_xmlproject:
  141. case no_null:
  142. case no_datasetfromrow:
  143. case no_datasetfromdictionary:
  144. case no_getgraphresult:
  145. case no_getgraphloopresult:
  146. case no_rows:
  147. return true;
  148. case no_choosen:
  149. case no_limit:
  150. case no_keyedlimit:
  151. case no_sorted:
  152. case no_stepped:
  153. case no_distributed:
  154. case no_preservemeta:
  155. case no_unordered:
  156. case no_grouped:
  157. case no_compound_diskread:
  158. case no_compound_disknormalize:
  159. case no_compound_diskaggregate:
  160. case no_compound_diskcount:
  161. case no_compound_diskgroupaggregate:
  162. case no_compound_indexread:
  163. case no_compound_indexnormalize:
  164. case no_compound_indexaggregate:
  165. case no_compound_indexcount:
  166. case no_compound_indexgroupaggregate:
  167. case no_compound_childread:
  168. case no_compound_childnormalize:
  169. case no_compound_childaggregate:
  170. case no_compound_childcount:
  171. case no_compound_childgroupaggregate:
  172. case no_compound_selectnew:
  173. case no_compound_inline:
  174. case no_section:
  175. case no_sectioninput:
  176. case no_nofold:
  177. case no_nohoist:
  178. case no_nocombine:
  179. case no_dataset_alias:
  180. break;
  181. default:
  182. return false;
  183. }
  184. expr = expr->queryChild(0);
  185. }
  186. }
  187. IHqlExpression * getVirtualSelector(IHqlExpression * dataset)
  188. {
  189. IHqlExpression * table = queryPhysicalRootTable(dataset);
  190. if (!table)
  191. table = dataset;
  192. return LINK(table->queryNormalizedSelector());
  193. }
  194. IHqlExpression * getFilepos(IHqlExpression * dataset, bool isLocal)
  195. {
  196. IHqlExpression * attr = isLocal ? createLocalAttribute() : NULL;
  197. return createValue(no_filepos, LINK(fposType), getVirtualSelector(dataset), attr);
  198. }
  199. IHqlExpression * getFileLogicalName(IHqlExpression * dataset)
  200. {
  201. return createValue(no_file_logicalname, makeVarStringType(UNKNOWN_LENGTH, NULL, NULL), getVirtualSelector(dataset));
  202. }
  203. IHqlExpression * getVirtualReplacement(IHqlExpression * field, IHqlExpression * virtualDef, IHqlExpression * dataset)
  204. {
  205. IAtom * virtualKind = virtualDef->queryName();
  206. if (virtualKind == filepositionAtom)
  207. return getFilepos(dataset, false);
  208. else if (virtualKind == localFilePositionAtom)
  209. return getFilepos(dataset, true);
  210. else if (virtualKind == sizeofAtom)
  211. return createValue(no_sizeof, LINK(sizetType), getVirtualSelector(dataset));
  212. else if (virtualKind == logicalFilenameAtom)
  213. return createValue(no_implicitcast, field->getType(), getFileLogicalName(dataset));
  214. throwError1(HQLERR_UnknownVirtualAttr, str(virtualKind));
  215. return NULL;
  216. }
  217. static IHqlExpression * getExplicitlyPromotedCompare(IHqlExpression * filter)
  218. {
  219. switch (filter->getOperator())
  220. {
  221. case no_in:
  222. case no_notin:
  223. return LINK(filter);
  224. }
  225. IHqlExpression * l = filter->queryChild(0);
  226. IHqlExpression * r = filter->queryChild(1);
  227. ITypeInfo * lType = queryUnqualifiedType(l->queryType());
  228. ITypeInfo * rType = queryUnqualifiedType(r->queryType());
  229. if (lType == rType)
  230. return LINK(filter);
  231. //Add explicit casts to the type. ensureExprType won't add a (string) to a string2 field.
  232. Owned<ITypeInfo> promotedType = getPromotedECLType(lType, rType);
  233. HqlExprArray args;
  234. if (lType == promotedType)
  235. args.append(*LINK(l));
  236. else
  237. args.append(*createValue(no_implicitcast, LINK(promotedType), LINK(l)));
  238. args.append(*ensureExprType(r, promotedType));
  239. return filter->clone(args);
  240. }
  241. static IHqlExpression * createFileposCall(HqlCppTranslator & translator, IIdAtom * name, const char * provider, const char * rowname)
  242. {
  243. HqlExprArray args;
  244. args.append(*createVariable(provider, makeBoolType()));
  245. args.append(*createVariable(rowname, makeBoolType())); // really a row
  246. return translator.bindFunctionCall(name, args);
  247. }
  248. //---------------------------------------------------------------------------
  249. void VirtualFieldsInfo::gatherVirtualFields(IHqlExpression * _record, bool ignoreVirtuals, bool ensureSerialized)
  250. {
  251. OwnedHqlExpr record = ensureSerialized ? getSerializedForm(_record, diskAtom) : LINK(_record);
  252. if (record != _record)
  253. requiresDeserialize = true;
  254. //MORE: This should really recurse through records to check for nested virtual fields.
  255. // e.g., inside ifblocks, or even records....
  256. ForEachChild(idx, record)
  257. {
  258. IHqlExpression * cur = record->queryChild(idx);
  259. IHqlExpression * virtualAttr = NULL;
  260. if (!ignoreVirtuals)
  261. virtualAttr = cur->queryAttribute(virtualAtom);
  262. if (virtualAttr)
  263. {
  264. selects.append(*LINK(cur));
  265. if (isUnknownSize(cur->queryType()))
  266. simpleVirtualsAtEnd = false;
  267. if (virtuals.find(*virtualAttr) == NotFound)
  268. virtuals.append(*LINK(virtualAttr));
  269. }
  270. else
  271. {
  272. //Also adds attributes...
  273. physicalFields.append(*LINK(cur));
  274. if (virtuals.ordinality())
  275. simpleVirtualsAtEnd = false;
  276. }
  277. }
  278. }
  279. IHqlExpression * VirtualFieldsInfo::createPhysicalRecord()
  280. {
  281. if (physicalFields.ordinality() == 1)
  282. if (physicalFields.item(0).getOperator() == no_record)
  283. return LINK(&physicalFields.item(0));
  284. return createRecord(physicalFields);
  285. }
  286. //---------------------------------------------------------------------------
  287. class VirtualRecordTransformCreator : public RecordTransformCreator
  288. {
  289. public:
  290. VirtualRecordTransformCreator(IHqlExpression * _dataset) { dataset = _dataset; }
  291. virtual IHqlExpression * getMissingAssignValue(IHqlExpression * expr)
  292. {
  293. IHqlExpression * virtualAttr = expr->queryAttribute(virtualAtom);
  294. assertex(virtualAttr);
  295. return getVirtualReplacement(expr, virtualAttr->queryChild(0), dataset);
  296. }
  297. protected:
  298. IHqlExpression * dataset;
  299. };
  300. IHqlExpression * createTableWithoutVirtuals(VirtualFieldsInfo & info, IHqlExpression * tableExpr)
  301. {
  302. IHqlExpression * record = tableExpr->queryChild(1);
  303. OwnedHqlExpr diskRecord = info.createPhysicalRecord();
  304. //Clone the annotations to improve the regenerated text in the graph
  305. OwnedHqlExpr diskRecordWithMeta = record->cloneAllAnnotations(diskRecord);
  306. HqlExprArray args;
  307. unwindChildren(args, tableExpr);
  308. args.replace(*LINK(diskRecordWithMeta), 1);
  309. IHqlExpression * newDataset = tableExpr->queryBody()->clone(args);
  310. VirtualRecordTransformCreator mapper(newDataset);
  311. IHqlExpression * newTransform = mapper.createMappingTransform(no_newtransform, record, newDataset);
  312. OwnedHqlExpr projected = createDatasetF(no_newusertable, newDataset, LINK(record), newTransform, createAttribute(_internal_Atom), NULL);
  313. return tableExpr->cloneAllAnnotations(projected);
  314. }
  315. IHqlExpression * buildTableWithoutVirtuals(VirtualFieldsInfo & info, IHqlExpression * expr)
  316. {
  317. IHqlExpression * tableExpr = queryPhysicalRootTable(expr);
  318. OwnedHqlExpr projected = createTableWithoutVirtuals(info, tableExpr);
  319. return replaceExpression(expr, tableExpr, projected);
  320. }
  321. static IHqlExpression * nextDiskField(IHqlExpression * diskRecord, unsigned & diskIndex)
  322. {
  323. loop
  324. {
  325. IHqlExpression * cur = diskRecord->queryChild(diskIndex++);
  326. if (!cur || !cur->isAttribute())
  327. return cur;
  328. }
  329. }
  330. static void createPhysicalLogicalAssigns(HqlExprArray & assigns, IHqlExpression * self, IHqlExpression * diskRecord, IHqlExpression * record, IHqlExpression * diskDataset, bool allowTranslate, unsigned fileposIndex)
  331. {
  332. unsigned numFields = record->numChildren();
  333. unsigned diskIndex = 0;
  334. for (unsigned idx2=0; idx2 < numFields; idx2++)
  335. {
  336. IHqlExpression * cur = record->queryChild(idx2);
  337. switch (cur->getOperator())
  338. {
  339. case no_ifblock:
  340. {
  341. IHqlExpression * ifblock = nextDiskField(diskRecord, diskIndex);
  342. assertex(ifblock && ifblock->getOperator() == no_ifblock);
  343. createPhysicalLogicalAssigns(assigns, self, ifblock->queryChild(1), cur->queryChild(1), diskDataset, false, NotFound);
  344. break;
  345. }
  346. case no_record:
  347. {
  348. IHqlExpression * srcRecord = nextDiskField(diskRecord, diskIndex);
  349. assertex(srcRecord && srcRecord->getOperator() == no_record);
  350. createPhysicalLogicalAssigns(assigns, self, srcRecord, cur, diskDataset, allowTranslate, NotFound);
  351. break;
  352. }
  353. case no_field:
  354. {
  355. OwnedHqlExpr target = createSelectExpr(LINK(self), LINK(cur));
  356. OwnedHqlExpr newValue;
  357. if (idx2 == fileposIndex)
  358. newValue.setown(getFilepos(diskDataset, false));
  359. else
  360. {
  361. IHqlExpression * curPhysical = nextDiskField(diskRecord, diskIndex);
  362. OwnedHqlExpr physicalSelect = createSelectExpr(LINK(diskDataset), LINK(curPhysical));
  363. if (cur->isDatarow() && !cur->hasAttribute(blobAtom) && (!isInPayload() || (physicalSelect->queryType() != target->queryType())))
  364. {
  365. HqlExprArray subassigns;
  366. OwnedHqlExpr childSelf = createSelector(no_self, cur, NULL);
  367. createPhysicalLogicalAssigns(subassigns, childSelf, curPhysical->queryRecord(), cur->queryRecord(), physicalSelect, false, NotFound);
  368. OwnedHqlExpr transform = createValue(no_transform, makeTransformType(cur->queryRecord()->getType()), subassigns);
  369. newValue.setown(createRow(no_createrow, transform.getClear()));
  370. }
  371. else
  372. newValue.setown(convertIndexPhysical2LogicalValue(cur, physicalSelect, allowTranslate));
  373. }
  374. if (newValue)
  375. assigns.append(*createAssign(target.getClear(), newValue.getClear()));
  376. break;
  377. }
  378. }
  379. }
  380. }
  381. static void createPhysicalLogicalAssigns(HqlExprArray & assigns, IHqlExpression * diskDataset, IHqlExpression * tableExpr, bool hasFilePosition)
  382. {
  383. IHqlExpression * record = tableExpr->queryRecord();
  384. unsigned fileposIndex = (hasFilePosition ? record->numChildren() - 1 : NotFound);
  385. OwnedHqlExpr self = getSelf(record);
  386. createPhysicalLogicalAssigns(assigns, self, diskDataset->queryRecord(), record, diskDataset, true, fileposIndex);
  387. }
  388. static IHqlExpression * mapIfBlock(HqlMapTransformer & mapper, IHqlExpression * cur);
  389. static IHqlExpression * mapIfBlockRecord(HqlMapTransformer & mapper, IHqlExpression * record)
  390. {
  391. HqlExprArray mapped;
  392. ForEachChild(i, record)
  393. {
  394. IHqlExpression * cur = record->queryChild(i);
  395. if (cur->getOperator() == no_ifblock)
  396. mapped.append(*mapIfBlock(mapper, cur));
  397. else
  398. mapped.append(*LINK(cur));
  399. }
  400. return record->clone(mapped);
  401. }
  402. static IHqlExpression * mapIfBlock(HqlMapTransformer & mapper, IHqlExpression * cur)
  403. {
  404. HqlExprArray args;
  405. unwindChildren(args, cur);
  406. args.replace(*mapper.transformRoot(&args.item(0)), 0);
  407. args.replace(*mapIfBlockRecord(mapper, &args.item(1)), 1);
  408. return cur->clone(args);
  409. }
  410. static IHqlExpression * createPhysicalIndexRecord(HqlMapTransformer & mapper, IHqlExpression * tableExpr, IHqlExpression * record, bool hasInternalFileposition, bool allowTranslate)
  411. {
  412. HqlExprArray physicalFields;
  413. unsigned max = record->numChildren() - (hasInternalFileposition ? 1 : 0);
  414. for (unsigned idx=0; idx < max; idx++)
  415. {
  416. IHqlExpression * cur = record->queryChild(idx);
  417. IHqlExpression * newField = NULL;
  418. if (cur->isAttribute())
  419. physicalFields.append(*LINK(cur));
  420. else if (cur->getOperator() == no_ifblock)
  421. physicalFields.append(*mapIfBlock(mapper, cur));
  422. else if (cur->getOperator() == no_record)
  423. physicalFields.append(*createPhysicalIndexRecord(mapper, tableExpr, cur, false, allowTranslate));
  424. else if (cur->hasAttribute(blobAtom))
  425. {
  426. newField = createField(cur->queryId(), makeIntType(8, false), NULL, NULL);
  427. }
  428. else
  429. {
  430. OwnedHqlExpr select = createSelectExpr(LINK(tableExpr), LINK(cur));
  431. if (!allowTranslate)
  432. newField = LINK(cur);
  433. else if (cur->isDatarow() && !isInPayload())
  434. {
  435. //MORE: Mappings for ifblocks using self.a.b (!)
  436. HqlMapTransformer childMapper;
  437. OwnedHqlExpr newRecord = createPhysicalIndexRecord(childMapper, select, cur->queryRecord(), false, allowTranslate);
  438. HqlExprArray args;
  439. unwindChildren(args, cur);
  440. newField = createField(cur->queryId(), newRecord->getType(), args);
  441. }
  442. else
  443. {
  444. //This should support other non serialized formats. E.g., link counted strings.
  445. //Simplest would be to move getSerializedForm code + call that first.
  446. if (cur->hasAttribute(_linkCounted_Atom) || cur->isDatarow())
  447. {
  448. newField = getSerializedForm(cur, diskAtom);
  449. assertex(newField != cur || cur->isDatarow());
  450. }
  451. else
  452. {
  453. OwnedHqlExpr hozed = getHozedKeyValue(select);
  454. if (hozed->queryType() == select->queryType())
  455. newField = LINK(cur);
  456. else
  457. newField = createField(cur->queryId(), hozed->getType(), extractFieldAttrs(cur));
  458. }
  459. }
  460. }
  461. if (newField)
  462. {
  463. physicalFields.append(*newField);
  464. if (cur != newField)
  465. {
  466. IHqlExpression * self = querySelfReference();
  467. OwnedHqlExpr select = createSelectExpr(LINK(self), LINK(cur));
  468. OwnedHqlExpr physicalSelect = createSelectExpr(LINK(self), LINK(newField));
  469. OwnedHqlExpr newValue = convertIndexPhysical2LogicalValue(cur, physicalSelect, true);
  470. mapper.setMapping(select, newValue);
  471. }
  472. }
  473. }
  474. return createRecord(physicalFields);
  475. }
  476. IHqlExpression * HqlCppTranslator::convertToPhysicalIndex(IHqlExpression * tableExpr)
  477. {
  478. LinkedHqlExpr * match = physicalIndexCache.getValue(tableExpr);
  479. if (match)
  480. return LINK(*match);
  481. if (tableExpr->hasAttribute(_original_Atom))
  482. return LINK(tableExpr);
  483. assertex(tableExpr->getOperator() == no_newkeyindex);
  484. IHqlExpression * record = tableExpr->queryRecord();
  485. HqlMapTransformer mapper;
  486. bool hasFileposition = getBoolAttribute(tableExpr, filepositionAtom, true);
  487. IHqlExpression * diskRecord = createPhysicalIndexRecord(mapper, tableExpr, record, hasFileposition, true);
  488. unsigned payload = numPayloadFields(tableExpr);
  489. assertex(payload || !hasFileposition);
  490. unsigned newPayload = hasFileposition ? payload-1 : payload;
  491. HqlExprArray args;
  492. unwindChildren(args, tableExpr);
  493. args.replace(*diskRecord, 1);
  494. removeAttribute(args, _payload_Atom);
  495. args.append(*createAttribute(_payload_Atom, getSizetConstant(newPayload)));
  496. args.append(*createAttribute(_original_Atom, LINK(tableExpr)));
  497. //remove the preload attribute and replace with correct value
  498. IHqlExpression * newDataset = createDataset(tableExpr->getOperator(), args);
  499. HqlExprArray assigns;
  500. createPhysicalLogicalAssigns(assigns, newDataset, tableExpr, hasFileposition);
  501. OwnedHqlExpr projectedTable = createDataset(no_newusertable, newDataset, createComma(LINK(record), createValue(no_newtransform, makeTransformType(record->getType()), assigns)));
  502. physicalIndexCache.setValue(tableExpr, projectedTable);
  503. return projectedTable.getClear();
  504. }
  505. IHqlExpression * convertToPhysicalTable(IHqlExpression * tableExpr, bool ensureSerialized)
  506. {
  507. VirtualFieldsInfo fieldInfo;
  508. fieldInfo.gatherVirtualFields(tableExpr->queryRecord(), tableExpr->hasAttribute(_noVirtual_Atom), ensureSerialized);
  509. if (fieldInfo.hasVirtualsOrDeserialize())
  510. return createTableWithoutVirtuals(fieldInfo, tableExpr);
  511. return LINK(tableExpr);
  512. }
  513. IHqlExpression * HqlCppTranslator::buildIndexFromPhysical(IHqlExpression * expr)
  514. {
  515. IHqlExpression * tableExpr = queryPhysicalRootTable(expr);
  516. OwnedHqlExpr newProject = convertToPhysicalIndex(tableExpr);
  517. return replaceExpression(expr, tableExpr, newProject);
  518. }
  519. //---------------------------------------------------------------------------
  520. class SourceSteppingInfo
  521. {
  522. public:
  523. inline bool exists() { return rawStepping.exists(); }
  524. IHqlExpression * firstStepped()
  525. {
  526. if (rawStepping.exists())
  527. return rawStepping.fields->queryChild(0);
  528. return NULL;
  529. }
  530. void extractRaw()
  531. {
  532. rawSteppingProject.extractFields(rawStepping);
  533. }
  534. void checkKeyable(MonitorExtractor & monitors)
  535. {
  536. if (!rawStepping.exists())
  537. return;
  538. unsigned prev = NotFound;
  539. ForEachChild(i, rawStepping.fields)
  540. {
  541. IHqlExpression * cur = rawStepping.fields->queryChild(i);
  542. unsigned thisMatch = monitors.queryKeySelectIndex(cur);
  543. if (thisMatch == NotFound)
  544. throwError1(HQLERR_StepFieldNotKeyed, str(cur->queryChild(1)->queryName()));
  545. if ((prev != NotFound) && (thisMatch != prev+1))
  546. throwError1(HQLERR_StepFieldNotContiguous, str(cur->queryChild(1)->queryName()));
  547. prev = thisMatch;
  548. }
  549. }
  550. void generateMembers(HqlCppTranslator & translator, BuildCtx & ctx)
  551. {
  552. rawStepping.generateSteppingMetaMember(translator, ctx, "RawSteppingMeta");
  553. if (outputStepping.exists())
  554. {
  555. if (outputStepping.exists() && outputStepping.ds != rawStepping.ds)
  556. {
  557. outputStepping.generateSteppingMetaMember(translator, ctx, "ProjectedSteppingMeta");
  558. BuildCtx transformctx(ctx);
  559. transformctx.addQuotedCompoundLiteral("virtual void mapOutputToInput(ARowBuilder & crSelf, const void * _projected, unsigned numFields)");
  560. translator.ensureRowAllocated(transformctx, "crSelf");
  561. transformctx.addQuotedLiteral("const byte * pr = (const byte *)_projected;");
  562. translator.bindTableCursor(transformctx, rawStepping.ds, "crSelf.row()");
  563. translator.bindTableCursor(transformctx, outputStepping.ds, "pr");
  564. StringBuffer s;
  565. ForEachChild(i, outputStepping.fields)
  566. {
  567. IHqlExpression * curOutput = outputStepping.fields->queryChild(i);
  568. IHqlExpression * curRawExpr = rawSteppingProject.fields->queryChild(i);
  569. IHqlExpression * curRawSelect = rawStepping.fields->queryChild(i);
  570. OwnedHqlExpr original = outputStepping.invertTransform(curRawExpr, curOutput);
  571. transformctx.addQuoted(s.clear().append("if (numFields < ").append(i+1).append(") return;"));
  572. translator.buildAssign(transformctx, curRawSelect, original);
  573. }
  574. }
  575. }
  576. else
  577. {
  578. OwnedHqlExpr fail = createValue(no_fail, makeVoidType(), createConstant("Cannot step output of index read"));
  579. BuildCtx transformctx(ctx);
  580. transformctx.addQuotedCompoundLiteral("virtual void mapOutputToInput(void * _original, const void * _projected, unsigned numFields)");
  581. translator.buildStmt(transformctx, fail);
  582. }
  583. }
  584. public:
  585. SteppingFieldSelection outputStepping;
  586. SteppingFieldSelection rawSteppingProject;
  587. SteppingFieldSelection rawStepping;
  588. };
  589. //---------------------------------------------------------------------------
  590. class SourceBuilder
  591. {
  592. public:
  593. SourceBuilder(HqlCppTranslator & _translator, IHqlExpression *_tableExpr, IHqlExpression *_nameExpr)
  594. : tableExpr(_tableExpr), translator(_translator)
  595. {
  596. nameExpr.setown(foldHqlExpression(_nameExpr));
  597. needDefaultTransform = true;
  598. needToCallTransform = false;
  599. isPreloaded = false;
  600. isCompoundCount = false;
  601. transformCanFilter = false;
  602. IHqlExpression * preload = tableExpr ? tableExpr->queryAttribute(preloadAtom) : NULL;
  603. if (preload)
  604. {
  605. isPreloaded = true;
  606. preloadSize.set(preload->queryChild(0));
  607. }
  608. failedFilterValue.set(queryZero());
  609. isNormalize = false;
  610. aggregation = false;
  611. instance = NULL;
  612. returnIfFilterFails = true;
  613. useFilterMappings = true;
  614. generateUnfilteredTransform = false;
  615. allowDynamicFormatChange = tableExpr && !tableExpr->hasAttribute(fixedAtom);
  616. onlyExistsAggreate = false;
  617. monitorsForGrouping = false;
  618. useImplementationClass = false;
  619. isUnfilteredCount = false;
  620. isVirtualLogicalFilenameUsed = false;
  621. requiresOrderedMerge = false;
  622. rootSelfRow = NULL;
  623. activityKind = TAKnone;
  624. }
  625. virtual ~SourceBuilder() {}
  626. virtual void buildMembers(IHqlExpression * expr) = 0;
  627. virtual void buildTransformFpos(BuildCtx & transformCtx) = 0;
  628. virtual void extractMonitors(IHqlExpression * ds, SharedHqlExpr & unkeyedFilter, HqlExprArray & conds);
  629. virtual void buildTransformElements(BuildCtx & ctx, IHqlExpression * expr, bool ignoreFilters);
  630. virtual void buildTransform(IHqlExpression * expr) = 0;
  631. virtual void analyse(IHqlExpression * expr);
  632. void buildFilenameMember();
  633. void appendFilter(SharedHqlExpr & unkeyedFilter, IHqlExpression * expr);
  634. void buildKeyedLimitHelper(IHqlExpression * expr);
  635. void buildLimits(BuildCtx & classctx, IHqlExpression * expr, unique_id_t id);
  636. void buildReadMembers( IHqlExpression * expr);
  637. void buildSteppingMeta(IHqlExpression * expr, MonitorExtractor * monitors);
  638. void buildTransformBody(BuildCtx & transformCtx, IHqlExpression * expr, bool returnSize, bool ignoreFilters, bool bindInputRow);
  639. void checkDependencies(BuildCtx & ctx, IHqlExpression * expr);
  640. bool containsStepping(IHqlExpression * expr);
  641. ABoundActivity * buildActivity(BuildCtx & ctx, IHqlExpression * expr, ThorActivityKind activityKind, const char *kind, ABoundActivity *input);
  642. void gatherVirtualFields(bool ignoreVirtuals, bool ensureSerialized);
  643. bool recordHasVirtuals() { return fieldInfo.hasVirtuals(); }
  644. bool recordHasVirtualsOrDeserialize() { return fieldInfo.hasVirtualsOrDeserialize(); }
  645. bool isSourceInvariant(IHqlExpression * dataset, IHqlExpression * expr);
  646. bool hasExistChoosenLimit() { return (choosenValue && getIntValue(choosenValue) == 1); }
  647. protected:
  648. void assignLocalExtract(BuildCtx & ctx, ParentExtract * extractBuilder, IHqlExpression * dataset, const char * argName);
  649. void associateFilePositions(BuildCtx & ctx, const char * provider, const char * rowname);
  650. void buildSteppedHelpers();
  651. void doBuildAggregateSelectIterator(BuildCtx & ctx, IHqlExpression * expr);
  652. void doBuildNormalizeIterators(BuildCtx & ctx, IHqlExpression * expr, bool isChildIterator);
  653. void buildAggregateHelpers(IHqlExpression * expr);
  654. void buildCountHelpers(IHqlExpression * expr, bool allowMultiple);
  655. virtual void buildFlagsMember(IHqlExpression * expr) {}
  656. void buildGlobalGroupAggregateHelpers(IHqlExpression * expr);
  657. void buildGroupAggregateHelpers(ParentExtract * extractBuilder, IHqlExpression * aggregate);
  658. void buildGroupAggregateCompareHelper(ParentExtract * extractBuilder, IHqlExpression * aggregate, HqlExprArray & recordFields, HqlExprArray & agrgegateFields);
  659. void buildGroupAggregateHashHelper(ParentExtract * extractBuilder, IHqlExpression * dataset, IHqlExpression * fields);
  660. void buildGroupAggregateProcessHelper(ParentExtract * extractBuilder, IHqlExpression * aggregate, const char * name, bool doneAny);
  661. void buildGroupingMonitors(IHqlExpression * expr, MonitorExtractor & monitors);
  662. void buildGroupAggregateTransformBody(BuildCtx & transformctx, IHqlExpression * expr, bool useExtract, bool bindInputRow);
  663. void buildNormalizeHelpers(IHqlExpression * expr);
  664. void buildTargetCursor(Shared<BoundRow> & tempRow, Shared<BoundRow> & rowBuilder, BuildCtx & ctx, IHqlExpression * expr);
  665. void associateTargetCursor(BuildCtx & subctx, BuildCtx & ctx, BoundRow * tempRow, BoundRow * rowBuilder, IHqlExpression * expr);
  666. IHqlExpression * ensureAggregateGroupingAliased(IHqlExpression * aggregate);
  667. void gatherSteppingMeta(IHqlExpression * expr, SourceSteppingInfo & info);
  668. void gatherSteppingMeta(IHqlExpression * expr, SteppingFieldSelection & outputStepping, SteppingFieldSelection & rawStepping);
  669. void gatherFieldUsage(SourceFieldUsage * fieldUsage, IHqlExpression * expr);
  670. void rebindFilepositons(BuildCtx & ctx, IHqlExpression * dataset, node_operator side, IHqlExpression * selSeq, bool isLocal);
  671. void rebindFilepositons(BuildCtx & ctx, IHqlExpression * dataset, node_operator side, IHqlExpression * selSeq);
  672. virtual void processTransformSelect(BuildCtx & ctx, IHqlExpression * expr);
  673. virtual void analyseGraph(IHqlExpression * expr);
  674. virtual bool isExplicitExists() { return false; }
  675. public:
  676. VirtualFieldsInfo fieldInfo;
  677. ActivityInstance * instance;
  678. BoundRow * rootSelfRow;
  679. HqlExprAttr tableExpr;
  680. HqlExprAttr nameExpr;
  681. HqlExprAttr fpos;
  682. HqlExprAttr lfpos;
  683. HqlExprAttr limitExpr;
  684. HqlExprAttr keyedLimitExpr;
  685. HqlExprAttr choosenValue;
  686. HqlExprAttr preloadSize;
  687. HqlExprAttr lastTransformer;
  688. HqlExprAttr alreadyDoneFlag;
  689. HqlExprArray originalFilters;
  690. HqlExprArray mappedFilters;
  691. HqlExprArray removedFilters;
  692. HqlExprAttr failedFilterValue;
  693. HqlExprAttr compoundCountVar;
  694. HqlExprAttr physicalRecord;
  695. HqlExprAttr steppedExpr;
  696. Linked<BuildCtx> globaliterctx;
  697. HqlExprCopyArray parentCursors;
  698. HqlExprAttr logicalFilenameMarker;
  699. ThorActivityKind activityKind;
  700. bool useFilterMappings;
  701. bool needDefaultTransform;
  702. bool needToCallTransform;
  703. bool isPreloaded;
  704. bool transformCanFilter;
  705. bool isCompoundCount;
  706. bool isNormalize;
  707. bool aggregation;
  708. bool returnIfFilterFails;
  709. bool allowDynamicFormatChange;
  710. bool onlyExistsAggreate;
  711. bool monitorsForGrouping;
  712. bool generateUnfilteredTransform;
  713. bool useImplementationClass;
  714. bool isUnfilteredCount;
  715. bool isVirtualLogicalFilenameUsed;
  716. bool requiresOrderedMerge;
  717. protected:
  718. HqlCppTranslator & translator;
  719. };
  720. struct HQLCPP_API HqlFilePositionDefinedValue : public HqlSimpleDefinedValue
  721. {
  722. public:
  723. HqlFilePositionDefinedValue(SourceBuilder & _builder, IHqlExpression * _original, IHqlExpression * _expr)
  724. : HqlSimpleDefinedValue(_original, _expr), builder(_builder)
  725. { }
  726. virtual IHqlExpression * queryExpr() const
  727. {
  728. builder.isVirtualLogicalFilenameUsed = true;
  729. return HqlSimpleDefinedValue::queryExpr();
  730. }
  731. public:
  732. SourceBuilder & builder;
  733. };
  734. bool SourceBuilder::isSourceInvariant(IHqlExpression * dataset, IHqlExpression * expr)
  735. {
  736. if (containsAssertKeyed(expr))
  737. return false;
  738. if (!containsActiveDataset(expr))
  739. return true;
  740. HqlExprCopyArray inScope;
  741. expr->gatherTablesUsed(NULL, &inScope);
  742. if (!canEvaluateInScope(parentCursors, inScope))
  743. return false;
  744. //Carefull.... It looks ok, but it is possible that the same dataset occurs at multiple levels (sqfilt.hql)
  745. //So need to be careful that the datasets being referenced aren't newly in scope for this activity...
  746. loop
  747. {
  748. if (inScope.contains(*dataset->queryNormalizedSelector()))
  749. return false;
  750. //To be strictly correct we need to walk no_select chain of root datasets!
  751. dataset = queryNextMultiLevelDataset(dataset, true);
  752. if (!dataset)
  753. return true;
  754. }
  755. }
  756. void SourceBuilder::analyse(IHqlExpression * expr)
  757. {
  758. IHqlExpression * body = expr->queryBody(true);
  759. if (expr != body)
  760. {
  761. switch (expr->getAnnotationKind())
  762. {
  763. case annotate_meta:
  764. //the onwarning state will be restored by the scope held in HqlCppTranslator::buildActivity
  765. translator.localOnWarnings->processMetaAnnotation(expr);
  766. break;
  767. case annotate_location:
  768. instance->addLocationAttribute(expr);
  769. break;
  770. case annotate_symbol:
  771. //don't clear onWarnings when we hit a symbol because the warnings within a compound activity aren't generated at the correct point
  772. instance->addNameAttribute(expr);
  773. break;
  774. }
  775. analyse(body);
  776. return;
  777. }
  778. node_operator op = expr->getOperator();
  779. switch (op)
  780. {
  781. case no_cachealias:
  782. analyse(expr->queryChild(1));
  783. return;
  784. case no_newkeyindex:
  785. case no_table:
  786. case no_fetch:
  787. case no_select: // handled below
  788. case no_null:
  789. case no_anon:
  790. case no_pseudods:
  791. case no_workunit_dataset:
  792. case no_call:
  793. case no_externalcall:
  794. case no_rows:
  795. case no_libraryinput:
  796. break;
  797. default:
  798. analyse(expr->queryChild(0));
  799. break;
  800. }
  801. switch (op)
  802. {
  803. case no_table:
  804. case no_newkeyindex:
  805. if (fieldInfo.hasVirtuals())
  806. {
  807. assertex(fieldInfo.simpleVirtualsAtEnd);
  808. needToCallTransform = true;
  809. needDefaultTransform = false;
  810. }
  811. break;
  812. case no_null:
  813. case no_anon:
  814. case no_pseudods:
  815. case no_workunit_dataset:
  816. case no_getgraphresult:
  817. case no_call:
  818. case no_externalcall:
  819. case no_rows:
  820. case no_libraryinput:
  821. break;
  822. case no_filter:
  823. {
  824. OwnedHqlExpr unkeyedFilter;
  825. HqlExprArray conds;
  826. unwindFilterConditions(conds, expr);
  827. extractMonitors(expr, unkeyedFilter, conds);
  828. if (unkeyedFilter)
  829. {
  830. transformCanFilter = true;
  831. originalFilters.append(*LINK(expr));
  832. mappedFilters.append(*unkeyedFilter.getClear());
  833. }
  834. else
  835. removedFilters.append(*LINK(expr));
  836. break;
  837. }
  838. case no_select:
  839. {
  840. bool isNew;
  841. IHqlExpression * ds = querySelectorDataset(expr, isNew);
  842. if (isNew && isMultiLevelDatasetSelector(expr, false))
  843. {
  844. if (!translator.resolveSelectorDataset(instance->startctx, ds))
  845. {
  846. analyse(ds);
  847. isNormalize = true;
  848. }
  849. }
  850. break;
  851. }
  852. case no_stepped:
  853. if (steppedExpr)
  854. throwError(HQLERR_MultipleStepped);
  855. steppedExpr.set(expr);
  856. break;
  857. case no_sorted:
  858. case no_distributed:
  859. case no_preservemeta:
  860. case no_unordered:
  861. case no_grouped:
  862. case no_alias_scope:
  863. case no_section:
  864. case no_sectioninput:
  865. case no_nofold:
  866. case no_nohoist:
  867. case no_nocombine:
  868. case no_dataset_alias:
  869. break;
  870. case no_preload:
  871. isPreloaded = true;
  872. preloadSize.set(queryRealChild(expr, 1));
  873. break;
  874. case no_limit:
  875. limitExpr.set(expr);
  876. break;
  877. case no_keyedlimit:
  878. keyedLimitExpr.set(expr);
  879. break;
  880. case no_choosen:
  881. {
  882. choosenValue.set(expr->queryChild(1));
  883. IHqlExpression * first = queryRealChild(expr, 2);
  884. if (first)
  885. {
  886. Owned<ITypeInfo> type = makeIntType(8, true);
  887. choosenValue.setown(createValue(no_sub, LINK(type), createValue(no_add, LINK(type), ensureExprType(choosenValue, type), ensureExprType(first, type)), createConstant(I64C(1))));
  888. }
  889. choosenValue.setown(foldHqlExpression(choosenValue));
  890. }
  891. break;
  892. case no_hqlproject:
  893. needToCallTransform = true;
  894. needDefaultTransform = false;
  895. lastTransformer.set(expr);
  896. break;
  897. case no_newusertable:
  898. needToCallTransform = true;
  899. needDefaultTransform = false;
  900. lastTransformer.set(expr);
  901. break;
  902. case no_aggregate:
  903. {
  904. needToCallTransform = true;
  905. needDefaultTransform = false;
  906. aggregation = true;
  907. lastTransformer.set(expr);
  908. break;
  909. }
  910. case no_newaggregate:
  911. {
  912. needToCallTransform = true;
  913. needDefaultTransform = false;
  914. aggregation = true;
  915. lastTransformer.set(expr);
  916. IHqlExpression * transform = expr->queryChild(2);
  917. node_operator aggOp = queryTransformSingleAggregate(transform);
  918. onlyExistsAggreate = ((aggOp == no_existsgroup) || (aggOp == no_none)); // The implicit project code can remove all the aggregate() operators....
  919. if (isCompoundCount)
  920. {
  921. IHqlExpression * rhs = transform->queryChild(0)->queryChild(1);
  922. IHqlExpression * filter = queryRealChild(rhs, 0);
  923. if (filter)
  924. transformCanFilter = true;
  925. }
  926. break;
  927. }
  928. case no_usertable:
  929. case no_selectfields:
  930. UNIMPLEMENTED;
  931. break;
  932. case no_fetch:
  933. needToCallTransform = true;
  934. needDefaultTransform = false;
  935. lastTransformer.set(expr);
  936. break;
  937. case no_compound_diskread:
  938. case no_compound_disknormalize:
  939. case no_compound_diskaggregate:
  940. case no_compound_diskcount:
  941. case no_compound_diskgroupaggregate:
  942. case no_compound_indexread:
  943. case no_compound_indexnormalize:
  944. case no_compound_indexaggregate:
  945. case no_compound_indexcount:
  946. case no_compound_indexgroupaggregate:
  947. case no_compound_childread:
  948. case no_compound_childnormalize:
  949. case no_compound_childaggregate:
  950. case no_compound_childcount:
  951. case no_compound_childgroupaggregate:
  952. case no_compound_fetch:
  953. case no_compound_selectnew:
  954. break;
  955. default:
  956. throwUnexpectedOp(op);
  957. }
  958. }
  959. void SourceBuilder::appendFilter(SharedHqlExpr & unkeyedFilter, IHqlExpression * expr)
  960. {
  961. if (expr)
  962. {
  963. if (expr->queryValue())
  964. {
  965. if (!expr->queryValue()->getBoolValue())
  966. unkeyedFilter.set(expr);
  967. }
  968. else
  969. {
  970. if (unkeyedFilter)
  971. unkeyedFilter.setown(createValue(no_and, unkeyedFilter.getClear(), LINK(expr)));
  972. else
  973. unkeyedFilter.set(expr);
  974. }
  975. }
  976. }
  977. void SourceBuilder::associateFilePositions(BuildCtx & ctx, const char * provider, const char * rowname)
  978. {
  979. if (fpos)
  980. {
  981. Owned<IHqlExpression> fposExpr = createFileposCall(translator, getFilePositionId, provider, rowname);
  982. ctx.associateExpr(fpos, fposExpr);
  983. }
  984. if (lfpos)
  985. {
  986. Owned<IHqlExpression> fposExpr = createFileposCall(translator, getLocalFilePositionId, provider, rowname);
  987. ctx.associateExpr(lfpos, fposExpr);
  988. }
  989. if (logicalFilenameMarker)
  990. {
  991. Owned<IHqlExpression> nameExpr = createFileposCall(translator, queryLogicalFilenameId, provider, rowname);
  992. ctx.associateOwn(*new HqlFilePositionDefinedValue(*this, logicalFilenameMarker, nameExpr));
  993. }
  994. }
  995. void SourceBuilder::rebindFilepositons(BuildCtx & ctx, IHqlExpression * dataset, node_operator side, IHqlExpression * selSeq, bool isLocal)
  996. {
  997. if (!tableExpr)
  998. return;
  999. OwnedHqlExpr searchPos = getFilepos(tableExpr, isLocal);
  1000. HqlExprAssociation * match = ctx.queryMatchExpr(searchPos);
  1001. if (match)
  1002. {
  1003. OwnedHqlExpr selector = createSelector(side, dataset, selSeq);
  1004. OwnedHqlExpr selectorFpos = getFilepos(selector, isLocal);
  1005. ctx.associateExpr(selectorFpos, match->queryExpr());
  1006. }
  1007. }
  1008. void SourceBuilder::rebindFilepositons(BuildCtx & ctx, IHqlExpression * dataset, node_operator side, IHqlExpression * selSeq)
  1009. {
  1010. bool savedIsVirtualLogicalFilenameUsed = isVirtualLogicalFilenameUsed; // don't allow the rebinding to set the flag.
  1011. rebindFilepositons(ctx, dataset, side, selSeq, true);
  1012. rebindFilepositons(ctx, dataset, side, selSeq, false);
  1013. OwnedHqlExpr searchLogicalFilename = getFileLogicalName(dataset);
  1014. HqlExprAssociation * match = ctx.queryMatchExpr(searchLogicalFilename);
  1015. if (match)
  1016. {
  1017. OwnedHqlExpr selector = createSelector(side, dataset, selSeq);
  1018. OwnedHqlExpr selectorLogicalFilename = getFileLogicalName(dataset);
  1019. ctx.associateOwn(*new HqlFilePositionDefinedValue(*this, selectorLogicalFilename, match->queryExpr()));
  1020. }
  1021. isVirtualLogicalFilenameUsed = savedIsVirtualLogicalFilenameUsed;
  1022. }
  1023. void SourceBuilder::buildFilenameMember()
  1024. {
  1025. //---- virtual const char * getFileName() { return "x.d00"; } ----
  1026. translator.buildFilenameFunction(*instance, instance->startctx, "getFileName", nameExpr, translator.hasDynamicFilename(tableExpr));
  1027. }
  1028. void SourceBuilder::buildReadMembers(IHqlExpression * expr)
  1029. {
  1030. buildFilenameMember();
  1031. //---- virtual bool needTransform() { return <bool>; } ----
  1032. if (needToCallTransform || transformCanFilter)
  1033. translator.doBuildBoolFunction(instance->classctx, "needTransform", true);
  1034. //---- virtual bool needTransform() { return <bool>; } ----
  1035. if (transformCanFilter)
  1036. translator.doBuildBoolFunction(instance->classctx, "transformMayFilter", true);
  1037. }
  1038. void SourceBuilder::buildLimits(BuildCtx & classctx, IHqlExpression * expr, unique_id_t id)
  1039. {
  1040. if (limitExpr)
  1041. translator.buildLimitHelpers(classctx, limitExpr, nameExpr, id);
  1042. if (choosenValue)
  1043. {
  1044. BuildCtx funcctx(classctx);
  1045. funcctx.addQuotedCompoundLiteral("virtual unsigned __int64 getChooseNLimit()");
  1046. OwnedHqlExpr newLimit = ensurePositiveOrZeroInt64(choosenValue);
  1047. translator.buildReturn(funcctx, newLimit);
  1048. }
  1049. }
  1050. void SourceBuilder::buildTransformBody(BuildCtx & transformCtx, IHqlExpression * expr, bool returnSize, bool ignoreFilters, bool bindInputRow)
  1051. {
  1052. if (tableExpr && bindInputRow)
  1053. {
  1054. IHqlExpression * mode = (tableExpr->getOperator() == no_table) ? tableExpr->queryChild(2) : NULL;
  1055. if (mode && mode->getOperator() == no_csv)
  1056. {
  1057. translator.bindCsvTableCursor(transformCtx, tableExpr, "Src", no_none, NULL, true, queryCsvEncoding(mode));
  1058. }
  1059. else
  1060. {
  1061. //NOTE: The source is not link counted - it comes from a prefetched row, and does not include any virtual file position field.
  1062. OwnedHqlExpr boundSrc = createVariable("left", makeRowReferenceType(physicalRecord));
  1063. transformCtx.associateOwn(*new BoundRow(tableExpr->queryNormalizedSelector(), boundSrc, translator.queryRecordOffsetMap(physicalRecord), no_none, NULL));
  1064. }
  1065. }
  1066. rootSelfRow = translator.bindSelf(transformCtx, expr, "crSelf");
  1067. buildTransformFpos(transformCtx);
  1068. buildTransformElements(transformCtx, expr, ignoreFilters);
  1069. if (returnSize)
  1070. {
  1071. CHqlBoundExpr boundTargetSize;
  1072. if (needDefaultTransform)
  1073. {
  1074. IHqlExpression * left = expr->queryNormalizedSelector();
  1075. OwnedHqlExpr source = ensureActiveRow(left);
  1076. translator.buildAssign(transformCtx, rootSelfRow->querySelector(), source);
  1077. }
  1078. translator.getRecordSize(transformCtx, rootSelfRow->querySelector(), boundTargetSize);
  1079. transformCtx.setNextDestructor();
  1080. transformCtx.addReturn(boundTargetSize.expr);
  1081. }
  1082. rootSelfRow = NULL;
  1083. }
  1084. void SourceBuilder::buildTargetCursor(Shared<BoundRow> & tempRow, Shared<BoundRow> & rowBuilder, BuildCtx & ctx, IHqlExpression * expr)
  1085. {
  1086. assertex(lastTransformer != NULL);
  1087. if (expr == lastTransformer)
  1088. {
  1089. rowBuilder.set(rootSelfRow);
  1090. }
  1091. else
  1092. {
  1093. tempRow.setown(translator.declareTempAnonRow(ctx, ctx, expr));
  1094. ctx.addGroup();
  1095. // group is important, otherwise sizeof(self.x) gets cached incorrectly
  1096. // not so sure, but references to LEFT(x) may be misresolved
  1097. rowBuilder.setown(translator.createRowBuilder(ctx, tempRow));
  1098. }
  1099. }
  1100. void SourceBuilder::associateTargetCursor(BuildCtx & subctx, BuildCtx & ctx, BoundRow * tempRow, BoundRow * rowBuilder, IHqlExpression * expr)
  1101. {
  1102. //First remove the old active dataset
  1103. //This is not strictly necessary, but it avoids the redundant row being serialized to any child queries
  1104. BoundRow * oldCursor = translator.resolveSelectorDataset(ctx, expr->queryChild(0));
  1105. ctx.removeAssociation(oldCursor);
  1106. //And add an association for expr
  1107. if (tempRow)
  1108. {
  1109. translator.finalizeTempRow(subctx, tempRow, rowBuilder);
  1110. translator.bindTableCursor(ctx, expr, tempRow->queryBound());
  1111. }
  1112. else
  1113. translator.bindTableCursor(ctx, expr, rowBuilder->queryBound());
  1114. }
  1115. void SourceBuilder::buildTransformElements(BuildCtx & ctx, IHqlExpression * expr, bool ignoreFilters)
  1116. {
  1117. //This function can be called again for the unfiltered tranform. Don't process annotations again.
  1118. if ((expr != instance->dataset) && !ignoreFilters)
  1119. instance->processAnnotations(expr);
  1120. expr = expr->queryBody();
  1121. node_operator op = expr->getOperator();
  1122. switch (op)
  1123. {
  1124. case no_cachealias:
  1125. buildTransformElements(ctx, expr->queryChild(1), ignoreFilters);
  1126. return;
  1127. case no_newkeyindex:
  1128. case no_table:
  1129. case no_fetch:
  1130. case no_select: // handled below
  1131. case no_null:
  1132. case no_anon:
  1133. case no_pseudods:
  1134. case no_workunit_dataset:
  1135. case no_getgraphresult:
  1136. case no_call:
  1137. case no_externalcall:
  1138. case no_rows:
  1139. case no_libraryinput:
  1140. break;
  1141. default:
  1142. buildTransformElements(ctx, expr->queryChild(0), ignoreFilters);
  1143. break;
  1144. }
  1145. switch (op)
  1146. {
  1147. case no_newkeyindex:
  1148. case no_table:
  1149. if (fieldInfo.hasVirtuals())
  1150. {
  1151. IHqlExpression * record = expr->queryRecord();
  1152. assertex(fieldInfo.simpleVirtualsAtEnd);
  1153. assertex(!recordRequiresSerialization(record, diskAtom));
  1154. CHqlBoundExpr bound;
  1155. StringBuffer s;
  1156. translator.getRecordSize(ctx, expr, bound);
  1157. size32_t virtualSize = 0;
  1158. ForEachChild(idx, record)
  1159. {
  1160. IHqlExpression * field = record->queryChild(idx);
  1161. IHqlExpression * virtualAttr = field->queryAttribute(virtualAtom);
  1162. if (virtualAttr)
  1163. {
  1164. size32_t fieldSize = field->queryType()->getSize();
  1165. assertex(fieldSize != UNKNOWN_LENGTH);
  1166. virtualSize += fieldSize;
  1167. }
  1168. }
  1169. if (!isFixedSizeRecord(record))
  1170. {
  1171. OwnedHqlExpr ensureSize = adjustValue(bound.expr, virtualSize);
  1172. s.clear().append("crSelf.ensureCapacity(");
  1173. translator.generateExprCpp(s, ensureSize).append(", NULL);");
  1174. ctx.addQuoted(s);
  1175. }
  1176. s.clear().append("memcpy(crSelf.row(), left, ");
  1177. translator.generateExprCpp(s, bound.expr).append(");");
  1178. ctx.addQuoted(s);
  1179. ForEachChild(idx2, record)
  1180. {
  1181. IHqlExpression * field = record->queryChild(idx2);
  1182. IHqlExpression * virtualAttr = field->queryAttribute(virtualAtom);
  1183. if (virtualAttr)
  1184. {
  1185. IHqlExpression * self = rootSelfRow->querySelector();
  1186. OwnedHqlExpr target = createSelectExpr(LINK(self), LINK(field));
  1187. OwnedHqlExpr value = getVirtualReplacement(field, virtualAttr->queryChild(0), expr);
  1188. translator.buildAssign(ctx, target, value);
  1189. }
  1190. }
  1191. }
  1192. break;
  1193. case no_null:
  1194. case no_anon:
  1195. case no_pseudods:
  1196. break;
  1197. case no_workunit_dataset:
  1198. case no_getgraphresult:
  1199. case no_call:
  1200. case no_externalcall:
  1201. case no_rows:
  1202. case no_libraryinput:
  1203. throwUnexpectedOp(op);
  1204. break;
  1205. case no_select:
  1206. processTransformSelect(ctx, expr);
  1207. break;
  1208. case no_sorted:
  1209. case no_stepped:
  1210. case no_distributed:
  1211. case no_preservemeta:
  1212. case no_unordered:
  1213. case no_grouped:
  1214. case no_preload:
  1215. case no_limit:
  1216. case no_keyedlimit:
  1217. case no_choosen:
  1218. case no_alias_scope:
  1219. case no_section:
  1220. case no_sectioninput:
  1221. case no_nofold:
  1222. case no_nohoist:
  1223. case no_nocombine:
  1224. break;
  1225. case no_filter:
  1226. {
  1227. if (!ignoreFilters)
  1228. {
  1229. LinkedHqlExpr cond;
  1230. if (useFilterMappings)
  1231. {
  1232. unsigned match = originalFilters.find(*expr);
  1233. if (match != NotFound)
  1234. cond.set(&mappedFilters.item(match));
  1235. }
  1236. else
  1237. {
  1238. HqlExprArray args;
  1239. unwindRealChildren(args, expr, 1);
  1240. cond.setown(createBalanced(no_and, queryBoolType(), args));
  1241. }
  1242. if (cond)
  1243. {
  1244. IHqlExpression * ds = expr->queryChild(0);
  1245. OwnedHqlExpr test = returnIfFilterFails ? getInverse(cond) : LINK(cond);
  1246. if (translator.queryOptions().foldFilter)
  1247. test.setown(foldScopedHqlExpression(translator.queryErrorProcessor(), ds->queryNormalizedSelector(), test));
  1248. if (translator.options.spotCSE)
  1249. test.setown(spotScalarCSE(test, ds, translator.queryOptions().spotCseInIfDatasetConditions));
  1250. if (!returnIfFilterFails)
  1251. translator.buildFilter(ctx, test);
  1252. else
  1253. {
  1254. LinkedHqlExpr mismatchReturnValue = failedFilterValue;
  1255. //If the output row has already been generated, then returning at this point will leak any
  1256. //child datasets. To avoid that we explicitly call the destructor on the output row.
  1257. if (recordRequiresDestructor(expr->queryRecord()))
  1258. {
  1259. if (lastTransformer && lastTransformer->queryNormalizedSelector() == expr->queryNormalizedSelector())
  1260. {
  1261. StringBuffer s;
  1262. translator.buildMetaForRecord(s, expr->queryRecord());
  1263. s.append(".destruct(crSelf.row())");
  1264. OwnedHqlExpr cleanupAction = createQuoted(s.str(), makeVoidType());
  1265. //Create a compound expression (destroy-old, return-value)
  1266. mismatchReturnValue.setown(createCompound(LINK(cleanupAction), LINK(mismatchReturnValue)));
  1267. }
  1268. }
  1269. translator.buildFilteredReturn(ctx, test, mismatchReturnValue);
  1270. }
  1271. }
  1272. }
  1273. }
  1274. break;
  1275. case no_hqlproject:
  1276. {
  1277. IHqlExpression * dataset = expr->queryChild(0);
  1278. IHqlExpression * selSeq = querySelSeq(expr);
  1279. OwnedHqlExpr leftSelect = createSelector(no_left, dataset, querySelSeq(expr));
  1280. //Following is a bit nasty....
  1281. //Converting the no_hqlproject to a no_newusertable means that some of the expressions
  1282. //are commoned up with expressions calculated by a previous filter, reducing the code.
  1283. //However, it isn't valid if transform contains an instance of newSelect -
  1284. //e.g. project(i(x), transform(exists(i....))) - see jholt25.xhql
  1285. //And unfortunately it fails silently.
  1286. //So we use queryReplaceSelector which fails if an ambiguity is introduced by the replacement
  1287. OwnedHqlExpr newSelect = ensureActiveRow(dataset->queryNormalizedSelector());
  1288. OwnedHqlExpr transform = queryNewReplaceSelector(expr->queryChild(1), leftSelect, newSelect);
  1289. BuildCtx subctx(ctx); // buildTargetCursor adds group if necessary
  1290. Linked<BoundRow> tempRow;
  1291. Linked<BoundRow> rowBuilder;
  1292. buildTargetCursor(tempRow, rowBuilder, subctx, expr);
  1293. if (!transform)
  1294. {
  1295. //The replace introduced an ambiguity => need to use the unmapped expression.
  1296. BoundRow * prevCursor = translator.resolveSelectorDataset(ctx, dataset->queryNormalizedSelector());
  1297. transform.set(expr->queryChild(1));
  1298. translator.rebindTableCursor(subctx, dataset, prevCursor, no_left, selSeq);
  1299. rebindFilepositons(subctx, dataset, no_left, selSeq);
  1300. }
  1301. if (returnIfFilterFails)
  1302. {
  1303. translator.associateSkipReturnMarker(subctx, failedFilterValue, NULL); // failedFilterValue already handles clearing the result
  1304. }
  1305. else
  1306. {
  1307. //MORE: Probably not implementable, should try and prevent this happening!
  1308. //translator.associateSkipReturnMarker(subctx, failedFilterValue, NULL);
  1309. }
  1310. translator.doTransform(subctx, transform, rowBuilder);
  1311. associateTargetCursor(subctx, ctx, tempRow, rowBuilder, expr);
  1312. }
  1313. break;
  1314. case no_newusertable:
  1315. {
  1316. BuildCtx subctx(ctx);
  1317. Linked<BoundRow> tempRow;
  1318. Linked<BoundRow> rowBuilder;
  1319. buildTargetCursor(tempRow, rowBuilder, subctx, expr);
  1320. IHqlExpression * transform = expr->queryChild(2);
  1321. if (returnIfFilterFails)
  1322. {
  1323. translator.associateSkipReturnMarker(subctx, failedFilterValue, NULL); // failedFilterValue already handles clearing the result
  1324. }
  1325. else
  1326. {
  1327. //MORE: Probably not implementable
  1328. //translator.associateSkipReturnMarker(subctx, failedFilterValue, NULL);
  1329. }
  1330. translator.doTransform(subctx, transform, rowBuilder);
  1331. associateTargetCursor(subctx, ctx, tempRow, rowBuilder, expr);
  1332. }
  1333. break;
  1334. case no_aggregate:
  1335. {
  1336. if (alreadyDoneFlag)
  1337. {
  1338. IHqlExpression * dataset = expr->queryChild(0);
  1339. IHqlExpression * selSeq = querySelSeq(expr);
  1340. IHqlExpression * transform = expr->queryChild(2);
  1341. BuildCtx subctx(ctx);
  1342. //Similar code to no_hqlproject. Should possibly try and do a similar mapping to the transforms
  1343. //(but complicated by having merge/first)
  1344. BoundRow * prevCursor = translator.resolveSelectorDataset(ctx, dataset);
  1345. translator.rebindTableCursor(subctx, dataset, prevCursor, no_left, selSeq);
  1346. rebindFilepositons(subctx, dataset, no_left, selSeq);
  1347. Linked<BoundRow> tempRow;
  1348. Linked<BoundRow> rowBuilder;
  1349. buildTargetCursor(tempRow, rowBuilder, subctx, expr);
  1350. translator.doBuildUserAggregateProcessTransform(subctx, rowBuilder, expr, transform, alreadyDoneFlag);
  1351. subctx.addAssign(alreadyDoneFlag, queryBoolExpr(true));
  1352. associateTargetCursor(subctx, ctx, tempRow, rowBuilder, expr);
  1353. }
  1354. }
  1355. break;
  1356. case no_newaggregate:
  1357. {
  1358. IHqlExpression * transform = expr->queryChild(2);
  1359. if (isCompoundCount)
  1360. {
  1361. //This should really be special cased in a count() baseclass, but can't be bothered at the moment.
  1362. IHqlExpression * rhs = transform->queryChild(0)->queryChild(1);
  1363. IHqlExpression * filter = queryRealChild(rhs, 0);
  1364. if (filter)
  1365. {
  1366. if (!returnIfFilterFails)
  1367. translator.buildFilter(ctx, filter);
  1368. else
  1369. {
  1370. OwnedHqlExpr test = getInverse(filter);
  1371. translator.buildFilteredReturn(ctx, test, failedFilterValue);
  1372. }
  1373. }
  1374. if (compoundCountVar)
  1375. {
  1376. OwnedHqlExpr inc = adjustValue(compoundCountVar, 1);
  1377. ctx.addAssign(compoundCountVar, inc);
  1378. }
  1379. }
  1380. else if (alreadyDoneFlag)
  1381. {
  1382. BuildCtx subctx(ctx);
  1383. Linked<BoundRow> tempRow;
  1384. Linked<BoundRow> rowBuilder;
  1385. buildTargetCursor(tempRow, rowBuilder, subctx, expr);
  1386. translator.doBuildAggregateProcessTransform(subctx, rowBuilder, expr, alreadyDoneFlag);
  1387. subctx.addAssign(alreadyDoneFlag, queryBoolExpr(true));
  1388. associateTargetCursor(subctx, ctx, tempRow, rowBuilder, expr);
  1389. }
  1390. }
  1391. break;
  1392. case no_usertable:
  1393. case no_selectfields:
  1394. UNIMPLEMENTED;
  1395. break;
  1396. case no_fetch:
  1397. {
  1398. BuildCtx subctx(ctx);
  1399. Linked<BoundRow> tempRow;
  1400. Linked<BoundRow> rowBuilder;
  1401. buildTargetCursor(tempRow, rowBuilder, subctx, expr);
  1402. // MORE - don't understand why this is required here but not in hqlproject above
  1403. IHqlExpression * dataset = expr->queryChild(0);
  1404. BoundRow * leftCursor;
  1405. switch (getDatasetKind(tableExpr))
  1406. {
  1407. case no_csv:
  1408. leftCursor = translator.bindCsvTableCursor(subctx, dataset, "Left", no_left, querySelSeq(expr), true, queryCsvTableEncoding(tableExpr));
  1409. break;
  1410. case no_xml:
  1411. case no_json:
  1412. leftCursor = translator.bindXmlTableCursor(subctx, dataset, "xmlLeft", no_left, querySelSeq(expr), true);
  1413. break;
  1414. default:
  1415. leftCursor = translator.bindTableCursor(subctx, dataset, "left", no_left, querySelSeq(expr));
  1416. break;
  1417. }
  1418. BoundRow * rightCursor = NULL;
  1419. LinkedHqlExpr transform = expr->queryChild(3);
  1420. if (!containsOnlyLeft(transform, true))
  1421. {
  1422. IHqlExpression * rhs = expr->queryChild(1);
  1423. IHqlExpression * memoryRhsRecord = rhs->queryRecord();
  1424. //RIGHT is the input data row.
  1425. //a) In hthor the remote file is read locally, so RIGHT is a deserialized unmodified row.
  1426. //b) in roxie it is serialized, and accessed in its serialized form.
  1427. //c) in thor it is serialized, but also (inefficiently) deserialized, so use the deserialized form.
  1428. //NOTE: Currently extractJoinFields either does nothing, or sends the entire row. Needless to say that could be improved -
  1429. //and would mean the roxie/thor code required changing
  1430. if (translator.targetRoxie())
  1431. {
  1432. OwnedHqlExpr serializedRhsRecord = getSerializedForm(memoryRhsRecord, diskAtom);
  1433. OwnedHqlExpr serializedRhs = createDataset(no_null, LINK(serializedRhsRecord));
  1434. rightCursor = translator.bindTableCursor(subctx, serializedRhs, "right", no_right, querySelSeq(expr));
  1435. transform.setown(replaceMemorySelectorWithSerializedSelector(transform, memoryRhsRecord, no_right, querySelSeq(expr), diskAtom));
  1436. }
  1437. else
  1438. {
  1439. rightCursor = translator.bindTableCursor(subctx, rhs, "right", no_right, querySelSeq(expr));
  1440. }
  1441. }
  1442. buildTransformFpos(subctx); // unusual, but this must occur after the row cursor is bound
  1443. translator.associateSkipReturnMarker(subctx, failedFilterValue, NULL); // failedFilterValue already handles clearing the result
  1444. translator.doTransform(subctx, transform, rowBuilder);
  1445. subctx.removeAssociation(leftCursor);
  1446. subctx.removeAssociation(rightCursor);
  1447. associateTargetCursor(subctx, ctx, tempRow, rowBuilder, expr);
  1448. }
  1449. break;
  1450. case no_compound_diskread:
  1451. case no_compound_disknormalize:
  1452. case no_compound_diskaggregate:
  1453. case no_compound_diskcount:
  1454. case no_compound_diskgroupaggregate:
  1455. case no_compound_indexread:
  1456. case no_compound_indexnormalize:
  1457. case no_compound_indexaggregate:
  1458. case no_compound_indexcount:
  1459. case no_compound_indexgroupaggregate:
  1460. case no_compound_childread:
  1461. case no_compound_childnormalize:
  1462. case no_compound_childaggregate:
  1463. case no_compound_childcount:
  1464. case no_compound_childgroupaggregate:
  1465. case no_compound_fetch:
  1466. case no_compound_selectnew:
  1467. break;
  1468. default:
  1469. throwUnexpectedOp(op);
  1470. }
  1471. }
  1472. void SourceBuilder::doBuildAggregateSelectIterator(BuildCtx & ctx, IHqlExpression * expr)
  1473. {
  1474. IHqlExpression * ds = expr->queryChild(0);
  1475. if (isNewSelector(expr))
  1476. buildTransformElements(ctx, ds, false);
  1477. Owned<IHqlCppDatasetCursor> cursor = translator.createDatasetSelector(ctx, expr->queryNormalizedSelector());
  1478. cursor->buildIterateLoop(ctx, false);
  1479. }
  1480. void SourceBuilder::processTransformSelect(BuildCtx & ctx, IHqlExpression * expr)
  1481. {
  1482. throwUnexpected();
  1483. }
  1484. void SourceBuilder::doBuildNormalizeIterators(BuildCtx & ctx, IHqlExpression * expr, bool isChildIterator)
  1485. {
  1486. HqlExprArray iterators;
  1487. IHqlExpression * root = gatherSelectorLevels(iterators, expr);
  1488. //MORE: transform also needs to be inside this iterctx
  1489. BuildCtx iterctx(*globaliterctx);
  1490. BuildCtx firstctx(instance->startctx);
  1491. CursorArray cursors;
  1492. if (isChildIterator)
  1493. {
  1494. assertex(!root);
  1495. firstctx.addQuotedCompoundLiteral("virtual bool first()");
  1496. }
  1497. else
  1498. {
  1499. assertex(root);
  1500. firstctx.addQuotedCompoundLiteral("virtual bool first(const void * _src)");
  1501. bool isProjected = (root->queryNormalizedSelector() != tableExpr->queryNormalizedSelector());
  1502. if (!isProjected)
  1503. {
  1504. iterctx.addQuotedLiteral("byte * src;");
  1505. associateFilePositions(iterctx, "activity->fpp", "activity->src"); // in case no projection in first()
  1506. translator.associateBlobHelper(iterctx, tableExpr, "fpp");
  1507. firstctx.addQuotedLiteral("src = (byte *)_src;");
  1508. }
  1509. else
  1510. {
  1511. firstctx.addQuotedLiteral("byte * src = (byte *)_src;");
  1512. }
  1513. translator.associateBlobHelper(firstctx, tableExpr, "fpp");
  1514. BoundRow * tableCursor = translator.bindTableCursor(firstctx, tableExpr, "src");
  1515. associateFilePositions(firstctx, "fpp", "src");
  1516. TransformSequenceBuilder builder(translator, queryBoolExpr(false));
  1517. builder.buildSequence(firstctx, &iterctx, root);
  1518. if (!isProjected)
  1519. cursors.append(*LINK(tableCursor));
  1520. else
  1521. {
  1522. BoundRow * match = translator.resolveSelectorDataset(firstctx, root);
  1523. assertex(match);
  1524. cursors.append(*LINK(match));
  1525. }
  1526. }
  1527. CompoundIteratorBuilder iterBuilder(translator, instance->nestedctx, iterctx);
  1528. if (iterators.ordinality() == 1)
  1529. {
  1530. StringBuffer s, iterName, cursorName;
  1531. iterBuilder.createSingleLevelIterator(iterName, cursorName, &iterators.item(0), cursors);
  1532. firstctx.addQuoted(s.clear().append("return (").append(cursorName).append(" = (byte *)").append(iterName).append(".first()) != 0;"));
  1533. BuildCtx nextctx(instance->startctx);
  1534. nextctx.addQuotedCompoundLiteral("virtual bool next()");
  1535. nextctx.addQuoted(s.clear().append("return (").append(cursorName).append(" = (byte *)").append(iterName).append(".next()) != 0;"));
  1536. }
  1537. else
  1538. {
  1539. iterBuilder.buildCompoundIterator(instance->onlyEvalOnceContext(), iterators, cursors);
  1540. firstctx.addQuotedLiteral("return iter.first();");
  1541. BuildCtx nextctx(instance->startctx);
  1542. nextctx.addQuotedCompoundLiteral("virtual bool next()");
  1543. nextctx.addQuotedLiteral("return iter.next();");
  1544. }
  1545. ForEachItemIn(i, cursors)
  1546. ctx.associate(cursors.item(i));
  1547. }
  1548. void SourceBuilder::checkDependencies(BuildCtx & ctx, IHqlExpression * expr)
  1549. {
  1550. //Add dependency information
  1551. Owned<ABoundActivity> bound = instance->getBoundActivity();
  1552. translator.addFileDependency(nameExpr, bound);
  1553. }
  1554. void SourceBuilder::extractMonitors(IHqlExpression * ds, SharedHqlExpr & unkeyedFilter, HqlExprArray & conds)
  1555. {
  1556. ForEachItemIn(i, conds)
  1557. {
  1558. IHqlExpression * filter = &conds.item(i);
  1559. #ifdef ALLOW_CHILD_ITERATORS_TO_CALL_CANMATCHANY
  1560. if (isSourceInvariant(ds, filter)) // more actually isSourceInvariant.
  1561. extendAndCondition(globalGuard, filter);
  1562. else
  1563. #endif
  1564. appendFilter(unkeyedFilter, filter);
  1565. }
  1566. }
  1567. void SourceBuilder::analyseGraph(IHqlExpression * expr)
  1568. {
  1569. analyse(expr);
  1570. SourceFieldUsage * fieldUsage = translator.querySourceFieldUsage(tableExpr);
  1571. if (fieldUsage && !fieldUsage->seenAll())
  1572. {
  1573. if (expr->queryNormalizedSelector() == tableExpr->queryNormalizedSelector())
  1574. fieldUsage->noteAll();
  1575. else
  1576. gatherFieldUsage(fieldUsage, expr);
  1577. }
  1578. }
  1579. void SourceBuilder::gatherFieldUsage(SourceFieldUsage * fieldUsage, IHqlExpression * expr)
  1580. {
  1581. loop
  1582. {
  1583. if (expr->queryBody() == tableExpr->queryBody())
  1584. return;
  1585. if (fieldUsage->seenAll())
  1586. return;
  1587. IHqlExpression * ds = expr->queryChild(0);
  1588. switch (expr->getOperator())
  1589. {
  1590. case no_fetch:
  1591. {
  1592. assertex(ds->queryBody() == tableExpr->queryBody());
  1593. IHqlExpression * selSeq = querySelSeq(expr);
  1594. OwnedHqlExpr left = createSelector(no_left, ds, selSeq);
  1595. ::gatherFieldUsage(fieldUsage, expr, left);
  1596. return;
  1597. }
  1598. }
  1599. assertex(getNumChildTables(expr) == 1);
  1600. if (ds->queryNormalizedSelector() == tableExpr->queryNormalizedSelector())
  1601. gatherParentFieldUsage(fieldUsage, expr);
  1602. expr = ds;
  1603. }
  1604. }
  1605. inline bool useDescriptiveGraphLabel(ThorActivityKind kind)
  1606. {
  1607. switch (kind)
  1608. {
  1609. case TAKcsvfetch:
  1610. case TAKxmlfetch:
  1611. case TAKjsonfetch:
  1612. case TAKfetch:
  1613. return false;
  1614. }
  1615. return true;
  1616. }
  1617. ABoundActivity * SourceBuilder::buildActivity(BuildCtx & ctx, IHqlExpression * expr, ThorActivityKind _activityKind, const char *kind, ABoundActivity *input)
  1618. {
  1619. activityKind = _activityKind;
  1620. translator.gatherActiveCursors(ctx, parentCursors);
  1621. bool isSpill = tableExpr && tableExpr->hasAttribute(_spill_Atom);
  1622. useImplementationClass = translator.queryOptions().minimizeActivityClasses && translator.targetRoxie() && isSpill;
  1623. Owned<ActivityInstance> localInstance = new ActivityInstance(translator, ctx, activityKind, expr, kind);
  1624. if (useImplementationClass)
  1625. localInstance->setImplementationClass(newMemorySpillReadArgId);
  1626. if ((activityKind >= TAKdiskread) && (activityKind <= TAKdiskgroupaggregate))
  1627. {
  1628. IHqlExpression * seq = querySequence(tableExpr);
  1629. translator.noteResultAccessed(ctx, seq, nameExpr);
  1630. }
  1631. instance = localInstance;
  1632. StringBuffer graphLabel;
  1633. graphLabel.append(getActivityText(activityKind));
  1634. bool isFiltered = false;
  1635. double filterLikelihood = 1.0;
  1636. if ((activityKind == TAKdiskread) || (activityKind == TAKcsvread) || (activityKind == TAKxmlread) || (activityKind == TAKjsonread))
  1637. {
  1638. graphLabel.clear();
  1639. if (expr != tableExpr)
  1640. {
  1641. IHqlExpression * cur = expr;
  1642. bool isProjected = false;
  1643. loop
  1644. {
  1645. switch (cur->getOperator())
  1646. {
  1647. case no_filter:
  1648. if (isKnownLikelihood(filterLikelihood))
  1649. {
  1650. double likelihood = queryActivityLikelihood(cur);
  1651. if (isKnownLikelihood(likelihood))
  1652. // Combine the likelihood of the 2 filter conditions
  1653. // N.B. this only works if the filter probability are independent
  1654. filterLikelihood *= likelihood;
  1655. else
  1656. // One of the filter probability is unknown, so the overall probability is unknown
  1657. setUnknownLikelihood(filterLikelihood);
  1658. }
  1659. isFiltered = true;
  1660. break;
  1661. case no_hqlproject:
  1662. case no_newusertable:
  1663. case no_transformascii:
  1664. case no_transformebcdic:
  1665. if (!cur->hasAttribute(_internal_Atom))
  1666. isProjected = true;
  1667. break;
  1668. case no_table:
  1669. cur = NULL;
  1670. break;
  1671. }
  1672. if (!cur)
  1673. break;
  1674. cur = cur->queryChild(0);
  1675. }
  1676. if (isFiltered)
  1677. {
  1678. graphLabel.append("Filtered\n");
  1679. }
  1680. if (isProjected)
  1681. graphLabel.append("Projected\n");
  1682. }
  1683. if ((translator.getTargetClusterType() == RoxieCluster) && isSpill)
  1684. graphLabel.append("Read");
  1685. else
  1686. graphLabel.append(getActivityText(activityKind));
  1687. }
  1688. if (isExplicitExists())
  1689. {
  1690. if (activityKind == TAKindexcount)
  1691. graphLabel.clear().append("Index Exists");
  1692. else
  1693. graphLabel.clear().append("Disk Exists");
  1694. }
  1695. if (useDescriptiveGraphLabel(activityKind))
  1696. {
  1697. StringBuffer eclChunk;
  1698. bool isLimited = false;
  1699. bool isStepped = false;
  1700. IHqlExpression * cur = expr;
  1701. instance->graphEclText.clear();
  1702. loop
  1703. {
  1704. eclChunk.clear();
  1705. switch (cur->getOperator())
  1706. {
  1707. case no_hqlproject:
  1708. case no_newusertable:
  1709. case no_transformascii:
  1710. case no_transformebcdic:
  1711. break;
  1712. case no_filter:
  1713. toECL(cur->queryBody(), eclChunk, false, true);
  1714. break;
  1715. case no_table:
  1716. case no_newkeyindex:
  1717. case no_select:
  1718. toECL(cur->queryBody(), eclChunk, false, true);
  1719. cur = NULL;
  1720. break;
  1721. case no_stepped:
  1722. isStepped = true;
  1723. break;
  1724. case no_limit:
  1725. case no_keyedlimit:
  1726. isLimited = true;
  1727. break;
  1728. default:
  1729. if (getNumChildTables(cur) == 0)
  1730. {
  1731. toECL(cur->queryBody(), eclChunk, false, true);
  1732. cur = NULL;
  1733. }
  1734. break;
  1735. }
  1736. instance->graphEclText.insert(0, eclChunk);
  1737. if (!cur)
  1738. break;
  1739. cur = cur->queryChild(0);
  1740. }
  1741. if (isLimited)
  1742. graphLabel.insert(0, "Limited\n");
  1743. if (isStepped)
  1744. graphLabel.insert(0, "Stepped\n");
  1745. if (localInstance->isLocal && !isSpill)
  1746. graphLabel.insert(0, "Local ");
  1747. }
  1748. if (nameExpr && nameExpr->queryValue())
  1749. {
  1750. if (isSpill)
  1751. graphLabel.append("\nSpill");
  1752. else
  1753. {
  1754. StringBuffer filename;
  1755. //Call getStringValue() rather than generateECL() to avoid 't quote \ etc. in the string
  1756. getStringValue(filename.append("'"), nameExpr).append("'");
  1757. const char * coloncolon = strstr(filename, "::");
  1758. if (coloncolon)
  1759. {
  1760. loop
  1761. {
  1762. const char * next = strstr(coloncolon+2, "::");
  1763. if (!next)
  1764. break;
  1765. coloncolon = next;
  1766. }
  1767. graphLabel.newline().append("'...").append(coloncolon);
  1768. }
  1769. else
  1770. graphLabel.newline().append(filename);
  1771. }
  1772. }
  1773. instance->graphLabel.set(graphLabel.str());
  1774. translator.buildActivityFramework(localInstance);
  1775. translator.buildInstancePrefix(localInstance);
  1776. analyseGraph(expr);
  1777. if (!useImplementationClass)
  1778. {
  1779. buildMembers(expr);
  1780. buildTransform(expr);
  1781. buildFlagsMember(expr);
  1782. if (tableExpr)
  1783. {
  1784. if (fieldInfo.hasVirtualsOrDeserialize())
  1785. {
  1786. OwnedHqlExpr diskTable = createDataset(no_anon, LINK(physicalRecord));
  1787. translator.buildMetaMember(instance->classctx, diskTable, false, "queryDiskRecordSize");
  1788. }
  1789. else
  1790. translator.buildMetaMember(instance->classctx, tableExpr, isGrouped(tableExpr), "queryDiskRecordSize");
  1791. }
  1792. }
  1793. else
  1794. {
  1795. assertex(!hasDynamic(tableExpr));
  1796. bool matched = translator.registerGlobalUsage(nameExpr);
  1797. if (!matched)
  1798. {
  1799. StringBuffer spillName;
  1800. getExprECL(nameExpr, spillName);
  1801. if (translator.queryOptions().allowThroughSpill)
  1802. throwError1(HQLERR_ReadSpillBeforeWriteFix, spillName.str());
  1803. else
  1804. throwError1(HQLERR_ReadSpillBeforeWrite, spillName.str());
  1805. }
  1806. translator.addFilenameConstructorParameter(*instance, "getFileName", nameExpr);
  1807. }
  1808. if (steppedExpr)
  1809. buildSteppedHelpers();
  1810. if (translator.targetRoxie())
  1811. instance->addAttributeBool("_isSpill", isSpill);
  1812. else if (needToCallTransform || transformCanFilter)
  1813. instance->addAttributeBool("_isTransformSpill", isSpill);
  1814. else
  1815. instance->addAttributeBool("_isSpill", isSpill);
  1816. if (isFiltered)
  1817. {
  1818. if (isKnownLikelihood(filterLikelihood))
  1819. {
  1820. StringBuffer text;
  1821. filterLikelihood *= 100;
  1822. text.setf("%3.2f%%", filterLikelihood);
  1823. instance->addAttribute("matchLikelihood", text);
  1824. }
  1825. }
  1826. IHqlExpression * spillReason = tableExpr ? queryAttributeChild(tableExpr, _spillReason_Atom, 0) : NULL;
  1827. if (spillReason && !translator.queryOptions().obfuscateOutput)
  1828. {
  1829. StringBuffer text;
  1830. getStringValue(text, spillReason);
  1831. instance->addAttribute("spillReason", text.str());
  1832. }
  1833. checkDependencies(ctx, expr);
  1834. translator.buildInstanceSuffix(localInstance);
  1835. if (input)
  1836. translator.buildConnectInputOutput(ctx, localInstance, input, 0, 0);
  1837. instance = NULL;
  1838. return localInstance->getBoundActivity();
  1839. }
  1840. void SourceBuilder::buildKeyedLimitHelper(IHqlExpression * self)
  1841. {
  1842. if (keyedLimitExpr)
  1843. {
  1844. IHqlExpression * limitValue = keyedLimitExpr->queryChild(1);
  1845. BuildCtx func1ctx(instance->startctx);
  1846. func1ctx.addQuotedCompoundLiteral("virtual unsigned __int64 getKeyedLimit()");
  1847. translator.buildReturn(func1ctx, limitValue);
  1848. if (isZero(limitValue))
  1849. translator.WARNING(CategoryUnusual, HQLWRN_KeyedLimitIsZero);
  1850. LinkedHqlExpr fail = keyedLimitExpr->queryChild(2);
  1851. if (!fail || fail->isAttribute())
  1852. fail.setown(translator.createFailAction("Keyed limit exceeded", limitValue, NULL, instance->activityId));
  1853. BuildCtx func2ctx(instance->startctx);
  1854. func2ctx.addQuotedCompoundLiteral("virtual void onKeyedLimitExceeded()");
  1855. translator.buildStmt(func2ctx, fail);
  1856. IHqlExpression * transform = queryAttributeChild(keyedLimitExpr, onFailAtom, 0);
  1857. if (transform)
  1858. {
  1859. BuildCtx transformctx(instance->startctx);
  1860. transformctx.addQuotedCompoundLiteral("virtual size32_t transformOnKeyedLimitExceeded(ARowBuilder & crSelf)");
  1861. translator.ensureRowAllocated(transformctx, "crSelf");
  1862. translator.buildTransformBody(transformctx, transform, NULL, NULL, self, NULL);
  1863. }
  1864. }
  1865. }
  1866. void SourceBuilder::buildSteppedHelpers()
  1867. {
  1868. StringBuffer steppedFlags;
  1869. IHqlExpression * priority = steppedExpr->queryAttribute(priorityAtom);
  1870. if (priority)
  1871. {
  1872. steppedFlags.append("|SSFhaspriority");
  1873. translator.doBuildFunction(instance->startctx, doubleType, "getPriority", priority->queryChild(0));
  1874. }
  1875. IHqlExpression * prefetch = steppedExpr->queryAttribute(prefetchAtom);
  1876. if (prefetch)
  1877. {
  1878. steppedFlags.append("|SSFhasprefetch");
  1879. translator.doBuildUnsignedFunction(instance->startctx, "getPrefetchSize", prefetch->queryChild(0));
  1880. }
  1881. IHqlExpression * filtered = steppedExpr->queryAttribute(filteredAtom);
  1882. if (filtered)
  1883. steppedFlags.append("|SSFalwaysfilter");
  1884. if (steppedFlags.length())
  1885. translator.doBuildUnsignedFunction(instance->classctx, "getSteppedFlags", steppedFlags.str()+1);
  1886. }
  1887. void SourceBuilder::assignLocalExtract(BuildCtx & ctx, ParentExtract * extractBuilder, IHqlExpression * dataset, const char * argName)
  1888. {
  1889. if (extractBuilder)
  1890. {
  1891. StringBuffer s;
  1892. s.append("byte * ");
  1893. translator.generateExprCpp(s, extractBuilder->queryExtractName());
  1894. s.append(" = (byte *) ").append(argName).append(";");
  1895. ctx.addQuoted(s);
  1896. }
  1897. else
  1898. {
  1899. StringBuffer s;
  1900. ctx.addQuoted(s.append("unsigned char * left = (unsigned char *) ").append(argName).append(";"));
  1901. translator.bindTableCursor(ctx, dataset, "left");
  1902. }
  1903. }
  1904. void SourceBuilder::buildGroupAggregateHashHelper(ParentExtract * extractBuilder, IHqlExpression * dataset, IHqlExpression * fields)
  1905. {
  1906. instance->classctx.addQuotedLiteral("virtual IHash * queryHash() { return &hash; }");
  1907. BuildCtx classctx(instance->nestedctx);
  1908. translator.beginNestedClass(classctx, "hash", "IHash", NULL, extractBuilder);
  1909. BuildCtx funcctx(classctx);
  1910. funcctx.addQuotedCompoundLiteral("virtual unsigned hash(const void * _self)");
  1911. assignLocalExtract(funcctx, extractBuilder, dataset, "_self");
  1912. OwnedHqlExpr hash = createValue(no_hash32, LINK(unsignedType), LINK(fields), createAttribute(internalAtom));
  1913. translator.buildReturn(funcctx, hash);
  1914. translator.endNestedClass();
  1915. }
  1916. void SourceBuilder::buildGroupAggregateCompareHelper(ParentExtract * extractBuilder, IHqlExpression * aggregate, HqlExprArray & recordFields, HqlExprArray & aggregateFields)
  1917. {
  1918. //Special case comparison of IF(<global-bool>, value, some-constant), so that the order expression only compares if <global-bool> is true
  1919. //specifically to improve david's BIRPs code.
  1920. HqlExprArray optimizedLeft, optimizedRight;
  1921. ForEachItemIn(i, recordFields)
  1922. {
  1923. IHqlExpression & curLeft = recordFields.item(i);
  1924. IHqlExpression & curRight = aggregateFields.item(i);
  1925. IHqlExpression * newLeft = NULL;
  1926. IHqlExpression * newRight = NULL;
  1927. if (curLeft.getOperator() == no_if)
  1928. {
  1929. IHqlExpression * cond = curLeft.queryChild(0);
  1930. if (isIndependentOfScope(cond))
  1931. {
  1932. IHqlExpression * elseValue = curLeft.queryChild(2);
  1933. if (elseValue->getOperator() == no_constant)
  1934. {
  1935. newLeft = LINK(&curLeft);
  1936. newRight = createValue(no_if, curRight.getType(), LINK(cond), LINK(&curRight), LINK(elseValue));
  1937. }
  1938. }
  1939. }
  1940. if (newLeft)
  1941. {
  1942. optimizedLeft.append(*newLeft);
  1943. optimizedRight.append(*newRight);
  1944. }
  1945. else
  1946. {
  1947. optimizedLeft.append(OLINK(curLeft));
  1948. optimizedRight.append(OLINK(curRight));
  1949. }
  1950. }
  1951. OwnedHqlExpr leftList = createSortList(optimizedLeft);
  1952. DatasetReference datasetRight(aggregate, no_activetable, NULL);
  1953. OwnedHqlExpr selSeq = createDummySelectorSequence();
  1954. OwnedHqlExpr rightList = createSortList(optimizedRight);
  1955. OwnedHqlExpr rightSelect = datasetRight.getSelector(no_right, selSeq);
  1956. OwnedHqlExpr rightResolved = datasetRight.mapCompound(rightList, rightSelect);
  1957. OwnedHqlExpr order = createValue(no_order, makeIntType(sizeof(signed), true), LINK(leftList), LINK(rightResolved));
  1958. //Now generate the nested class
  1959. instance->classctx.addQuotedLiteral("virtual ICompare * queryCompareRowElement() { return &compareRowElement; }");
  1960. BuildCtx classctx(instance->nestedctx);
  1961. translator.beginNestedClass(classctx, "compareRowElement", "ICompare", NULL, extractBuilder);
  1962. BuildCtx funcctx(classctx);
  1963. funcctx.addQuotedCompoundLiteral("virtual int docompare(const void * _left, const void * _right) const");
  1964. assignLocalExtract(funcctx, extractBuilder, aggregate->queryChild(0), "_left");
  1965. funcctx.addQuotedLiteral("const unsigned char * right = (const unsigned char *) _right;");
  1966. funcctx.associateExpr(constantMemberMarkerExpr, constantMemberMarkerExpr);
  1967. translator.bindTableCursor(funcctx, aggregate, "right", no_right, selSeq);
  1968. translator.doBuildReturnCompare(funcctx, order, no_eq, false, false);
  1969. translator.endNestedClass();
  1970. }
  1971. void SourceBuilder::buildGroupAggregateProcessHelper(ParentExtract * extractBuilder, IHqlExpression * aggregate, const char * name, bool doneAny)
  1972. {
  1973. StringBuffer s;
  1974. IHqlExpression * dataset = aggregate->queryChild(0);
  1975. IHqlExpression * tgtRecord = aggregate->queryChild(1);
  1976. OwnedHqlExpr resultDataset = createDataset(no_anon, LINK(tgtRecord));
  1977. BuildCtx funcctx(instance->nestedctx);
  1978. funcctx.addQuotedCompound(s.clear().append("virtual size32_t ").append(name).append("(ARowBuilder & crSelf, const void * _src)"));
  1979. translator.ensureRowAllocated(funcctx, "crSelf");
  1980. assignLocalExtract(funcctx, extractBuilder, dataset, "_src");
  1981. BoundRow * selfRow = translator.bindSelf(funcctx, resultDataset, "crSelf");
  1982. if (extractBuilder)
  1983. {
  1984. MemberEvalContext * evalContext = new MemberEvalContext(translator, extractBuilder, translator.queryEvalContext(funcctx), funcctx);
  1985. funcctx.associateOwn(*evalContext);
  1986. evalContext->initContext();
  1987. }
  1988. if (aggregate->getOperator() == no_aggregate)
  1989. {
  1990. //It is inefficient to call processUserAggregateTransform() twice, but this is an unusual construct, so leave as it
  1991. //is for the moment.
  1992. OwnedHqlExpr firstTransform;
  1993. OwnedHqlExpr nextTransform;
  1994. translator.processUserAggregateTransform(aggregate, aggregate->queryChild(2), firstTransform, nextTransform);
  1995. IHqlExpression * transform = doneAny ? nextTransform : firstTransform;
  1996. OwnedHqlExpr left = createSelector(no_left, dataset, querySelSeq(aggregate));
  1997. OwnedHqlExpr mappedTransform = replaceSelector(transform, left, dataset);
  1998. translator.doBuildUserAggregateProcessTransform(funcctx, selfRow, aggregate, mappedTransform, queryBoolExpr(doneAny));
  1999. }
  2000. else
  2001. translator.doBuildAggregateProcessTransform(funcctx, selfRow, aggregate, queryBoolExpr(doneAny));
  2002. translator.buildReturnRecordSize(funcctx, selfRow);
  2003. }
  2004. void SourceBuilder::buildGroupAggregateHelpers(ParentExtract * extractBuilder, IHqlExpression * aggregate)
  2005. {
  2006. IHqlExpression * dataset = aggregate->queryChild(0);
  2007. LinkedHqlExpr transform = aggregate->queryChild(2);
  2008. LinkedHqlExpr grouping = aggregate->queryChild(3);
  2009. if (aggregate->getOperator() == no_aggregate)
  2010. {
  2011. OwnedHqlExpr left = createSelector(no_left, dataset, querySelSeq(aggregate));
  2012. grouping.setown(replaceSelector(grouping, left, dataset));
  2013. transform.setown(replaceSelector(transform, left, dataset));
  2014. }
  2015. HqlExprArray recordFields, aggregateFields;
  2016. grouping->unwindList(recordFields, no_sortlist);
  2017. getMappedFields(aggregateFields, transform, recordFields, queryActiveTableSelector());
  2018. OwnedHqlExpr allRecordFields = createValueSafe(no_sortlist, makeSortListType(NULL), recordFields);
  2019. OwnedHqlExpr allAggregateFields = createValueSafe(no_sortlist, makeSortListType(NULL), aggregateFields);
  2020. //virtual size32_t processFirst(void * target, const void * src)
  2021. buildGroupAggregateProcessHelper(extractBuilder, aggregate, "processFirst", false);
  2022. //virtual size32_t processNext(void * target, const void * src)
  2023. buildGroupAggregateProcessHelper(extractBuilder, aggregate, "processNext", true);
  2024. //virtual IHash * queryHash()
  2025. buildGroupAggregateHashHelper(extractBuilder, dataset, allRecordFields);
  2026. //virtual ICompare * queryCompareElements()
  2027. DatasetReference outRef(aggregate, no_activetable, NULL);
  2028. translator.buildCompareMember(instance->nestedctx, "CompareElements", allAggregateFields, outRef); // compare transformed elements
  2029. //virtual ICompare * queryCompareRowElement()
  2030. buildGroupAggregateCompareHelper(extractBuilder, aggregate, recordFields, aggregateFields);
  2031. //virtual IHash * queryHashElement()
  2032. translator.buildHashOfExprsClass(instance->nestedctx, "HashElement", allAggregateFields, outRef, true);
  2033. }
  2034. IHqlExpression * SourceBuilder::ensureAggregateGroupingAliased(IHqlExpression * aggregate)
  2035. {
  2036. IHqlExpression * dataset = aggregate->queryChild(0);
  2037. IHqlExpression * grouping = aggregate->queryChild(3);
  2038. //Force complex grouping fields into aliases to reduce processing...
  2039. HqlMapTransformer transformer;
  2040. transformer.setMapping(dataset, dataset);
  2041. ForEachChild(i, grouping)
  2042. {
  2043. IHqlExpression * cur = grouping->queryChild(i);
  2044. if (translator.requiresTemp(instance->nestedctx, cur, true) && (cur->getOperator() != no_alias))
  2045. {
  2046. OwnedHqlExpr alias = createAliasOwn(LINK(cur), createAttribute(internalAtom));
  2047. transformer.setMapping(cur, alias);
  2048. }
  2049. }
  2050. return transformer.transformRoot(aggregate);
  2051. }
  2052. void SourceBuilder::buildGlobalGroupAggregateHelpers(IHqlExpression * expr)
  2053. {
  2054. IHqlExpression * aggregate = expr->queryChild(0);
  2055. node_operator op = aggregate->getOperator();
  2056. assertex(op == no_newaggregate || op == no_aggregate);
  2057. StringBuffer s;
  2058. //virtual size32_t clearAggregate(void * self) = 0;
  2059. translator.doBuildAggregateClearFunc(instance->startctx, aggregate);
  2060. //virtual size32_t mergeAggregate(ARowBuilder & crSelf, const void * src) = 0; //only call if transform called at least once on src.
  2061. translator.doBuildAggregateMergeFunc(instance->startctx, aggregate, requiresOrderedMerge);
  2062. //virtual void processRow(void * self, const void * src) = 0;
  2063. BuildCtx rowctx(instance->startctx);
  2064. rowctx.addQuotedCompoundLiteral("virtual void processRow(const void * src, IHThorGroupAggregateCallback * callback)");
  2065. rowctx.addQuotedLiteral("doProcessRow((byte *)src, callback);");
  2066. //virtual void processRows(void * self, size32_t srcLen, const void * src) = 0;
  2067. BuildCtx rowsctx(instance->startctx);
  2068. rowsctx.addQuotedCompoundLiteral("virtual void processRows(size32_t srcLen, const void * _left, IHThorGroupAggregateCallback * callback)");
  2069. rowsctx.addQuotedLiteral("unsigned char * left = (unsigned char *)_left;");
  2070. OwnedHqlExpr ds = createVariable("left", makeReferenceModifier(tableExpr->getType()));
  2071. OwnedHqlExpr len = createVariable("srcLen", LINK(sizetType));
  2072. OwnedHqlExpr fullDs = createTranslated(ds, len);
  2073. BoundRow * curRow = translator.buildDatasetIterate(rowsctx, fullDs, false);
  2074. s.clear().append("doProcessRow(");
  2075. translator.generateExprCpp(s, curRow->queryBound());
  2076. s.append(", callback);");
  2077. rowsctx.addQuoted(s);
  2078. }
  2079. void SourceBuilder::buildGroupingMonitors(IHqlExpression * expr, MonitorExtractor & monitors)
  2080. {
  2081. IHqlExpression * aggregate = expr->queryChild(0);
  2082. node_operator op = aggregate->getOperator();
  2083. assertex(op == no_newaggregate || op == no_aggregate);
  2084. IHqlExpression * dataset = aggregate->queryChild(0);
  2085. IHqlExpression * grouping = aggregate->queryChild(3);
  2086. //virtual void createGroupSegmentMonitors(IIndexReadContext *ctx) = 0;
  2087. BuildCtx groupctx(instance->startctx);
  2088. IHqlStmt * group = groupctx.addQuotedCompoundLiteral("virtual bool createGroupSegmentMonitors(IIndexReadContext * irc)");
  2089. monitorsForGrouping = true;
  2090. if (op == no_newaggregate)
  2091. translator.bindTableCursor(groupctx, dataset, "_dummy");
  2092. else
  2093. translator.bindTableCursor(groupctx, dataset, "_dummy", no_left, querySelSeq(aggregate));
  2094. unsigned maxOffset = 0;
  2095. ForEachChild(i, grouping)
  2096. {
  2097. unsigned nextOffset = 0;
  2098. if (!monitors.createGroupingMonitor(groupctx, "irc", grouping->queryChild(i), nextOffset))
  2099. {
  2100. monitorsForGrouping = false;
  2101. group->setIncluded(false);
  2102. break;
  2103. }
  2104. if (maxOffset < nextOffset)
  2105. maxOffset = nextOffset;
  2106. }
  2107. if (monitorsForGrouping)
  2108. groupctx.addReturn(queryBoolExpr(true));
  2109. if (monitorsForGrouping)
  2110. translator.doBuildUnsignedFunction(instance->classctx, "getGroupSegmentMonitorsSize", maxOffset);
  2111. }
  2112. void SourceBuilder::buildAggregateHelpers(IHqlExpression * expr)
  2113. {
  2114. IHqlExpression * aggregate = expr->queryChild(0);
  2115. node_operator op = aggregate->getOperator();
  2116. assertex(op == no_newaggregate || op == no_aggregate);
  2117. StringBuffer s;
  2118. alreadyDoneFlag.setown(instance->startctx.getTempDeclare(queryBoolType(), NULL));
  2119. instance->onstartctx.addAssign(alreadyDoneFlag, queryBoolExpr(false));
  2120. //virtual bool processedAnyRows() = 0;
  2121. translator.doBuildBoolFunction(instance->startctx, "processedAnyRows", alreadyDoneFlag);
  2122. //virtual size32_t clearAggregate(ARowBuilder & crSelf) = 0;
  2123. translator.doBuildAggregateClearFunc(instance->startctx, aggregate);
  2124. //virtual size32_t mergeAggregate(ARowBuilder & crSelf, const void * src) = 0; //only call if transform called at least once on src.
  2125. translator.doBuildAggregateMergeFunc(instance->startctx, aggregate, requiresOrderedMerge);
  2126. }
  2127. void SourceBuilder::buildCountHelpers(IHqlExpression * expr, bool allowMultiple)
  2128. {
  2129. StringBuffer s;
  2130. //---- virtual bool hasFilter() { return <bool>; } ----
  2131. if (transformCanFilter||isNormalize)
  2132. translator.doBuildBoolFunction(instance->classctx, "hasFilter", true);
  2133. if (allowMultiple)
  2134. {
  2135. bool isExists = hasExistChoosenLimit();
  2136. OwnedHqlExpr one = getSizetConstant(1);
  2137. if (transformCanFilter||isNormalize)
  2138. {
  2139. //virtual size32_t numValid(const void * src) = 0;
  2140. BuildCtx rowctx(instance->startctx);
  2141. rowctx.addQuotedCompoundLiteral("virtual size32_t numValid(const void * src)");
  2142. rowctx.addQuotedLiteral("return valid((byte *)src);");
  2143. //virtual size32_t numValid(size32_t srcLen, const void * src);
  2144. BuildCtx rowsctx(instance->startctx);
  2145. rowsctx.addQuotedCompoundLiteral("virtual size32_t numValid(size32_t srcLen, const void * _src)");
  2146. rowsctx.addQuotedLiteral("unsigned char * src = (unsigned char *)_src;");
  2147. OwnedHqlExpr ds = createVariable("src", makeReferenceModifier(tableExpr->getType()));
  2148. OwnedHqlExpr len = createVariable("srcLen", LINK(sizetType));
  2149. OwnedHqlExpr fullDs = createTranslated(ds, len);
  2150. if (isExists)
  2151. {
  2152. BuildCtx iterctx(rowsctx);
  2153. BoundRow * curRow = translator.buildDatasetIterate(iterctx, fullDs, false);
  2154. s.clear().append("if (valid(");
  2155. translator.generateExprCpp(s, curRow->queryBound());
  2156. s.append("))");
  2157. iterctx.addQuotedCompound(s);
  2158. iterctx.addReturn(one);
  2159. rowsctx.addQuotedLiteral("return 0;");
  2160. }
  2161. else
  2162. {
  2163. rowsctx.addQuotedLiteral("size32_t cnt = 0;");
  2164. BuildCtx iterctx(rowsctx);
  2165. BoundRow * curRow = translator.buildDatasetIterate(iterctx, fullDs, false);
  2166. s.clear().append("cnt += valid(");
  2167. translator.generateExprCpp(s, curRow->queryBound());
  2168. s.append(");");
  2169. iterctx.addQuoted(s);
  2170. rowsctx.addQuotedLiteral("return cnt;");
  2171. }
  2172. }
  2173. else
  2174. {
  2175. //virtual size32_t numValid(size32_t srcLen, const void * src);
  2176. BuildCtx rowsctx(instance->startctx);
  2177. rowsctx.addQuotedCompoundLiteral("virtual size32_t numValid(size32_t srcLen, const void * _src)");
  2178. if (isExists)
  2179. rowsctx.addReturn(one);
  2180. else
  2181. {
  2182. rowsctx.addQuotedLiteral("unsigned char * src = (unsigned char *)_src;");
  2183. CHqlBoundExpr bound;
  2184. bound.length.setown(createVariable("srcLen", LINK(sizetType)));
  2185. bound.expr.setown(createVariable("src", makeReferenceModifier(tableExpr->getType())));
  2186. OwnedHqlExpr count = translator.getBoundCount(bound);
  2187. rowsctx.addReturn(count);
  2188. }
  2189. }
  2190. }
  2191. }
  2192. void SourceBuilder::buildNormalizeHelpers(IHqlExpression * expr)
  2193. {
  2194. }
  2195. void SourceBuilder::buildGroupAggregateTransformBody(BuildCtx & transformCtx, IHqlExpression * expr, bool useExtract, bool bindInputRow)
  2196. {
  2197. buildTransformBody(transformCtx, expr, false, false, bindInputRow);
  2198. IHqlExpression * aggregate = expr->queryChild(0);
  2199. OwnedHqlExpr mappedAggregate = ensureAggregateGroupingAliased(aggregate);
  2200. Owned<ParentExtract> extractBuilder;
  2201. if (useExtract || (aggregate != mappedAggregate))
  2202. {
  2203. extractBuilder.setown(translator.createExtractBuilder(transformCtx, PETcallback, NULL, GraphCoLocal, true));
  2204. if (!translator.queryOptions().serializeRowsetInExtract)
  2205. extractBuilder->setAllowDestructor();
  2206. translator.beginExtract(transformCtx, extractBuilder);
  2207. buildGroupAggregateHelpers(extractBuilder, mappedAggregate);
  2208. translator.endExtract(transformCtx, extractBuilder);
  2209. }
  2210. else
  2211. buildGroupAggregateHelpers(NULL, mappedAggregate);
  2212. HqlExprArray args;
  2213. args.append(*createVariable("callback", makeBoolType()));
  2214. if (extractBuilder)
  2215. {
  2216. CHqlBoundExpr boundExtract;
  2217. extractBuilder->endCreateExtract(boundExtract);
  2218. args.append(*boundExtract.getTranslatedExpr());
  2219. }
  2220. else
  2221. {
  2222. BoundRow * match = translator.resolveDatasetRequired(transformCtx, aggregate->queryChild(0));
  2223. Owned<ITypeInfo> rowType = makeReferenceModifier(makeRowType(queryNullRecord()->getType()));
  2224. OwnedHqlExpr rowAddr = getPointer(match->queryBound());
  2225. OwnedHqlExpr castBound = createValue(no_typetransfer, LINK(rowType), LINK(rowAddr));
  2226. args.append(*createTranslated(castBound));
  2227. }
  2228. OwnedHqlExpr call = translator.bindFunctionCall(addAggregateRowId, args);
  2229. translator.buildStmt(transformCtx, call);
  2230. }
  2231. void SourceBuilder::gatherVirtualFields(bool ignoreVirtuals, bool ensureSerialized)
  2232. {
  2233. IHqlExpression * record = tableExpr->queryRecord();
  2234. fieldInfo.gatherVirtualFields(record, ignoreVirtuals, ensureSerialized);
  2235. if (fieldInfo.hasVirtuals())
  2236. physicalRecord.setown(fieldInfo.createPhysicalRecord());
  2237. else
  2238. physicalRecord.set(record);
  2239. }
  2240. /*
  2241. interface ICompoundSourceSteppingMeta : extends ISteppingMeta
  2242. {
  2243. virtual ISteppingMeta * queryRawSteppingMeta() = 0;
  2244. virtual ISteppingMeta * queryProjectedSteppingMeta() = 0; // if null no projection takes place
  2245. virtual void mapOutputToIInput(void * originalRow, const void * projectedRow, unsigned firstField, unsigned numFields) = 0;
  2246. };
  2247. */
  2248. bool SourceBuilder::containsStepping(IHqlExpression * expr)
  2249. {
  2250. loop
  2251. {
  2252. switch (expr->getOperator())
  2253. {
  2254. case no_stepped:
  2255. return true;
  2256. default:
  2257. {
  2258. if (expr->queryBody() == tableExpr)
  2259. return false;
  2260. unsigned numChildren = getNumChildTables(expr);
  2261. if (numChildren == 0)
  2262. return false;
  2263. assertex(numChildren == 1);
  2264. }
  2265. }
  2266. expr = expr->queryChild(0);
  2267. }
  2268. }
  2269. void SourceBuilder::gatherSteppingMeta(IHqlExpression * expr, SteppingFieldSelection & outputStepping, SteppingFieldSelection & rawStepping)
  2270. {
  2271. loop
  2272. {
  2273. switch (expr->getOperator())
  2274. {
  2275. case no_newusertable:
  2276. case no_hqlproject:
  2277. if (rawStepping.exists())
  2278. {
  2279. rawStepping.expandTransform(expr);
  2280. }
  2281. else
  2282. {
  2283. gatherSteppingMeta(expr->queryChild(0), outputStepping, rawStepping);
  2284. outputStepping.clear();
  2285. return;
  2286. //throwError(HQLERR_CantProjectStepping);
  2287. //gatherSteppingMeta(expr->queryChild(0), outputStepping, rawStepping);
  2288. //apply projection to the output fields and inverse to the raw fields.
  2289. }
  2290. break;
  2291. case no_stepped:
  2292. {
  2293. outputStepping.setStepping(expr);
  2294. rawStepping.setStepping(expr);
  2295. break;
  2296. }
  2297. default:
  2298. {
  2299. if (expr->queryBody() == tableExpr)
  2300. return;
  2301. unsigned numChildren = getNumChildTables(expr);
  2302. if (numChildren == 0)
  2303. return;
  2304. assertex(numChildren == 1);
  2305. }
  2306. }
  2307. expr = expr->queryChild(0);
  2308. }
  2309. }
  2310. void SourceBuilder::gatherSteppingMeta(IHqlExpression * expr, SourceSteppingInfo & info)
  2311. {
  2312. if (!steppedExpr)
  2313. return;
  2314. gatherSteppingMeta(expr, info.outputStepping, info.rawSteppingProject);
  2315. if (info.rawSteppingProject.exists())
  2316. info.extractRaw();
  2317. }
  2318. //-----------------------------------------------------------------------------------------------
  2319. //-- Disk file processing
  2320. //-----------------------------------------------------------------------------------------------
  2321. class DiskReadBuilderBase : public SourceBuilder
  2322. {
  2323. public:
  2324. DiskReadBuilderBase(HqlCppTranslator & _translator, IHqlExpression *_tableExpr, IHqlExpression *_nameExpr)
  2325. : SourceBuilder(_translator, _tableExpr, _nameExpr), monitors(_tableExpr, _translator, 0, true)
  2326. {
  2327. fpos.setown(getFilepos(tableExpr, false));
  2328. lfpos.setown(getFilepos(tableExpr, true));
  2329. logicalFilenameMarker.setown(getFileLogicalName(tableExpr));
  2330. mode = tableExpr->queryChild(2);
  2331. modeOp = mode->getOperator();
  2332. includeFormatCrc = (modeOp != no_csv);
  2333. }
  2334. virtual void buildMembers(IHqlExpression * expr);
  2335. virtual void buildTransformFpos(BuildCtx & transformCtx);
  2336. virtual void extractMonitors(IHqlExpression * ds, SharedHqlExpr & unkeyedFilter, HqlExprArray & conds);
  2337. protected:
  2338. virtual void buildFlagsMember(IHqlExpression * expr);
  2339. protected:
  2340. MonitorExtractor monitors;
  2341. OwnedHqlExpr globalGuard;
  2342. IHqlExpression * mode;
  2343. node_operator modeOp;
  2344. bool includeFormatCrc;
  2345. };
  2346. void DiskReadBuilderBase::buildMembers(IHqlExpression * expr)
  2347. {
  2348. StringBuffer s;
  2349. //Process any KEYED() information
  2350. if (monitors.isFiltered())
  2351. {
  2352. BuildCtx createSegmentCtx(instance->startctx);
  2353. createSegmentCtx.addQuotedCompoundLiteral("virtual void createSegmentMonitors(IIndexReadContext *irc)");
  2354. monitors.buildSegments(createSegmentCtx, "irc", true);
  2355. }
  2356. instance->addAttributeBool("_isKeyed", monitors.isFiltered());
  2357. //---- virtual unsigned getFlags()
  2358. instance->addAttributeBool("preload", isPreloaded);
  2359. bool matched = translator.registerGlobalUsage(tableExpr->queryChild(0));
  2360. if (translator.getTargetClusterType() == RoxieCluster)
  2361. {
  2362. instance->addAttributeBool("_isOpt", tableExpr->hasAttribute(optAtom));
  2363. instance->addAttributeBool("_isSpillGlobal", tableExpr->hasAttribute(jobTempAtom));
  2364. if (tableExpr->hasAttribute(jobTempAtom) && !matched)
  2365. throwUnexpected();
  2366. }
  2367. if (!matched && expr->hasAttribute(_spill_Atom))
  2368. {
  2369. StringBuffer spillName;
  2370. getExprECL(tableExpr->queryChild(0), spillName);
  2371. if (translator.queryOptions().allowThroughSpill)
  2372. throwError1(HQLERR_ReadSpillBeforeWriteFix, spillName.str());
  2373. else
  2374. throwError1(HQLERR_ReadSpillBeforeWrite, spillName.str());
  2375. }
  2376. //---- virtual bool canMatchAny() { return <value>; } ----
  2377. LinkedHqlExpr guard = globalGuard.get();
  2378. extendConditionOwn(guard, no_and, LINK(monitors.queryGlobalGuard()));
  2379. if (guard)
  2380. translator.doBuildBoolFunction(instance->startctx, "canMatchAny", guard);
  2381. translator.buildEncryptHelper(instance->startctx, tableExpr->queryAttribute(encryptAtom));
  2382. //---- virtual size32_t getPreloadSize() { return <value>; } ----
  2383. if (preloadSize)
  2384. {
  2385. BuildCtx subctx(instance->classctx);
  2386. subctx.addQuotedCompoundLiteral("virtual size32_t getPreloadSize()");
  2387. translator.buildReturn(subctx, preloadSize, sizetType);
  2388. instance->addAttributeInt("_preloadSize", preloadSize->queryValue()->getIntValue());
  2389. }
  2390. if (includeFormatCrc)
  2391. {
  2392. //Spill files can still have virtual attributes in their physical records => remove them.
  2393. OwnedHqlExpr noVirtualRecord = removeVirtualAttributes(physicalRecord);
  2394. translator.buildFormatCrcFunction(instance->classctx, "getFormatCrc", noVirtualRecord, NULL, 0);
  2395. }
  2396. buildLimits(instance->startctx, expr, instance->activityId);
  2397. buildKeyedLimitHelper(expr);
  2398. //Note the helper base class contains code like the following
  2399. //IThorDiskCallback * fpp;");
  2400. //virtual void setCallback(IThorDiskCallback * _tc) { fpp = _tc; }");
  2401. }
  2402. void DiskReadBuilderBase::buildFlagsMember(IHqlExpression * expr)
  2403. {
  2404. StringBuffer flags;
  2405. if (tableExpr->hasAttribute(_spill_Atom)) flags.append("|TDXtemporary");
  2406. if (tableExpr->hasAttribute(jobTempAtom)) flags.append("|TDXjobtemp");
  2407. if (tableExpr->hasAttribute(groupedAtom)) flags.append("|TDXgrouped");
  2408. if (tableExpr->hasAttribute(__compressed__Atom)) flags.append("|TDXcompress");
  2409. if (tableExpr->hasAttribute(unsortedAtom)) flags.append("|TDRunsorted");
  2410. if (tableExpr->hasAttribute(optAtom)) flags.append("|TDRoptional");
  2411. if (tableExpr->hasAttribute(_workflowPersist_Atom)) flags.append("|TDXupdateaccessed");
  2412. if (isPreloaded) flags.append("|TDRpreload");
  2413. if (monitors.isFiltered()) flags.append("|TDRkeyed");
  2414. if (limitExpr)
  2415. {
  2416. if (limitExpr->hasAttribute(onFailAtom))
  2417. flags.append("|TDRlimitcreates");
  2418. else if (limitExpr->hasAttribute(skipAtom))
  2419. flags.append("|TDRlimitskips");
  2420. }
  2421. if (keyedLimitExpr)
  2422. {
  2423. if (keyedLimitExpr->hasAttribute(onFailAtom))
  2424. flags.append("|TDRkeyedlimitcreates|TDRcountkeyedlimit"); // is count correct?
  2425. else if (keyedLimitExpr->hasAttribute(skipAtom))
  2426. flags.append("|TDRkeyedlimitskips|TDRcountkeyedlimit");
  2427. else if (keyedLimitExpr->hasAttribute(countAtom))
  2428. flags.append("|TDRcountkeyedlimit");
  2429. }
  2430. if (onlyExistsAggreate) flags.append("|TDRaggregateexists");
  2431. if (monitorsForGrouping) flags.append("|TDRgroupmonitors");
  2432. if (!nameExpr->isConstant()) flags.append("|TDXvarfilename");
  2433. if (translator.hasDynamicFilename(tableExpr)) flags.append("|TDXdynamicfilename");
  2434. if (isUnfilteredCount) flags.append("|TDRunfilteredcount");
  2435. if (isVirtualLogicalFilenameUsed) flags.append("|TDRfilenamecallback");
  2436. if (requiresOrderedMerge) flags.append("|TDRorderedmerge");
  2437. if (flags.length())
  2438. translator.doBuildUnsignedFunction(instance->classctx, "getFlags", flags.str()+1);
  2439. }
  2440. void DiskReadBuilderBase::buildTransformFpos(BuildCtx & transformCtx)
  2441. {
  2442. if (modeOp == no_csv)
  2443. associateFilePositions(transformCtx, "fpp", "dataSrc[0]");
  2444. else
  2445. associateFilePositions(transformCtx, "fpp", "left");
  2446. }
  2447. void DiskReadBuilderBase::extractMonitors(IHqlExpression * ds, SharedHqlExpr & unkeyedFilter, HqlExprArray & conds)
  2448. {
  2449. ForEachItemIn(i, conds)
  2450. {
  2451. IHqlExpression * filter = &conds.item(i);
  2452. if (isSourceInvariant(ds, filter)) // more actually isSourceInvariant.
  2453. extendConditionOwn(globalGuard, no_and, LINK(filter));
  2454. else
  2455. {
  2456. node_operator op = filter->getOperator();
  2457. switch (op)
  2458. {
  2459. case no_assertkeyed:
  2460. case no_assertwild:
  2461. {
  2462. //MORE: This needs to test that the fields are at fixed offsets, fixed length, and collatable.
  2463. OwnedHqlExpr extraFilter;
  2464. monitors.extractFilters(filter, extraFilter);
  2465. //NB: Even if it is keyed then (part of) the test condition might be duplicated.
  2466. appendFilter(unkeyedFilter, extraFilter);
  2467. break;
  2468. }
  2469. default:
  2470. // Add this condition to the catchall filter
  2471. appendFilter(unkeyedFilter, filter);
  2472. break;
  2473. }
  2474. }
  2475. }
  2476. }
  2477. //---------------------------------------------------------------------------
  2478. class DiskReadBuilder : public DiskReadBuilderBase
  2479. {
  2480. public:
  2481. DiskReadBuilder(HqlCppTranslator & _translator, IHqlExpression *_tableExpr, IHqlExpression *_nameExpr)
  2482. : DiskReadBuilderBase(_translator, _tableExpr, _nameExpr)
  2483. {
  2484. }
  2485. virtual void buildTransform(IHqlExpression * expr);
  2486. virtual void buildMembers(IHqlExpression * expr);
  2487. };
  2488. void DiskReadBuilder::buildMembers(IHqlExpression * expr)
  2489. {
  2490. buildReadMembers(expr);
  2491. DiskReadBuilderBase::buildMembers(expr);
  2492. //---- virtual const char * getPipeProgram() { return "grep"; } ----
  2493. if (modeOp==no_pipe)
  2494. {
  2495. if (expr->hasAttribute(_disallowed_Atom))
  2496. throwError(HQLERR_PipeNotAllowed);
  2497. BuildCtx pipeCtx(instance->startctx);
  2498. pipeCtx.addQuotedCompoundLiteral("virtual const char * getPipeProgram()");
  2499. translator.buildReturn(pipeCtx, mode->queryChild(0), unknownVarStringType);
  2500. IHqlExpression * csvFromPipe = tableExpr->queryAttribute(csvAtom);
  2501. IHqlExpression * xmlFromPipe = tableExpr->queryAttribute(xmlAtom);
  2502. bool usesContents = false;
  2503. if (csvFromPipe)
  2504. {
  2505. if (isValidCsvRecord(tableExpr->queryRecord()))
  2506. {
  2507. StringBuffer csvInstanceName;
  2508. translator.buildCsvReadTransformer(tableExpr, csvInstanceName, csvFromPipe);
  2509. StringBuffer s;
  2510. s.append("virtual ICsvToRowTransformer * queryCsvTransformer() { return &").append(csvInstanceName).append("; }");
  2511. instance->classctx.addQuoted(s);
  2512. }
  2513. else
  2514. {
  2515. throwUnexpected(); // should be caught earlier
  2516. }
  2517. }
  2518. else if (xmlFromPipe)
  2519. {
  2520. translator.doBuildXmlReadMember(*instance, expr, "queryXmlTransformer", usesContents);
  2521. translator.doBuildVarStringFunction(instance->classctx, "getXmlIteratorPath", queryAttributeChild(xmlFromPipe, rowAtom, 0));
  2522. }
  2523. StringBuffer flags;
  2524. if (tableExpr->hasAttribute(groupAtom)) // not supported in parser?
  2525. flags.append("|TPFgroupeachrow");
  2526. if (tableExpr->hasAttribute(optAtom)) // not supported in parser?
  2527. flags.append("|TPFnofail");
  2528. if (csvFromPipe)
  2529. flags.append("|TPFreadcsvfrompipe");
  2530. if (xmlFromPipe)
  2531. flags.append("|TPFreadxmlfrompipe");
  2532. if (usesContents)
  2533. flags.append("|TPFreadusexmlcontents");
  2534. if (flags.length())
  2535. translator.doBuildUnsignedFunction(instance->classctx, "getPipeFlags", flags.str()+1);
  2536. }
  2537. }
  2538. void DiskReadBuilder::buildTransform(IHqlExpression * expr)
  2539. {
  2540. if (modeOp == no_pipe)
  2541. {
  2542. assertex(!(needToCallTransform || transformCanFilter));
  2543. return;
  2544. }
  2545. if (modeOp == no_csv)
  2546. {
  2547. translator.buildCsvParameters(instance->nestedctx, mode, NULL, true);
  2548. BuildCtx funcctx(instance->startctx);
  2549. funcctx.addQuotedCompoundLiteral("virtual size32_t transform(ARowBuilder & crSelf, unsigned * lenSrc, const char * * dataSrc)");
  2550. translator.ensureRowAllocated(funcctx, "crSelf");
  2551. //associateVirtualCallbacks(*this, funcctx, tableExpr);
  2552. buildTransformBody(funcctx, expr, true, false, true);
  2553. rootSelfRow = NULL;
  2554. unsigned maxColumns = countTotalFields(tableExpr->queryRecord(), false);
  2555. translator.doBuildUnsignedFunction(instance->classctx, "getMaxColumns", maxColumns);
  2556. return;
  2557. }
  2558. BuildCtx transformCtx(instance->startctx);
  2559. if (instance->kind == TAKdiskread)
  2560. transformCtx.addQuotedCompoundLiteral("virtual size32_t transform(ARowBuilder & crSelf, const void * _left)");
  2561. else
  2562. transformCtx.addQuotedCompoundLiteral("virtual size32_t transform(ARowBuilder & crSelf, const void * _left, IFilePositionProvider * fpp)");
  2563. translator.ensureRowAllocated(transformCtx, "crSelf");
  2564. transformCtx.addQuotedLiteral("unsigned char * left = (unsigned char *)_left;");
  2565. buildTransformBody(transformCtx, expr, true, false, true);
  2566. }
  2567. //---------------------------------------------------------------------------
  2568. ABoundActivity * HqlCppTranslator::doBuildActivityDiskRead(BuildCtx & ctx, IHqlExpression * expr)
  2569. {
  2570. // assertex(!isGroupedActivity(expr));
  2571. IHqlExpression *tableExpr = queryPhysicalRootTable(expr);
  2572. if (!tableExpr)
  2573. return buildCachedActivity(ctx, expr->queryChild(0)); // Somehow a null appeared.
  2574. HqlExprAttr mode = tableExpr->queryChild(2);
  2575. node_operator modeOp = mode->getOperator();
  2576. bool isPiped = modeOp==no_pipe;
  2577. DiskReadBuilder info(*this, tableExpr, tableExpr->queryChild(0));
  2578. info.gatherVirtualFields(tableExpr->hasAttribute(_noVirtual_Atom) || isPiped, needToSerializeRecord(modeOp));
  2579. unsigned optFlags = (options.foldOptimized ? HOOfold : 0);
  2580. if (!info.fieldInfo.simpleVirtualsAtEnd || info.fieldInfo.requiresDeserialize ||
  2581. (info.recordHasVirtuals() && (modeOp == no_csv || !isSimpleSource(expr))))
  2582. {
  2583. OwnedHqlExpr transformed = buildTableWithoutVirtuals(info.fieldInfo, expr);
  2584. //Need to wrap a possible no_usertable, otherwise the localisation can go wrong.
  2585. if (expr->getOperator() == no_table)
  2586. transformed.setown(createDataset(no_compound_diskread, LINK(transformed)));
  2587. OwnedHqlExpr optimized = optimizeHqlExpression(queryErrorProcessor(), transformed, optFlags);
  2588. traceExpression("after disk optimize", optimized);
  2589. return doBuildActivityDiskRead(ctx, optimized);
  2590. }
  2591. OwnedHqlExpr optimized;
  2592. if (expr->getOperator() == no_table)
  2593. optimized.set(expr);
  2594. else
  2595. optimized.setown(optimizeHqlExpression(queryErrorProcessor(), expr, optFlags));
  2596. if (optimized != expr)
  2597. return buildActivity(ctx, optimized, false);
  2598. if (optimized->getOperator() != no_compound_diskread)
  2599. optimized.setown(createDataset(no_compound_diskread, LINK(optimized)));
  2600. if (isPiped)
  2601. return info.buildActivity(ctx, expr, TAKpiperead, "PipeRead", NULL);
  2602. if (modeOp == no_csv)
  2603. return info.buildActivity(ctx, expr, TAKcsvread, "CsvRead", NULL);
  2604. return info.buildActivity(ctx, expr, TAKdiskread, "DiskRead", NULL);
  2605. }
  2606. //---------------------------------------------------------------------------
  2607. class DiskNormalizeBuilder : public DiskReadBuilderBase
  2608. {
  2609. public:
  2610. DiskNormalizeBuilder(HqlCppTranslator & _translator, IHqlExpression *_tableExpr, IHqlExpression *_nameExpr)
  2611. : DiskReadBuilderBase(_translator, _tableExpr, _nameExpr)
  2612. {
  2613. }
  2614. virtual void buildTransform(IHqlExpression * expr);
  2615. virtual void buildMembers(IHqlExpression * expr);
  2616. protected:
  2617. virtual void analyseGraph(IHqlExpression * expr);
  2618. virtual void processTransformSelect(BuildCtx & ctx, IHqlExpression * expr)
  2619. {
  2620. doBuildNormalizeIterators(ctx, expr, false);
  2621. }
  2622. };
  2623. void DiskNormalizeBuilder::analyseGraph(IHqlExpression * expr)
  2624. {
  2625. DiskReadBuilderBase::analyseGraph(expr);
  2626. needDefaultTransform = (expr->queryNormalizedSelector()->getOperator() == no_select);
  2627. }
  2628. void DiskNormalizeBuilder::buildMembers(IHqlExpression * expr)
  2629. {
  2630. buildFilenameMember();
  2631. DiskReadBuilderBase::buildMembers(expr);
  2632. buildNormalizeHelpers(expr);
  2633. }
  2634. void DiskNormalizeBuilder::buildTransform(IHqlExpression * expr)
  2635. {
  2636. globaliterctx.setown(new BuildCtx(instance->startctx));
  2637. globaliterctx->addGroup();
  2638. BuildCtx transformCtx(instance->startctx);
  2639. transformCtx.addQuotedCompoundLiteral("virtual size32_t transform(ARowBuilder & crSelf)");
  2640. translator.ensureRowAllocated(transformCtx, "crSelf");
  2641. buildTransformBody(transformCtx, expr, true, false, false);
  2642. }
  2643. //---------------------------------------------------------------------------
  2644. ABoundActivity * HqlCppTranslator::doBuildActivityDiskNormalize(BuildCtx & ctx, IHqlExpression * expr)
  2645. {
  2646. assertex(!isGroupedActivity(expr));
  2647. IHqlExpression *tableExpr = queryPhysicalRootTable(expr);
  2648. HqlExprAttr mode = tableExpr->queryChild(2);
  2649. assertex(mode->getOperator()!=no_pipe);
  2650. DiskNormalizeBuilder info(*this, tableExpr, tableExpr->queryChild(0));
  2651. info.gatherVirtualFields(tableExpr->hasAttribute(_noVirtual_Atom), needToSerializeRecord(mode));
  2652. LinkedHqlExpr transformed = expr;
  2653. if (info.recordHasVirtualsOrDeserialize())
  2654. transformed.setown(buildTableWithoutVirtuals(info.fieldInfo, expr));
  2655. unsigned optFlags = (options.foldOptimized ? HOOfold : 0);
  2656. OwnedHqlExpr optimized = optimizeHqlExpression(queryErrorProcessor(), transformed, optFlags);
  2657. if (optimized != expr)
  2658. return buildActivity(ctx, optimized, false);
  2659. return info.buildActivity(ctx, expr, TAKdisknormalize, "DiskNormalize", NULL);
  2660. }
  2661. //---------------------------------------------------------------------------
  2662. class DiskAggregateBuilder : public DiskReadBuilderBase
  2663. {
  2664. public:
  2665. DiskAggregateBuilder(HqlCppTranslator & _translator, IHqlExpression *_tableExpr, IHqlExpression *_nameExpr)
  2666. : DiskReadBuilderBase(_translator, _tableExpr, _nameExpr)
  2667. {
  2668. failedFilterValue.clear();
  2669. }
  2670. virtual void buildTransform(IHqlExpression * expr);
  2671. virtual void buildMembers(IHqlExpression * expr);
  2672. protected:
  2673. virtual void processTransformSelect(BuildCtx & ctx, IHqlExpression * expr)
  2674. {
  2675. doBuildAggregateSelectIterator(ctx, expr);
  2676. }
  2677. virtual void analyseGraph(IHqlExpression * expr)
  2678. {
  2679. DiskReadBuilderBase::analyseGraph(expr);
  2680. returnIfFilterFails = !isNormalize;
  2681. }
  2682. };
  2683. void DiskAggregateBuilder::buildMembers(IHqlExpression * expr)
  2684. {
  2685. StringBuffer s;
  2686. buildFilenameMember();
  2687. DiskReadBuilderBase::buildMembers(expr);
  2688. buildAggregateHelpers(expr);
  2689. //virtual void processRow(void * self, const void * src) = 0;
  2690. BuildCtx rowctx(instance->startctx);
  2691. rowctx.addQuotedCompoundLiteral("virtual void processRow(ARowBuilder & crSelf, const void * src)");
  2692. rowctx.addQuotedLiteral("doProcessRow(crSelf, (byte *)src);");
  2693. //virtual void processRows(void * self, size32_t srcLen, const void * src) = 0;
  2694. BuildCtx rowsctx(instance->startctx);
  2695. rowsctx.addQuotedCompoundLiteral("virtual void processRows(ARowBuilder & crSelf, size32_t srcLen, const void * _left)");
  2696. rowsctx.addQuotedLiteral("unsigned char * left = (unsigned char *)_left;");
  2697. OwnedHqlExpr ds = createVariable("left", makeReferenceModifier(tableExpr->getType()));
  2698. OwnedHqlExpr len = createVariable("srcLen", LINK(sizetType));
  2699. OwnedHqlExpr fullDs = createTranslated(ds, len);
  2700. Owned<IHqlCppDatasetCursor> iter = translator.createDatasetSelector(rowsctx, fullDs);
  2701. BoundRow * curRow = iter->buildIterateLoop(rowsctx, false);
  2702. s.clear().append("doProcessRow(crSelf, ");
  2703. translator.generateExprCpp(s, curRow->queryBound());
  2704. s.append(");");
  2705. rowsctx.addQuoted(s);
  2706. }
  2707. void DiskAggregateBuilder::buildTransform(IHqlExpression * expr)
  2708. {
  2709. BuildCtx transformCtx(instance->startctx);
  2710. transformCtx.addQuotedCompoundLiteral("void doProcessRow(ARowBuilder & crSelf, byte * left)");
  2711. translator.ensureRowAllocated(transformCtx, "crSelf");
  2712. buildTransformBody(transformCtx, expr, false, false, true);
  2713. }
  2714. //---------------------------------------------------------------------------
  2715. class DiskCountBuilder : public DiskReadBuilderBase
  2716. {
  2717. public:
  2718. DiskCountBuilder(HqlCppTranslator & _translator, IHqlExpression *_tableExpr, IHqlExpression *_nameExpr, node_operator _aggOp)
  2719. : DiskReadBuilderBase(_translator, _tableExpr, _nameExpr)
  2720. {
  2721. aggOp = _aggOp;
  2722. isCompoundCount = true;
  2723. failedFilterValue.set(queryZero());
  2724. }
  2725. virtual void buildTransform(IHqlExpression * expr);
  2726. virtual void buildMembers(IHqlExpression * expr);
  2727. virtual bool isExplicitExists() { return (aggOp == no_existsgroup); }
  2728. protected:
  2729. virtual void processTransformSelect(BuildCtx & ctx, IHqlExpression * expr)
  2730. {
  2731. doBuildAggregateSelectIterator(ctx, expr);
  2732. }
  2733. virtual void analyseGraph(IHqlExpression * expr)
  2734. {
  2735. DiskReadBuilderBase::analyseGraph(expr);
  2736. returnIfFilterFails = !isNormalize;
  2737. if (aggOp == no_existsgroup)
  2738. choosenValue.setown(getSizetConstant(1));
  2739. }
  2740. protected:
  2741. node_operator aggOp;
  2742. };
  2743. void DiskCountBuilder::buildMembers(IHqlExpression * expr)
  2744. {
  2745. isUnfilteredCount = !(transformCanFilter||isNormalize);
  2746. buildFilenameMember();
  2747. DiskReadBuilderBase::buildMembers(expr);
  2748. buildCountHelpers(expr, true);
  2749. }
  2750. void DiskCountBuilder::buildTransform(IHqlExpression * expr)
  2751. {
  2752. if (transformCanFilter||isNormalize)
  2753. {
  2754. BuildCtx transformCtx(instance->startctx);
  2755. transformCtx.addQuotedCompoundLiteral("size32_t valid(byte * _left)");
  2756. transformCtx.addQuotedLiteral("unsigned char * left = (unsigned char *)_left;");
  2757. OwnedHqlExpr cnt;
  2758. if (isNormalize)
  2759. {
  2760. compoundCountVar.setown(transformCtx.getTempDeclare(sizetType, queryZero()));
  2761. cnt.set(compoundCountVar);
  2762. }
  2763. else
  2764. cnt.setown(getSizetConstant(1));
  2765. BuildCtx subctx(transformCtx);
  2766. buildTransformBody(subctx, expr, false, false, true);
  2767. transformCtx.addReturn(cnt);
  2768. }
  2769. }
  2770. //---------------------------------------------------------------------------
  2771. ABoundActivity * HqlCppTranslator::doBuildActivityDiskAggregate(BuildCtx & ctx, IHqlExpression * expr)
  2772. {
  2773. assertex(!isGroupedActivity(expr));
  2774. IHqlExpression *tableExpr = queryPhysicalRootTable(expr);
  2775. HqlExprAttr mode = tableExpr->queryChild(2);
  2776. assertex(mode->getOperator()!=no_pipe);
  2777. DiskAggregateBuilder info(*this, tableExpr, tableExpr->queryChild(0));
  2778. info.gatherVirtualFields(tableExpr->hasAttribute(_noVirtual_Atom), needToSerializeRecord(mode));
  2779. LinkedHqlExpr transformed = expr;
  2780. if (info.recordHasVirtualsOrDeserialize())
  2781. transformed.setown(buildTableWithoutVirtuals(info.fieldInfo, expr));
  2782. transformed.setown(optimizeHqlExpression(queryErrorProcessor(), transformed, getSourceAggregateOptimizeFlags()));
  2783. if (transformed != expr)
  2784. return buildActivity(ctx, transformed, false);
  2785. node_operator aggOp = querySimpleAggregate(expr, true, false);
  2786. if (aggOp == no_countgroup || aggOp == no_existsgroup)
  2787. {
  2788. DiskCountBuilder info(*this, tableExpr, tableExpr->queryChild(0), aggOp);
  2789. info.gatherVirtualFields(tableExpr->hasAttribute(_noVirtual_Atom), needToSerializeRecord(mode));
  2790. return info.buildActivity(ctx, expr, TAKdiskcount, "DiskCount", NULL);
  2791. }
  2792. else
  2793. return info.buildActivity(ctx, expr, TAKdiskaggregate, "DiskAggregate", NULL);
  2794. }
  2795. //---------------------------------------------------------------------------
  2796. class DiskGroupAggregateBuilder : public DiskReadBuilderBase
  2797. {
  2798. public:
  2799. DiskGroupAggregateBuilder(HqlCppTranslator & _translator, IHqlExpression *_tableExpr, IHqlExpression *_nameExpr)
  2800. : DiskReadBuilderBase(_translator, _tableExpr, _nameExpr)
  2801. {
  2802. failedFilterValue.clear();
  2803. }
  2804. virtual void buildTransform(IHqlExpression * expr);
  2805. virtual void buildMembers(IHqlExpression * expr);
  2806. protected:
  2807. virtual void processTransformSelect(BuildCtx & ctx, IHqlExpression * expr)
  2808. {
  2809. doBuildAggregateSelectIterator(ctx, expr);
  2810. }
  2811. virtual void analyseGraph(IHqlExpression * expr)
  2812. {
  2813. DiskReadBuilderBase::analyseGraph(expr);
  2814. returnIfFilterFails = !isNormalize;
  2815. }
  2816. };
  2817. void DiskGroupAggregateBuilder::buildMembers(IHqlExpression * expr)
  2818. {
  2819. buildFilenameMember();
  2820. buildGroupingMonitors(expr, monitors);
  2821. DiskReadBuilderBase::buildMembers(expr);
  2822. buildGlobalGroupAggregateHelpers(expr);
  2823. }
  2824. void DiskGroupAggregateBuilder::buildTransform(IHqlExpression * expr)
  2825. {
  2826. BuildCtx transformCtx(instance->startctx);
  2827. transformCtx.addQuotedCompoundLiteral("void doProcessRow(byte * left, IHThorGroupAggregateCallback * callback)");
  2828. bool accessesCallback = containsOperator(expr, no_filepos) || containsOperator(expr, no_file_logicalname);
  2829. buildGroupAggregateTransformBody(transformCtx, expr, isNormalize || accessesCallback, true);
  2830. }
  2831. //---------------------------------------------------------------------------
  2832. ABoundActivity * HqlCppTranslator::doBuildActivityDiskGroupAggregate(BuildCtx & ctx, IHqlExpression * expr)
  2833. {
  2834. IHqlExpression *tableExpr = queryPhysicalRootTable(expr);
  2835. HqlExprAttr mode = tableExpr->queryChild(2);
  2836. assertex(mode->getOperator()!=no_pipe);
  2837. DiskGroupAggregateBuilder info(*this, tableExpr, tableExpr->queryChild(0));
  2838. info.gatherVirtualFields(tableExpr->hasAttribute(_noVirtual_Atom), needToSerializeRecord(mode));
  2839. LinkedHqlExpr transformed = expr;
  2840. if (info.recordHasVirtualsOrDeserialize())
  2841. transformed.setown(buildTableWithoutVirtuals(info.fieldInfo, expr));
  2842. transformed.setown(optimizeHqlExpression(queryErrorProcessor(), transformed, getSourceAggregateOptimizeFlags()));
  2843. if (transformed != expr)
  2844. return buildActivity(ctx, transformed, false);
  2845. return info.buildActivity(ctx, expr, TAKdiskgroupaggregate, "DiskGroupAggregate", NULL);
  2846. }
  2847. //-----------------------------------------------------------------------------------------------
  2848. //-- Child dataset processing
  2849. //-----------------------------------------------------------------------------------------------
  2850. class ChildBuilderBase : public SourceBuilder
  2851. {
  2852. public:
  2853. ChildBuilderBase(HqlCppTranslator & _translator, IHqlExpression *_tableExpr, IHqlExpression *_nameExpr)
  2854. : SourceBuilder(_translator, _tableExpr, _nameExpr)
  2855. {
  2856. }
  2857. virtual void buildMembers(IHqlExpression * expr) {}
  2858. virtual void buildTransformFpos(BuildCtx & transformCtx) {}
  2859. };
  2860. class ChildNormalizeBuilder : public ChildBuilderBase
  2861. {
  2862. public:
  2863. ChildNormalizeBuilder(HqlCppTranslator & _translator, IHqlExpression *_tableExpr, IHqlExpression *_nameExpr)
  2864. : ChildBuilderBase(_translator, _tableExpr, _nameExpr)
  2865. {
  2866. }
  2867. virtual void buildTransform(IHqlExpression * expr);
  2868. virtual void buildMembers(IHqlExpression * expr);
  2869. protected:
  2870. virtual void analyseGraph(IHqlExpression * expr);
  2871. virtual void processTransformSelect(BuildCtx & ctx, IHqlExpression * expr)
  2872. {
  2873. doBuildNormalizeIterators(ctx, expr, true);
  2874. }
  2875. };
  2876. void ChildNormalizeBuilder::analyseGraph(IHqlExpression * expr)
  2877. {
  2878. ChildBuilderBase::analyseGraph(expr);
  2879. needDefaultTransform = (expr->queryNormalizedSelector()->getOperator() == no_select);
  2880. }
  2881. void ChildNormalizeBuilder::buildMembers(IHqlExpression * expr)
  2882. {
  2883. ChildBuilderBase::buildMembers(expr);
  2884. buildNormalizeHelpers(expr);
  2885. }
  2886. void ChildNormalizeBuilder::buildTransform(IHqlExpression * expr)
  2887. {
  2888. globaliterctx.setown(new BuildCtx(instance->startctx));
  2889. globaliterctx->addGroup();
  2890. BuildCtx transformCtx(instance->startctx);
  2891. transformCtx.addQuotedCompoundLiteral("virtual size32_t transform(ARowBuilder & crSelf)");
  2892. translator.ensureRowAllocated(transformCtx, "crSelf");
  2893. buildTransformBody(transformCtx, expr, true, false, false);
  2894. }
  2895. //---------------------------------------------------------------------------
  2896. ABoundActivity * HqlCppTranslator::doBuildActivityChildNormalize(BuildCtx & ctx, IHqlExpression * expr)
  2897. {
  2898. ChildNormalizeBuilder info(*this, NULL, NULL);
  2899. OwnedHqlExpr optimized = optimizeHqlExpression(queryErrorProcessor(), expr, HOOfold);
  2900. if (optimized != expr)
  2901. return buildActivity(ctx, optimized, false);
  2902. return info.buildActivity(ctx, expr, TAKchildnormalize, "ChildNormalize", NULL);
  2903. }
  2904. //---------------------------------------------------------------------------
  2905. class ChildAggregateBuilder : public ChildBuilderBase
  2906. {
  2907. public:
  2908. ChildAggregateBuilder(HqlCppTranslator & _translator, IHqlExpression *_tableExpr, IHqlExpression *_nameExpr)
  2909. : ChildBuilderBase(_translator, _tableExpr, _nameExpr)
  2910. {
  2911. failedFilterValue.clear();
  2912. }
  2913. virtual void buildTransform(IHqlExpression * expr);
  2914. virtual void buildMembers(IHqlExpression * expr);
  2915. protected:
  2916. virtual void processTransformSelect(BuildCtx & ctx, IHqlExpression * expr)
  2917. {
  2918. doBuildAggregateSelectIterator(ctx, expr);
  2919. }
  2920. virtual void analyseGraph(IHqlExpression * expr)
  2921. {
  2922. ChildBuilderBase::analyseGraph(expr);
  2923. returnIfFilterFails = false;
  2924. }
  2925. };
  2926. void ChildAggregateBuilder::buildMembers(IHqlExpression * expr)
  2927. {
  2928. ChildBuilderBase::buildMembers(expr);
  2929. buildAggregateHelpers(expr);
  2930. }
  2931. void ChildAggregateBuilder::buildTransform(IHqlExpression * expr)
  2932. {
  2933. BuildCtx transformCtx(instance->startctx);
  2934. transformCtx.addQuotedCompoundLiteral("virtual void processRows(ARowBuilder & crSelf)");
  2935. translator.ensureRowAllocated(transformCtx, "crSelf");
  2936. buildTransformBody(transformCtx, expr, false, false, false);
  2937. }
  2938. //---------------------------------------------------------------------------
  2939. ABoundActivity * HqlCppTranslator::doBuildActivityChildAggregate(BuildCtx & ctx, IHqlExpression * expr)
  2940. {
  2941. ChildAggregateBuilder info(*this, NULL, NULL);
  2942. OwnedHqlExpr transformed = optimizeHqlExpression(queryErrorProcessor(), expr, getSourceAggregateOptimizeFlags());
  2943. if (transformed != expr)
  2944. return buildActivity(ctx, transformed, false);
  2945. return info.buildActivity(ctx, expr, TAKchildaggregate, "ChildAggregate", NULL);
  2946. }
  2947. //---------------------------------------------------------------------------
  2948. class ChildGroupAggregateBuilder : public ChildBuilderBase
  2949. {
  2950. public:
  2951. ChildGroupAggregateBuilder(HqlCppTranslator & _translator, IHqlExpression *_tableExpr, IHqlExpression *_nameExpr)
  2952. : ChildBuilderBase(_translator, _tableExpr, _nameExpr)
  2953. {
  2954. failedFilterValue.clear();
  2955. }
  2956. virtual void buildTransform(IHqlExpression * expr);
  2957. virtual void buildMembers(IHqlExpression * expr);
  2958. protected:
  2959. virtual void processTransformSelect(BuildCtx & ctx, IHqlExpression * expr)
  2960. {
  2961. doBuildAggregateSelectIterator(ctx, expr);
  2962. }
  2963. virtual void analyseGraph(IHqlExpression * expr)
  2964. {
  2965. ChildBuilderBase::analyseGraph(expr);
  2966. returnIfFilterFails = false;
  2967. }
  2968. };
  2969. void ChildGroupAggregateBuilder::buildMembers(IHqlExpression * expr)
  2970. {
  2971. ChildBuilderBase::buildMembers(expr);
  2972. IHqlExpression * aggregate = expr->queryChild(0);
  2973. assertex(aggregate->getOperator() == no_newaggregate);
  2974. StringBuffer s;
  2975. //virtual size32_t clearAggregate(void * self) = 0;
  2976. translator.doBuildAggregateClearFunc(instance->startctx, aggregate);
  2977. //virtual size32_t mergeAggregate(ARowBuilder & crSelf, const void * src) - never actually called.
  2978. instance->startctx.addQuotedLiteral("virtual size32_t mergeAggregate(ARowBuilder & crSelf, const void * src) { return 0; }");
  2979. }
  2980. void ChildGroupAggregateBuilder::buildTransform(IHqlExpression * expr)
  2981. {
  2982. BuildCtx transformCtx(instance->startctx);
  2983. transformCtx.addQuotedCompoundLiteral("void processRows(IHThorGroupAggregateCallback * callback)");
  2984. buildGroupAggregateTransformBody(transformCtx, expr, true, false);
  2985. }
  2986. //---------------------------------------------------------------------------
  2987. ABoundActivity * HqlCppTranslator::doBuildActivityChildGroupAggregate(BuildCtx & ctx, IHqlExpression * expr)
  2988. {
  2989. ChildGroupAggregateBuilder info(*this, NULL, NULL);
  2990. OwnedHqlExpr transformed = optimizeHqlExpression(queryErrorProcessor(), expr, getSourceAggregateOptimizeFlags());
  2991. if (transformed != expr)
  2992. return buildActivity(ctx, transformed, false);
  2993. return info.buildActivity(ctx, expr, TAKchildgroupaggregate, "ChildGroupAggregate", NULL);
  2994. }
  2995. //---------------------------------------------------------------------------
  2996. class ChildThroughNormalizeBuilder : public ChildBuilderBase
  2997. {
  2998. public:
  2999. ChildThroughNormalizeBuilder(HqlCppTranslator & _translator, IHqlExpression *_tableExpr, IHqlExpression *_nameExpr)
  3000. : ChildBuilderBase(_translator, _tableExpr, _nameExpr)
  3001. {
  3002. }
  3003. virtual void buildTransform(IHqlExpression * expr);
  3004. virtual void buildMembers(IHqlExpression * expr);
  3005. protected:
  3006. virtual void analyseGraph(IHqlExpression * expr);
  3007. virtual void processTransformSelect(BuildCtx & ctx, IHqlExpression * expr)
  3008. {
  3009. translator.bindTableCursor(ctx, expr, "left");
  3010. doBuildNormalizeIterators(ctx, expr, false);
  3011. }
  3012. };
  3013. void ChildThroughNormalizeBuilder::analyseGraph(IHqlExpression * expr)
  3014. {
  3015. ChildBuilderBase::analyseGraph(expr);
  3016. needDefaultTransform = (expr->queryNormalizedSelector()->getOperator() == no_select);
  3017. }
  3018. void ChildThroughNormalizeBuilder::buildMembers(IHqlExpression * expr)
  3019. {
  3020. ChildBuilderBase::buildMembers(expr);
  3021. buildNormalizeHelpers(expr);
  3022. }
  3023. void ChildThroughNormalizeBuilder::buildTransform(IHqlExpression * expr)
  3024. {
  3025. globaliterctx.setown(new BuildCtx(instance->startctx));
  3026. globaliterctx->addGroup();
  3027. BuildCtx transformCtx(instance->startctx);
  3028. transformCtx.addQuotedCompoundLiteral("virtual size32_t transform(ARowBuilder & crSelf)");
  3029. translator.ensureRowAllocated(transformCtx, "crSelf");
  3030. buildTransformBody(transformCtx, expr, true, false, false);
  3031. }
  3032. ABoundActivity * HqlCppTranslator::doBuildActivityCompoundSelectNew(BuildCtx & ctx, IHqlExpression * expr)
  3033. {
  3034. OwnedHqlExpr optimized = optimizeHqlExpression(queryErrorProcessor(), expr, HOOfold);
  3035. if (optimized->getOperator() == no_null)
  3036. return buildCachedActivity(ctx, optimized);
  3037. IHqlExpression * select = queryRoot(optimized);
  3038. if (!(select && select->getOperator() == no_select && select->hasAttribute(newAtom)))
  3039. {
  3040. if (optimized->getOperator() == no_compound_selectnew)
  3041. return buildCachedActivity(ctx, optimized->queryChild(0));
  3042. return buildCachedActivity(ctx, optimized);
  3043. }
  3044. IHqlExpression * ds = select->queryChild(0);
  3045. Owned<ABoundActivity> childActivity = buildCachedActivity(ctx, ds);
  3046. OwnedHqlExpr fakeDataset = createDataset(no_anon, LINK(ds->queryRecord()));
  3047. OwnedHqlExpr fakeSelect = createNewSelectExpr(LINK(fakeDataset), LINK(select->queryChild(1)));
  3048. OwnedHqlExpr activityExpr = replaceExpression(optimized, select, fakeSelect);
  3049. ChildThroughNormalizeBuilder info(*this, fakeDataset, NULL);
  3050. info.gatherVirtualFields(true, true); // ,false?
  3051. return info.buildActivity(ctx, activityExpr, TAKchildthroughnormalize, "ChildThroughNormalize", childActivity);
  3052. }
  3053. //-----------------------------------------------------------------------------------------------
  3054. //-- Index processing
  3055. //-----------------------------------------------------------------------------------------------
  3056. /*
  3057. Note on generating segment monitors for conditions (ty)x = Y
  3058. 1) casting from Tx to Ty loses information. E.g., (int1)string2field = int1value.
  3059. In this case it is almost impossible to generate a segment monitor because we would need to work out all
  3060. the possibly values for x which could generate the value y.
  3061. The only exception is an inequality (and no_notin), which we can use to remove some of the candidates. The test will always need
  3062. duplicating since we cannot remove all the expected values.
  3063. 2) Casting from Ty to Tx loses information. E.g., (string)string2field = stringvalue
  3064. In this case we can process the filter without prefiltering by testing whether isExact: (Ty)(Tx)Y == Y, and following the following rules:
  3065. a) no_eq. If isExact, add value else nothing.
  3066. b) no_ne. If isExtact, remove value else nothing.
  3067. c) no_gt. Always add > value
  3068. d) no_ge. If isExact add >= value else add > value
  3069. e) no_lt. If isExact add < value else add <= value
  3070. f) no_le. Always add <= value
  3071. 3) Note casts must be present on both sides to indicate exactly what type the comparison will be done as. This includes (string)which would
  3072. normally be missing if the field was of type string<n>.
  3073. */
  3074. static node_operator getModifiedOp(node_operator op, bool duplicate)
  3075. {
  3076. if (!duplicate)
  3077. return op;
  3078. switch (op)
  3079. {
  3080. case no_eq:
  3081. case no_le:
  3082. case no_ge:
  3083. case no_in:
  3084. return op;
  3085. //err on the side of caution for the segment monitors -
  3086. //the test later which check it more accurately.
  3087. case no_gt:
  3088. return no_ge;
  3089. case no_lt:
  3090. return no_le;
  3091. case no_ne:
  3092. case no_notin:
  3093. return no_none;
  3094. default:
  3095. UNIMPLEMENTED;
  3096. }
  3097. }
  3098. void KeyFailureInfo::merge(const KeyFailureInfo & other)
  3099. {
  3100. if (code < other.code)
  3101. set(other.code, other.field);
  3102. }
  3103. void KeyFailureInfo::reportError(HqlCppTranslator & translator, IHqlExpression * condition)
  3104. {
  3105. StringBuffer ecl;
  3106. getExprECL(condition, ecl);
  3107. switch (code)
  3108. {
  3109. case KFRunknown:
  3110. translator.throwError1(HQLERR_KeyedJoinTooComplex, ecl.str());
  3111. case KFRnokey:
  3112. translator.throwError1(HQLERR_KeyAccessNoKeyField, ecl.str());
  3113. case KFRtoocomplex:
  3114. translator.throwError1(HQLERR_KeyedJoinTooComplex, ecl.str());
  3115. case KFRcast:
  3116. translator.throwError2(HQLERR_KeyAccessNeedCast, ecl.str(), str(field->queryName()));
  3117. case KFRor:
  3118. translator.throwError1(HQLERR_OrMultipleKeyfields, ecl.str());
  3119. }
  3120. }
  3121. const char * BuildMonitorState::getSetName()
  3122. {
  3123. if (!setNames.isItem(numActiveSets))
  3124. {
  3125. StringBuffer name;
  3126. getUniqueId(name.append("set"));
  3127. setNames.append(*new StringAttrItem(name.str()));
  3128. StringBuffer s;
  3129. funcctx.setNextConstructor();
  3130. funcctx.addQuoted(s.append("Owned<IStringSet> ").append(name).append(";"));
  3131. }
  3132. return setNames.item(numActiveSets++).text;
  3133. }
  3134. void BuildMonitorState::popSetName()
  3135. {
  3136. numActiveSets--;
  3137. }
  3138. IHqlExpression * KeyConditionInfo::createConjunction()
  3139. {
  3140. LinkedHqlExpr result = preFilter;
  3141. ForEachItemIn(i, conditions)
  3142. extendAndCondition(result, conditions.item(i).expr);
  3143. extendAndCondition(result, postFilter);
  3144. return result.getClear();
  3145. }
  3146. MonitorExtractor::MonitorExtractor(IHqlExpression * _tableExpr, HqlCppTranslator & _translator, int _numKeyableFields, bool _allowTranslatedConds) : translator(_translator)
  3147. {
  3148. tableExpr = _tableExpr;
  3149. allowTranslatedConds = _allowTranslatedConds;
  3150. if (_numKeyableFields <= 0)
  3151. {
  3152. //-ve number means remove a certain number of fields from the record
  3153. IHqlExpression * record = tableExpr->queryRecord();
  3154. numKeyableFields = 0;
  3155. ForEachChild(i, record)
  3156. if (!record->queryChild(i)->isAttribute())
  3157. numKeyableFields++;
  3158. numKeyableFields += _numKeyableFields; // remove payload fields.
  3159. }
  3160. else
  3161. numKeyableFields = (unsigned)_numKeyableFields;
  3162. onlyHozedCompares = !allowTranslatedConds;
  3163. expandKeyableFields();
  3164. cleanlyKeyedExplicitly = false;
  3165. keyedExplicitly = false;
  3166. allowDynamicFormatChange = tableExpr && !tableExpr->hasAttribute(fixedAtom);
  3167. }
  3168. void MonitorExtractor::callAddAll(BuildCtx & ctx, IHqlExpression * targetVar)
  3169. {
  3170. HqlExprArray args;
  3171. args.append(*LINK(targetVar));
  3172. translator.callProcedure(ctx, addAllId, args);
  3173. }
  3174. static IHqlExpression * createExpandedRecord(IHqlExpression * expr);
  3175. static ITypeInfo * getExpandedFieldType(ITypeInfo * type, IHqlExpression * expr)
  3176. {
  3177. Linked<ITypeInfo> expandedType = type;
  3178. if (type->getSize() == UNKNOWN_LENGTH)
  3179. expandedType.clear();
  3180. switch (type->getTypeCode())
  3181. {
  3182. case type_packedint:
  3183. expandedType.setown(makeIntType(type->queryPromotedType()->getSize(), type->isSigned()));
  3184. break;
  3185. case type_bitfield:
  3186. expandedType.set(type->queryPromotedType());
  3187. break;
  3188. case type_varstring:
  3189. case type_varunicode:
  3190. #if 0
  3191. if (type->getSize() != UNKNOWN_LENGTH)
  3192. {
  3193. unsigned len = type->getStringLen();
  3194. switch (type->getTypeCode())
  3195. {
  3196. case type_varstring:
  3197. expandedType.setown(makeStringType(len, LINK(type->queryCharset()), LINK(type->queryCollation())));
  3198. break;
  3199. case type_varunicode:
  3200. expandedType.setown(makeUnicodeType(len, type->queryLocale()));
  3201. break;
  3202. }
  3203. break;
  3204. }
  3205. #endif //fall through
  3206. case type_data:
  3207. case type_qstring:
  3208. case type_string:
  3209. case type_unicode:
  3210. case type_utf8:
  3211. if (type->getSize() == UNKNOWN_LENGTH)
  3212. {
  3213. unsigned maxLength = UNKNOWN_LENGTH;
  3214. IHqlExpression * maxSizeExpr = expr ? queryAttributeChild(expr, maxSizeAtom, 0) : NULL;
  3215. if (maxSizeExpr)
  3216. {
  3217. unsigned maxSize = (unsigned)maxSizeExpr->queryValue()->getIntValue();
  3218. switch (type->getTypeCode())
  3219. {
  3220. case type_data:
  3221. case type_string:
  3222. maxLength = maxSize - sizeof(size32_t);
  3223. break;
  3224. case type_qstring:
  3225. maxLength = rtlQStrLength(maxSize - sizeof(size32_t));
  3226. break;
  3227. case type_unicode:
  3228. maxLength = (maxSize-sizeof(size32_t))/sizeof(UChar);
  3229. break;
  3230. case type_utf8:
  3231. maxLength = (maxSize-sizeof(size32_t))/4;
  3232. break;
  3233. case type_varstring:
  3234. maxLength = maxSize - 1;
  3235. break;
  3236. case type_varunicode:
  3237. maxLength = (maxSize/sizeof(UChar)) - 1;
  3238. break;
  3239. }
  3240. }
  3241. else
  3242. {
  3243. IHqlExpression * maxLengthExpr = expr ? queryAttributeChild(expr, maxLengthAtom, 0) : NULL;
  3244. if (maxLengthExpr)
  3245. maxLength = (unsigned)maxLengthExpr->queryValue()->getIntValue();
  3246. }
  3247. if (maxLength != UNKNOWN_LENGTH)
  3248. {
  3249. switch (type->getTypeCode())
  3250. {
  3251. case type_data:
  3252. expandedType.setown(makeDataType(maxLength));
  3253. break;
  3254. case type_qstring:
  3255. expandedType.setown(makeQStringType(maxLength));
  3256. break;
  3257. case type_string:
  3258. expandedType.setown(makeStringType(maxLength, LINK(type->queryCharset()), LINK(type->queryCollation())));
  3259. break;
  3260. case type_unicode:
  3261. expandedType.setown(makeUnicodeType(maxLength, type->queryLocale()));
  3262. break;
  3263. case type_utf8:
  3264. expandedType.setown(makeUtf8Type(maxLength, type->queryLocale()));
  3265. break;
  3266. case type_varstring:
  3267. expandedType.setown(makeVarStringType(maxLength, LINK(type->queryCharset()), LINK(type->queryCollation())));
  3268. break;
  3269. case type_varunicode:
  3270. expandedType.setown(makeVarUnicodeType(maxLength, type->queryLocale()));
  3271. break;
  3272. }
  3273. }
  3274. }
  3275. else
  3276. {
  3277. //This could ensure the strings are ascii, but the ebcdic strings are still comparable, and the order will be more logical
  3278. //if they remain as ebcdic.
  3279. }
  3280. break;
  3281. case type_table:
  3282. case type_groupedtable:
  3283. case type_set:
  3284. expandedType.clear();
  3285. break;
  3286. case type_row:
  3287. {
  3288. OwnedHqlExpr newRecord = createExpandedRecord(queryRecord(type));
  3289. if (isEmptyRecord(newRecord))
  3290. expandedType.clear();
  3291. else
  3292. expandedType.setown(makeRowType(LINK(newRecord->queryRecordType())));
  3293. break;
  3294. }
  3295. case type_alien:
  3296. {
  3297. IHqlAlienTypeInfo * alien = queryAlienType(type);
  3298. expandedType.set(alien->queryLogicalType());
  3299. break;
  3300. }
  3301. }
  3302. return expandedType.getClear();
  3303. }
  3304. static void createExpanded(HqlExprArray & fields, IHqlExpression * expr)
  3305. {
  3306. switch (expr->getOperator())
  3307. {
  3308. case no_ifblock:
  3309. //if blocks need to generate translated segment monitors to be keyed, so don't expand them
  3310. break;
  3311. case no_record:
  3312. {
  3313. ForEachChild(i, expr)
  3314. createExpanded(fields, expr->queryChild(i));
  3315. break;
  3316. }
  3317. case no_field:
  3318. {
  3319. ITypeInfo * type = expr->queryType();
  3320. Owned<ITypeInfo> expandedType = getExpandedFieldType(type, expr);
  3321. if (expandedType)
  3322. {
  3323. if (expandedType == type)
  3324. fields.append(*LINK(expr));
  3325. else
  3326. {
  3327. HqlExprArray attrs;
  3328. unwindChildren(attrs, expr);
  3329. //MORE: Any default will now have the wrong type => remove it for the moment (ideally it would be projected)
  3330. removeAttribute(attrs, defaultAtom);
  3331. fields.append(*createField(expr->queryId(), LINK(expandedType), attrs));
  3332. }
  3333. }
  3334. break;
  3335. }
  3336. case no_attr:
  3337. case no_attr_link:
  3338. case no_attr_expr:
  3339. fields.append(*LINK(expr));
  3340. break;
  3341. }
  3342. }
  3343. static IHqlExpression * createExpandedRecord(IHqlExpression * expr)
  3344. {
  3345. HqlExprArray fields;
  3346. createExpanded(fields, expr);
  3347. return cloneOrLink(expr, fields);
  3348. }
  3349. bool MonitorExtractor::createGroupingMonitor(BuildCtx ctx, const char * listName, IHqlExpression * expr, unsigned & maxOffset)
  3350. {
  3351. switch (expr->getOperator())
  3352. {
  3353. case no_if:
  3354. {
  3355. IHqlExpression * cond = expr->queryChild(0);
  3356. if (expr->queryChild(2)->isConstant() && isIndependentOfScope(cond))
  3357. {
  3358. BuildCtx subctx(ctx);
  3359. translator.buildFilter(subctx, expr->queryChild(0));
  3360. createGroupingMonitor(subctx, listName, expr->queryChild(1), maxOffset);
  3361. return true; // may still be keyed
  3362. }
  3363. break;
  3364. }
  3365. case no_select:
  3366. {
  3367. size32_t offset = 0;
  3368. ForEachItemIn(i, keyableSelects)
  3369. {
  3370. IHqlExpression & cur = keyableSelects.item(i);
  3371. size32_t curSize = cur.queryType()->getSize();
  3372. if (curSize == UNKNOWN_LENGTH)
  3373. break;
  3374. if (expr == &cur)
  3375. {
  3376. //MORE: Check the type of the field is legal.
  3377. ctx.addQuotedF("%s->append(createWildKeySegmentMonitor(%u, %u));", listName, offset, curSize);
  3378. maxOffset = offset+curSize;
  3379. return true;
  3380. }
  3381. offset += curSize;
  3382. }
  3383. break;
  3384. }
  3385. case no_constant:
  3386. return true;
  3387. }
  3388. ctx.addReturn(queryBoolExpr(false));
  3389. return false;
  3390. }
  3391. void MonitorExtractor::expandKeyableFields()
  3392. {
  3393. HqlExprArray fields;
  3394. IHqlExpression * tableRecord = tableExpr->queryRecord();
  3395. unsigned cnt = 0;
  3396. ForEachChild(i, tableRecord)
  3397. {
  3398. if (cnt == numKeyableFields)
  3399. break;
  3400. IHqlExpression * cur = tableRecord->queryChild(i);
  3401. if (!cur->isAttribute())
  3402. {
  3403. fields.append(*LINK(cur));
  3404. cnt++;
  3405. }
  3406. }
  3407. keyableRecord.setown(createRecord(fields));
  3408. expandedRecord.setown(createExpandedRecord(keyableRecord));
  3409. IHqlExpression * selector = tableExpr->queryNormalizedSelector();
  3410. OwnedHqlExpr expandedSelector = createDataset(no_anon, LINK(expandedRecord), createUniqueId());
  3411. firstOffsetField = NotFound;
  3412. expandSelects(keyableRecord, expandedRecord->querySimpleScope(), selector, expandedSelector);
  3413. if (firstOffsetField == NotFound)
  3414. firstOffsetField = keyableSelects.ordinality();
  3415. }
  3416. void MonitorExtractor::expandSelects(IHqlExpression * expr, IHqlSimpleScope * expandedScope, IHqlExpression * keySelector, IHqlExpression * expandedSelector)
  3417. {
  3418. switch (expr->getOperator())
  3419. {
  3420. case no_record:
  3421. {
  3422. ForEachChild(i, expr)
  3423. expandSelects(expr->queryChild(i), expandedScope, keySelector, expandedSelector);
  3424. break;
  3425. }
  3426. case no_ifblock:
  3427. expandSelects(expr->queryChild(1), expandedScope, keySelector, expandedSelector);
  3428. break;
  3429. case no_field:
  3430. {
  3431. OwnedHqlExpr match = expandedScope->lookupSymbol(expr->queryId());
  3432. if (match)
  3433. {
  3434. OwnedHqlExpr keySelected = createSelectExpr(LINK(keySelector), LINK(expr));
  3435. OwnedHqlExpr expandedSelected = createSelectExpr(LINK(expandedSelector), LINK(match));
  3436. IHqlExpression * record = expr->queryRecord();
  3437. if (record)
  3438. expandSelects(record, match->queryRecord()->querySimpleScope(), keySelected, expandedSelected);
  3439. else
  3440. {
  3441. if ((expr != match) && (firstOffsetField == NotFound))
  3442. {
  3443. ITypeInfo * exprType = expr->queryType();
  3444. ITypeInfo * matchType = match->queryType();
  3445. if ((exprType->getSize() != matchType->getSize()) ||
  3446. (exprType->getTypeCode() == type_bitfield || exprType->getTypeCode() == type_bitfield))
  3447. firstOffsetField = keyableSelects.ordinality();
  3448. }
  3449. keyableSelects.append(*LINK(keySelected));
  3450. expandedSelects.append(*LINK(expandedSelected));
  3451. }
  3452. }
  3453. else
  3454. {
  3455. if (firstOffsetField == NotFound)
  3456. firstOffsetField = keyableSelects.ordinality();
  3457. }
  3458. break;
  3459. }
  3460. }
  3461. }
  3462. void MonitorExtractor::buildKeySegmentInExpr(BuildMonitorState & buildState, KeySelectorInfo & selectorInfo, BuildCtx & ctx, const char * target, IHqlExpression & thisKey, MonitorFilterKind filterKind)
  3463. {
  3464. //Generally this slightly increases the code size, but reduces the number of
  3465. //temporary sets which is generally more efficient.
  3466. OwnedHqlExpr simplified = querySimplifyInExpr(&thisKey);
  3467. if (simplified)
  3468. {
  3469. OwnedHqlExpr folded = foldHqlExpression(simplified);
  3470. buildKeySegmentExpr(buildState, selectorInfo, ctx, target, *folded, filterKind);
  3471. return;
  3472. }
  3473. IHqlExpression * expandedSelector = selectorInfo.expandedSelector;
  3474. ITypeInfo * fieldType = expandedSelector->queryType();
  3475. unsigned curSize = fieldType->getSize();
  3476. createStringSet(ctx, target, curSize, fieldType);
  3477. OwnedHqlExpr targetVar = createVariable(target, makeVoidType());
  3478. IHqlExpression * lhs = thisKey.queryChild(0);
  3479. OwnedHqlExpr values = normalizeListCasts(thisKey.queryChild(1));
  3480. IIdAtom * func = addRangeId;
  3481. if (thisKey.getOperator() == no_notin)
  3482. {
  3483. callAddAll(ctx, targetVar);
  3484. func = killRangeId;
  3485. }
  3486. if (values->getOperator() != no_list)
  3487. {
  3488. //iterate through the set
  3489. BuildCtx subctx(ctx);
  3490. CHqlBoundExpr boundCurElement;
  3491. Owned<IHqlCppSetCursor> cursor = translator.createSetSelector(ctx, values);
  3492. bool done = false;
  3493. CHqlBoundExpr isAll;
  3494. cursor->buildIsAll(subctx, isAll);
  3495. if (isAll.expr->queryValue())
  3496. {
  3497. if (isAll.expr->queryValue()->getBoolValue())
  3498. {
  3499. callAddAll(subctx, targetVar);
  3500. done = true;
  3501. //If ALL allowed exceptions then we would need to do more....
  3502. }
  3503. }
  3504. else
  3505. {
  3506. IHqlStmt * stmt = subctx.addFilter(isAll.expr);
  3507. callAddAll(subctx, targetVar);
  3508. subctx.selectElse(stmt);
  3509. }
  3510. if (!done)
  3511. {
  3512. cursor->buildIterateLoop(subctx, boundCurElement, false);
  3513. OwnedHqlExpr curValue = boundCurElement.getTranslatedExpr();
  3514. OwnedHqlExpr test = createBoolExpr(no_eq, LINK(lhs), LINK(curValue));
  3515. OwnedHqlExpr promoted = getExplicitlyPromotedCompare(test);
  3516. OwnedHqlExpr compare, normalized;
  3517. extractCompareInformation(subctx, promoted, compare, normalized, expandedSelector, selectorInfo.isComputed);
  3518. if (compare)
  3519. translator.buildFilter(subctx, compare);
  3520. HqlExprArray args;
  3521. args.append(*LINK(targetVar));
  3522. unsigned srcSize = normalized->queryType()->getSize();
  3523. if (srcSize < curSize)
  3524. {
  3525. OwnedHqlExpr lengthExpr = getSizetConstant(srcSize);
  3526. OwnedHqlExpr rangeLower = getRangeLimit(fieldType, lengthExpr, normalized, -1);
  3527. OwnedHqlExpr rangeUpper = getRangeLimit(fieldType, lengthExpr, normalized, +1);
  3528. CHqlBoundExpr boundLower, boundUpper;
  3529. translator.buildExpr(subctx, rangeLower, boundLower);
  3530. translator.buildExpr(subctx, rangeUpper, boundUpper);
  3531. args.append(*getPointer(boundLower.expr));
  3532. args.append(*getPointer(boundUpper.expr));
  3533. }
  3534. else
  3535. {
  3536. OwnedHqlExpr address = getMonitorValueAddress(subctx, normalized);
  3537. args.append(*LINK(address));
  3538. args.append(*LINK(address));
  3539. }
  3540. translator.callProcedure(subctx, func, args);
  3541. }
  3542. }
  3543. else
  3544. {
  3545. ForEachChild(idx2, values)
  3546. {
  3547. BuildCtx subctx(ctx);
  3548. IHqlExpression * cur = values->queryChild(idx2);
  3549. OwnedHqlExpr test = createBoolExpr(no_eq, LINK(lhs), LINK(cur));
  3550. OwnedHqlExpr promoted = getExplicitlyPromotedCompare(test);
  3551. OwnedHqlExpr compare, normalized;
  3552. extractCompareInformation(subctx, promoted, compare, normalized, expandedSelector, selectorInfo.isComputed);
  3553. if (compare)
  3554. translator.buildFilter(subctx, compare);
  3555. OwnedHqlExpr address = getMonitorValueAddress(subctx, normalized);
  3556. HqlExprArray args;
  3557. args.append(*LINK(targetVar));
  3558. args.append(*LINK(address));
  3559. args.append(*LINK(address));
  3560. translator.callProcedure(subctx, func, args);
  3561. }
  3562. }
  3563. }
  3564. static IHqlExpression * createCompareRecast(node_operator op, IHqlExpression * value, IHqlExpression * recastValue)
  3565. {
  3566. if (recastValue->queryValue())
  3567. return LINK(queryBoolExpr(op == no_ne));
  3568. return createValue(op, makeBoolType(), LINK(value), LINK(recastValue));
  3569. }
  3570. void MonitorExtractor::extractCompareInformation(BuildCtx & ctx, IHqlExpression * expr, SharedHqlExpr & compare, SharedHqlExpr & normalized, IHqlExpression * expandedSelector, bool isComputed)
  3571. {
  3572. extractCompareInformation(ctx, expr->queryChild(0), expr->queryChild(1), compare, normalized, expandedSelector, isComputed);
  3573. }
  3574. void MonitorExtractor::extractCompareInformation(BuildCtx & ctx, IHqlExpression * lhs, IHqlExpression * value, SharedHqlExpr & compare, SharedHqlExpr & normalized, IHqlExpression * expandedSelector, bool isComputed)
  3575. {
  3576. LinkedHqlExpr compareValue = value->queryBody();
  3577. OwnedHqlExpr recastValue;
  3578. if (isComputed)
  3579. normalized.setown(ensureExprType(compareValue, expandedSelector->queryType()));
  3580. else
  3581. {
  3582. if ((lhs->getOperator() != no_select) || (lhs->queryType() != compareValue->queryType()))
  3583. {
  3584. OwnedHqlExpr temp = castToFieldAndBack(lhs, compareValue);
  3585. if (temp != compareValue)
  3586. {
  3587. //Force into a temporary variable since it will be used more than once, and reapply the field casting/
  3588. compareValue.setown(translator.buildSimplifyExpr(ctx, compareValue));
  3589. //cast to promoted type because sometimes evaluating can convert string to string<n>
  3590. Owned<ITypeInfo> promotedType = getPromotedECLType(lhs->queryType(), compareValue->queryType());
  3591. compareValue.setown(ensureExprType(compareValue, promotedType));
  3592. recastValue.setown(castToFieldAndBack(lhs, compareValue));
  3593. }
  3594. }
  3595. normalized.setown(invertTransforms(lhs, compareValue));
  3596. }
  3597. normalized.setown(foldHqlExpression(normalized));
  3598. if (recastValue && recastValue != compareValue)
  3599. compare.setown(createCompareRecast(no_eq, compareValue, recastValue));
  3600. }
  3601. void MonitorExtractor::createStringSet(BuildCtx & ctx, const char * target, unsigned size, ITypeInfo * type)
  3602. {
  3603. if (onlyHozedCompares)
  3604. ctx.addQuotedF("%s.setown(createRtlStringSet(%u));", target, size);
  3605. else
  3606. {
  3607. bool isBigEndian = !type->isInteger() || !isLittleEndian(type);
  3608. ctx.addQuotedF("%s.setown(createRtlStringSetEx(%u,%d,%d));", target, size, isBigEndian, type->isSigned());
  3609. }
  3610. }
  3611. void MonitorExtractor::buildKeySegmentCompareExpr(BuildMonitorState & buildState, KeySelectorInfo & selectorInfo, BuildCtx & ctx, const char * targetSet, IHqlExpression & thisKey)
  3612. {
  3613. OwnedHqlExpr targetVar = createVariable(targetSet, makeVoidType());
  3614. createStringSet(ctx, targetSet, selectorInfo.size, selectorInfo.expandedSelector->queryType());
  3615. if (!exprReferencesDataset(&thisKey, tableExpr))
  3616. {
  3617. BuildCtx subctx(ctx);
  3618. translator.buildFilter(subctx, &thisKey);
  3619. callAddAll(subctx, targetVar);
  3620. return;
  3621. }
  3622. OwnedHqlExpr compare;
  3623. OwnedHqlExpr normalized;
  3624. BuildCtx subctx(ctx);
  3625. extractCompareInformation(subctx, &thisKey, compare, normalized, selectorInfo.expandedSelector, selectorInfo.isComputed);
  3626. OwnedHqlExpr address = getMonitorValueAddress(subctx, normalized);
  3627. HqlExprArray args;
  3628. args.append(*LINK(targetVar));
  3629. node_operator op = thisKey.getOperator();
  3630. switch (op)
  3631. {
  3632. case no_eq:
  3633. if (compare)
  3634. translator.buildFilter(subctx, compare);
  3635. args.append(*LINK(address));
  3636. args.append(*LINK(address));
  3637. translator.callProcedure(subctx, addRangeId, args);
  3638. break;
  3639. case no_ne:
  3640. subctx.addQuoted(StringBuffer().appendf("%s->addAll();", targetSet));
  3641. if (compare)
  3642. translator.buildFilter(subctx, compare);
  3643. args.append(*LINK(address));
  3644. args.append(*LINK(address));
  3645. translator.callProcedure(subctx, killRangeId, args);
  3646. break;
  3647. case no_le:
  3648. args.append(*createValue(no_nullptr, makeVoidType()));
  3649. args.append(*LINK(address));
  3650. translator.callProcedure(subctx, addRangeId, args);
  3651. break;
  3652. case no_lt:
  3653. // e) no_lt. If isExact add < value else add <= value
  3654. if (compare)
  3655. {
  3656. OwnedHqlExpr invCompare = getInverse(compare);
  3657. IHqlStmt * cond = translator.buildFilterViaExpr(subctx, invCompare);
  3658. //common this up...
  3659. args.append(*createValue(no_nullptr, makeVoidType()));
  3660. args.append(*LINK(address));
  3661. translator.callProcedure(subctx, addRangeId, args);
  3662. subctx.selectElse(cond);
  3663. args.append(*LINK(targetVar));
  3664. }
  3665. subctx.addQuoted(StringBuffer().appendf("%s->addAll();", targetSet));
  3666. args.append(*LINK(address));
  3667. args.append(*createValue(no_nullptr, makeVoidType()));
  3668. translator.callProcedure(subctx, killRangeId, args);
  3669. break;
  3670. case no_ge:
  3671. // d) no_ge. If isExact add >= value else add > value
  3672. if (compare)
  3673. {
  3674. OwnedHqlExpr invCompare = getInverse(compare);
  3675. IHqlStmt * cond = translator.buildFilterViaExpr(subctx, invCompare);
  3676. //common this up...
  3677. subctx.addQuoted(StringBuffer().appendf("%s->addAll();", targetSet));
  3678. args.append(*createValue(no_nullptr, makeVoidType()));
  3679. args.append(*LINK(address));
  3680. translator.callProcedure(subctx, killRangeId, args);
  3681. subctx.selectElse(cond);
  3682. args.append(*LINK(targetVar));
  3683. }
  3684. args.append(*LINK(address));
  3685. args.append(*createValue(no_nullptr, makeVoidType()));
  3686. translator.callProcedure(subctx, addRangeId, args);
  3687. break;
  3688. case no_gt:
  3689. subctx.addQuoted(StringBuffer().appendf("%s->addAll();", targetSet));
  3690. args.append(*createValue(no_nullptr, makeVoidType()));
  3691. args.append(*LINK(address));
  3692. translator.callProcedure(subctx, killRangeId, args);
  3693. break;
  3694. case no_between:
  3695. case no_notbetween:
  3696. {
  3697. //NB: This should only be generated for substring queries. User betweens are converted
  3698. //to two separate comparisons to cope with range issues.
  3699. args.append(*LINK(address));
  3700. CHqlBoundExpr rhs2;
  3701. OwnedHqlExpr adjustedUpper = invertTransforms(thisKey.queryChild(0), thisKey.queryChild(2));
  3702. OwnedHqlExpr foldedUpper = foldHqlExpression(adjustedUpper);
  3703. OwnedHqlExpr hozedValue = getHozedKeyValue(foldedUpper);
  3704. IIdAtom * name = hozedValue->queryId();
  3705. if ((name != createRangeHighId) && (name != createQStrRangeHighId))
  3706. hozedValue.setown(ensureExprType(hozedValue, selectorInfo.expandedSelector->queryType()));
  3707. translator.buildExpr(subctx, hozedValue, rhs2);
  3708. translator.ensureHasAddress(subctx, rhs2);
  3709. args.append(*getPointer(rhs2.expr));
  3710. if (op == no_between)
  3711. translator.callProcedure(subctx, addRangeId, args);
  3712. else
  3713. {
  3714. subctx.addQuoted(StringBuffer().appendf("%s->addAll();", targetSet));
  3715. translator.callProcedure(subctx, killRangeId, args);
  3716. }
  3717. break;
  3718. }
  3719. default:
  3720. throwUnexpectedOp(op);
  3721. }
  3722. }
  3723. IHqlExpression * MonitorExtractor::unwindConjunction(HqlExprArray & matches, IHqlExpression * expr)
  3724. {
  3725. node_operator op = expr->getOperator();
  3726. expr->unwindList(matches, op);
  3727. OwnedHqlExpr invariant;
  3728. ForEachItemInRev(i, matches)
  3729. {
  3730. IHqlExpression & cur = matches.item(i);
  3731. if (isIndexInvariant(&cur))
  3732. {
  3733. invariant.setown(extendConditionOwn(op, LINK(&cur), invariant.getClear()));
  3734. matches.remove(i);
  3735. }
  3736. }
  3737. return invariant.getClear();
  3738. }
  3739. //Note this function may change the incoming ctx if filterKind is not NoMonitorFilter
  3740. void MonitorExtractor::buildKeySegmentExpr(BuildMonitorState & buildState, KeySelectorInfo & selectorInfo, BuildCtx & ctx, const char * requiredSet, IHqlExpression & thisKey, MonitorFilterKind filterKind)
  3741. {
  3742. const char * targetSet = requiredSet;
  3743. StringBuffer s;
  3744. unsigned curSize = selectorInfo.size;
  3745. node_operator op = thisKey.getOperator();
  3746. BuildCtx subctx(ctx);
  3747. BuildCtx * appendCtx = &ctx;
  3748. StringBuffer createMonitorText;
  3749. switch (op)
  3750. {
  3751. case no_in:
  3752. case no_notin:
  3753. {
  3754. if (!targetSet)
  3755. targetSet = buildState.getSetName();
  3756. buildKeySegmentInExpr(buildState, selectorInfo, ctx, targetSet, thisKey, filterKind);
  3757. break;
  3758. }
  3759. case no_if:
  3760. {
  3761. MonitorFilterKind childFilter = targetSet ? NoMonitorFilter : filterKind;
  3762. IHqlStmt * ifStmt = translator.buildFilterViaExpr(subctx, thisKey.queryChild(0));
  3763. buildKeySegmentExpr(buildState, selectorInfo, subctx, targetSet, *thisKey.queryChild(1), childFilter);
  3764. subctx.selectElse(ifStmt);
  3765. buildKeySegmentExpr(buildState, selectorInfo, subctx, targetSet, *thisKey.queryChild(2), childFilter);
  3766. break;
  3767. }
  3768. case no_and:
  3769. {
  3770. HqlExprArray matches;
  3771. OwnedHqlExpr invariant = unwindConjunction(matches, &thisKey);
  3772. unsigned numMatches = matches.ordinality();
  3773. if (!targetSet && numMatches > 1)
  3774. targetSet = buildState.getSetName();
  3775. IHqlStmt * ifStmt = NULL;
  3776. if (invariant)
  3777. {
  3778. ifStmt = translator.buildFilterViaExpr(subctx, invariant);
  3779. if (filterKind == MonitorFilterSkipEmpty)
  3780. ctx.set(subctx);
  3781. }
  3782. buildKeySegmentExpr(buildState, selectorInfo, subctx, targetSet, matches.item(0), NoMonitorFilter);
  3783. for (unsigned i=1; i< numMatches; i++)
  3784. {
  3785. IHqlExpression & cur = matches.item(i);
  3786. const char * curTarget = buildState.getSetName();
  3787. BuildCtx childctx(subctx);
  3788. buildKeySegmentExpr(buildState, selectorInfo, childctx, curTarget, cur, MonitorFilterSkipAll);
  3789. childctx.addQuotedF("%s.setown(rtlIntersectSet(%s,%s));", targetSet, targetSet, curTarget);
  3790. buildState.popSetName();
  3791. }
  3792. if (invariant && (filterKind != MonitorFilterSkipEmpty))
  3793. {
  3794. subctx.selectElse(ifStmt);
  3795. if (targetSet)
  3796. createStringSet(subctx, targetSet, curSize, selectorInfo.selector->queryType());
  3797. else
  3798. buildEmptyKeySegment(buildState, subctx, selectorInfo);
  3799. }
  3800. break;
  3801. }
  3802. case no_or:
  3803. {
  3804. HqlExprArray matches;
  3805. OwnedHqlExpr invariant = unwindConjunction(matches, &thisKey);
  3806. unsigned numMatches = matches.ordinality();
  3807. if (invariant)
  3808. {
  3809. if (filterKind == MonitorFilterSkipAll)
  3810. {
  3811. OwnedHqlExpr test = getInverse(invariant);
  3812. translator.buildFilter(subctx, test);
  3813. ctx.set(subctx);
  3814. }
  3815. else
  3816. {
  3817. IHqlStmt * ifStmt = translator.buildFilterViaExpr(subctx, invariant);
  3818. if (targetSet)
  3819. {
  3820. createStringSet(subctx, targetSet, curSize, selectorInfo.selector->queryType());
  3821. OwnedHqlExpr targetVar = createVariable(targetSet, makeVoidType());
  3822. callAddAll(subctx, targetVar);
  3823. }
  3824. else
  3825. buildWildKeySegment(buildState, subctx, selectorInfo);
  3826. subctx.selectElse(ifStmt);
  3827. }
  3828. }
  3829. appendCtx = &subctx;
  3830. if (!targetSet && numMatches > 1)
  3831. targetSet = buildState.getSetName();
  3832. buildKeySegmentExpr(buildState, selectorInfo, subctx, targetSet, matches.item(0), NoMonitorFilter);
  3833. for (unsigned i=1; i < numMatches; i++)
  3834. {
  3835. IHqlExpression & cur = matches.item(i);
  3836. const char * curTarget = buildState.getSetName();
  3837. BuildCtx childctx(subctx);
  3838. buildKeySegmentExpr(buildState, selectorInfo, childctx, curTarget, cur, MonitorFilterSkipEmpty);
  3839. childctx.addQuotedF("%s.setown(rtlUnionSet(%s, %s));", targetSet, targetSet, curTarget);
  3840. buildState.popSetName();
  3841. }
  3842. break;
  3843. }
  3844. case no_eq:
  3845. {
  3846. if (!targetSet)
  3847. {
  3848. if (buildSingleKeyMonitor(createMonitorText, selectorInfo, subctx, thisKey))
  3849. break;
  3850. targetSet = buildState.getSetName();
  3851. }
  3852. buildKeySegmentCompareExpr(buildState, selectorInfo, ctx, targetSet, thisKey);
  3853. break;
  3854. }
  3855. default:
  3856. {
  3857. if (!targetSet)
  3858. targetSet = buildState.getSetName();
  3859. buildKeySegmentCompareExpr(buildState, selectorInfo, ctx, targetSet, thisKey);
  3860. break;
  3861. }
  3862. }
  3863. if (targetSet && !requiredSet)
  3864. {
  3865. unsigned offset = (selectorInfo.isComputed || selectorInfo.mapOffset) ? 0 : selectorInfo.offset;
  3866. createMonitorText.appendf("createKeySegmentMonitor(%s, %s.getClear(), %u, %u)",
  3867. boolToText(selectorInfo.keyedKind != KeyedYes), targetSet, offset, selectorInfo.size);
  3868. buildState.popSetName();
  3869. }
  3870. if (createMonitorText.length())
  3871. {
  3872. if (selectorInfo.expandNeeded || selectorInfo.isComputed)
  3873. generateFormatWrapping(createMonitorText, selectorInfo.selector, selectorInfo.expandedSelector, buildState.curOffset);
  3874. else if (selectorInfo.mapOffset)
  3875. generateOffsetWrapping(createMonitorText, selectorInfo.selector, buildState.curOffset);
  3876. appendCtx->addQuotedF("%s->append(%s);", buildState.listName, createMonitorText.str());
  3877. }
  3878. }
  3879. IHqlExpression * MonitorExtractor::getMonitorValueAddress(BuildCtx & ctx, IHqlExpression * _value)
  3880. {
  3881. LinkedHqlExpr value = _value;
  3882. CHqlBoundExpr bound;
  3883. ITypeInfo * type = value->queryType();
  3884. switch (type->getTypeCode())
  3885. {
  3886. case type_varstring: case type_varunicode:
  3887. {
  3888. assertex(type->getSize() != UNKNOWN_LENGTH);
  3889. CHqlBoundTarget tempTarget;
  3890. translator.createTempFor(ctx, type, tempTarget, typemod_none, FormatNatural);
  3891. //clear the variable.
  3892. HqlExprArray args;
  3893. args.append(*getPointer(tempTarget.expr));
  3894. args.append(*getZero());
  3895. args.append(*getSizetConstant(type->getSize()));
  3896. OwnedHqlExpr call = translator.bindTranslatedFunctionCall(memsetId, args);
  3897. ctx.addExpr(call);
  3898. //then assign over the top
  3899. translator.buildExprAssign(ctx, tempTarget, value);
  3900. bound.setFromTarget(tempTarget);
  3901. break;
  3902. }
  3903. default:
  3904. translator.buildExpr(ctx, value, bound);
  3905. translator.ensureHasAddress(ctx, bound);
  3906. break;
  3907. }
  3908. return getPointer(bound.expr);
  3909. }
  3910. /*
  3911. interface IKeySegmentOffsetTranslator : public IInterface
  3912. {
  3913. virtual const void * getSegmentBase(const void * row) const = 0;
  3914. };
  3915. interface IKeySegmentFormatTranslator : public IInterface
  3916. {
  3917. virtual void extractField(void * target, const void * row) const = 0;
  3918. };
  3919. */
  3920. void MonitorExtractor::generateOffsetWrapping(StringBuffer & createMonitorText, IHqlExpression * selector, unsigned curOffset)
  3921. {
  3922. //MORE: Common up classes if same field filtered more than once.
  3923. OwnedHqlExpr key = createAttribute(wrapperAtom, LINK(selector));
  3924. BuildCtx declarectx(*translator.code, declareAtom);
  3925. StringBuffer s, suffix, instanceName, className, factoryName;
  3926. unique_id_t id = translator.getUniqueId();
  3927. appendUniqueId(className.append("c"), id);
  3928. appendUniqueId(instanceName.append("i"), id);
  3929. appendUniqueId(factoryName.append("f"), id);
  3930. declarectx.setPriority(SegMonitorPrio);
  3931. BuildCtx classctx(declarectx);
  3932. //MORE: Use a base class for implementing this to save Link()/Release()
  3933. s.clear().append("struct ").append(className).append(" : public RtlCInterface, public IKeySegmentOffsetTranslator");
  3934. suffix.append(instanceName).append(";");
  3935. classctx.addQuotedCompound(s, suffix);
  3936. classctx.addQuotedLiteral("virtual void Link() const { RtlCInterface::Link(); }");
  3937. classctx.addQuotedLiteral("virtual bool Release() const { return RtlCInterface::Release(); }");
  3938. classctx.addQuoted(s.clear().append("virtual const char * queryFactoryName() const { return \"").append(factoryName).append("\"; }"));
  3939. classctx.addQuotedCompoundLiteral("virtual const void * getSegmentBase(const void * _row) const");
  3940. classctx.addQuotedLiteral("const byte * row = (const byte *)_row;");
  3941. classctx.associateExpr(constantMemberMarkerExpr, constantMemberMarkerExpr);
  3942. translator.bindTableCursor(classctx, tableExpr, "row");
  3943. CHqlBoundExpr bound;
  3944. Owned<IReferenceSelector> ds = translator.buildReference(classctx, selector);
  3945. ds->buildAddress(classctx, bound);
  3946. classctx.addReturn(bound.expr);
  3947. declarectx.addQuoted(s.clear().append("IKeySegmentOffsetTranslator * ").append(factoryName).append("() { return new ").append(className).append("; }"));
  3948. if (translator.options.spanMultipleCpp)
  3949. {
  3950. s.clear().append("extern IKeySegmentOffsetTranslator * ").append(factoryName).append("();");
  3951. BuildCtx protoctx(*translator.code, mainprototypesAtom);
  3952. protoctx.addQuoted(s);
  3953. }
  3954. //TAKE NOTE: Child activities are only colocal if whole graph is colocal, or targeting roxie and activity is colocal.
  3955. //Email Richard if any objections to changing parameters so byte * not void *. Would save quite a lot of generated boilerplate code.
  3956. s.clear().append("createVarOffsetKeySegmentMonitor(").append(createMonitorText).append(",").append(curOffset).append(",").append(factoryName).append("())");
  3957. createMonitorText.swapWith(s);
  3958. }
  3959. void MonitorExtractor::generateFormatWrapping(StringBuffer & createMonitorText, IHqlExpression * selector, IHqlExpression * expandedSelector, unsigned curOffset)
  3960. {
  3961. BuildCtx declarectx(*translator.code, declareAtom);
  3962. StringBuffer s, suffix, instanceName, className, factoryName;
  3963. unique_id_t id = translator.getUniqueId();
  3964. appendUniqueId(className.append("c"), id);
  3965. appendUniqueId(instanceName.append("i"), id);
  3966. appendUniqueId(factoryName.append("f"), id);
  3967. declarectx.setNextPriority(SegMonitorPrio);
  3968. BuildCtx classctx(declarectx);
  3969. //MORE: Use a base class for implementing this to save Link()/Release()
  3970. s.clear().append("struct ").append(className).append(" : public RtlCInterface, public IKeySegmentFormatTranslator");
  3971. suffix.append(instanceName).append(";");
  3972. classctx.addQuotedCompound(s, suffix);
  3973. classctx.addQuotedLiteral("virtual void Link() const { RtlCInterface::Link(); }");
  3974. classctx.addQuotedLiteral("virtual bool Release() const { return RtlCInterface::Release(); }");
  3975. classctx.addQuoted(s.clear().append("virtual const char * queryFactoryName() const { return \"").append(factoryName).append("\"; }"));
  3976. classctx.addQuoted(s.clear().append("virtual unsigned queryHashCode() const { return ").append(getExpressionCRC(selector)).append("; }"));
  3977. classctx.addQuotedCompoundLiteral("virtual void extractField(void * _target, const void * _row) const");
  3978. classctx.associateExpr(constantMemberMarkerExpr, constantMemberMarkerExpr);
  3979. classctx.addQuotedLiteral("const byte * row = (const byte *)_row;");
  3980. classctx.addQuotedLiteral("byte * target = (byte *)_target;");
  3981. OwnedHqlExpr castValue = ensureExprType(selector, expandedSelector->queryType());
  3982. LinkedHqlExpr targetField = expandedSelector->queryChild(1);
  3983. OwnedHqlExpr simpleRecord = createRecord(targetField);
  3984. OwnedHqlExpr targetDataset = createDataset(no_anon, LINK(simpleRecord));
  3985. OwnedHqlExpr target = createSelectExpr(LINK(targetDataset), LINK(targetField));
  3986. translator.bindTableCursor(classctx, tableExpr, "row");
  3987. translator.bindTableCursor(classctx, targetDataset, "target");
  3988. translator.buildAssign(classctx, target, castValue);
  3989. declarectx.setNextPriority(SegMonitorPrio);
  3990. declarectx.addQuoted(s.clear().append("IKeySegmentFormatTranslator * ").append(factoryName).append("() { return new ").append(className).append("; }"));
  3991. if (translator.spanMultipleCppFiles())
  3992. {
  3993. s.clear().append("extern IKeySegmentFormatTranslator * ").append(factoryName).append("();");
  3994. BuildCtx protoctx(*translator.code, mainprototypesAtom);
  3995. protoctx.addQuoted(s);
  3996. }
  3997. //Now generate the key segment monitor...
  3998. s.clear().append("createTranslatedKeySegmentMonitor(").append(createMonitorText).append(",").append(curOffset).append(",").append(factoryName).append("())");
  3999. createMonitorText.swapWith(s);
  4000. }
  4001. bool MonitorExtractor::buildSingleKeyMonitor(StringBuffer & createMonitorText, KeySelectorInfo & selectorInfo, BuildCtx & ctx, IHqlExpression & thisKey)
  4002. {
  4003. BuildCtx subctx(ctx);
  4004. OwnedHqlExpr compare, normalized;
  4005. StringBuffer funcName;
  4006. extractCompareInformation(subctx, &thisKey, compare, normalized, selectorInfo.expandedSelector, selectorInfo.isComputed);
  4007. if (compare)
  4008. return false;
  4009. ITypeInfo * type = selectorInfo.expandedSelector->queryType();
  4010. type_t tc = type->getTypeCode();
  4011. if ((tc == type_int) || (tc == type_swapint))
  4012. {
  4013. if (isLittleEndian(type))
  4014. {
  4015. if (type->isSigned())
  4016. funcName.append("createSingleLittleSignedKeySegmentMonitor");
  4017. else if (type->getSize() != 1)
  4018. funcName.append("createSingleLittleKeySegmentMonitor");
  4019. }
  4020. else
  4021. {
  4022. if (type->isSigned())
  4023. funcName.append("createSingleBigSignedKeySegmentMonitor");
  4024. else
  4025. funcName.append("createSingleKeySegmentMonitor");
  4026. }
  4027. }
  4028. if (!funcName.length())
  4029. funcName.append("createSingleKeySegmentMonitor");
  4030. OwnedHqlExpr address = getMonitorValueAddress(subctx, normalized);
  4031. StringBuffer addrText;
  4032. translator.generateExprCpp(addrText, address);
  4033. unsigned offset = (selectorInfo.isComputed || selectorInfo.mapOffset) ? 0 : selectorInfo.offset;
  4034. createMonitorText.append(funcName)
  4035. .appendf("(%s, %u, %u, %s)",
  4036. boolToText(selectorInfo.keyedKind != KeyedYes), offset, selectorInfo.size, addrText.str());
  4037. return true;
  4038. }
  4039. KeyedKind getKeyedKind(HqlCppTranslator & translator, KeyConditionArray & matches)
  4040. {
  4041. KeyedKind keyedKind = KeyedNo;
  4042. ForEachItemIn(i, matches)
  4043. {
  4044. KeyCondition & cur = matches.item(i);
  4045. if (cur.keyedKind != keyedKind)
  4046. {
  4047. if (keyedKind == KeyedNo)
  4048. keyedKind = cur.keyedKind;
  4049. else
  4050. translator.throwError1(HQLERR_InconsistentKeyedOpt, str(cur.selector->queryChild(1)->queryName()));
  4051. }
  4052. }
  4053. return keyedKind;
  4054. }
  4055. void MonitorExtractor::buildEmptyKeySegment(BuildMonitorState & buildState, BuildCtx & ctx, KeySelectorInfo & selectorInfo)
  4056. {
  4057. StringBuffer s;
  4058. ctx.addQuoted(s.appendf("%s->append(createEmptyKeySegmentMonitor(%s, %u, %u));", buildState.listName, boolToText(selectorInfo.keyedKind != KeyedYes), selectorInfo.offset, selectorInfo.size));
  4059. }
  4060. void MonitorExtractor::buildWildKeySegment(BuildMonitorState & buildState, BuildCtx & ctx, unsigned offset, unsigned size)
  4061. {
  4062. StringBuffer s;
  4063. ctx.addQuoted(s.appendf("%s->append(createWildKeySegmentMonitor(%u, %u));", buildState.listName, offset, size));
  4064. }
  4065. void MonitorExtractor::buildWildKeySegment(BuildMonitorState & buildState, BuildCtx & ctx, KeySelectorInfo & selectorInfo)
  4066. {
  4067. buildWildKeySegment(buildState, ctx, selectorInfo.offset, selectorInfo.size);
  4068. }
  4069. void MonitorExtractor::buildKeySegment(BuildMonitorState & buildState, BuildCtx & ctx, unsigned whichField, unsigned curSize)
  4070. {
  4071. IHqlExpression * selector = &keyableSelects.item(whichField);
  4072. IHqlExpression * expandedSelector = &expandedSelects.item(whichField);
  4073. IHqlExpression * field = selector->queryChild(1);
  4074. KeyConditionArray matches;
  4075. bool isImplicit = true;
  4076. bool prevWildWasKeyed = buildState.wildWasKeyed;
  4077. buildState.wildWasKeyed = false;
  4078. ForEachItemIn(cond, keyed.conditions)
  4079. {
  4080. KeyCondition & cur = keyed.conditions.item(cond);
  4081. if (cur.selector == selector)
  4082. {
  4083. cur.generated = true;
  4084. if (cur.isWild)
  4085. {
  4086. isImplicit = false;
  4087. if (cur.wasKeyed)
  4088. buildState.wildWasKeyed = true;
  4089. else if (buildState.implicitWildField && !ignoreUnkeyed)
  4090. {
  4091. StringBuffer s, keyname;
  4092. translator.throwError3(HQLERR_WildFollowsGap, getExprECL(field, s).str(), str(buildState.implicitWildField->queryChild(1)->queryName()), queryKeyName(keyname));
  4093. }
  4094. }
  4095. else
  4096. {
  4097. matches.append(OLINK(cur));
  4098. if (buildState.implicitWildField && !ignoreUnkeyed)
  4099. {
  4100. StringBuffer s,keyname;
  4101. if (cur.isKeyed())
  4102. translator.throwError3(HQLERR_KeyedFollowsGap, getExprECL(field, s).str(), str(buildState.implicitWildField->queryChild(1)->queryName()), queryKeyName(keyname));
  4103. else if (!buildState.doneImplicitWarning)
  4104. {
  4105. translator.WARNING3(CategoryEfficiency, HQLWRN_KeyedFollowsGap, getExprECL(field, s).str(), str(buildState.implicitWildField->queryChild(1)->queryName()), queryKeyName(keyname));
  4106. buildState.doneImplicitWarning = true;
  4107. }
  4108. }
  4109. }
  4110. }
  4111. }
  4112. if (buildState.wildWasKeyed && (matches.ordinality() == 0))
  4113. {
  4114. StringBuffer keyname;
  4115. translator.WARNING2(CategoryFolding, HQLWRN_FoldRemoveKeyed, str(field->queryName()), queryKeyName(keyname));
  4116. }
  4117. StringBuffer s;
  4118. KeyedKind keyedKind = getKeyedKind(translator, matches);
  4119. KeySelectorInfo selectorInfo(keyedKind, selector, expandedSelector, buildState.curOffset, curSize, (whichField >= firstOffsetField), false);
  4120. bool ignoreKeyedExtend = false;
  4121. if ((keyedKind == KeyedExtend) && buildState.wildPending() && !ignoreUnkeyed)
  4122. {
  4123. if (keyedKind == KeyedExtend)
  4124. {
  4125. if (prevWildWasKeyed)
  4126. buildState.wildWasKeyed = true;
  4127. else
  4128. {
  4129. StringBuffer keyname;
  4130. translator.WARNING2(CategoryEfficiency, HQLERR_OptKeyedFollowsWild, getExprECL(field, s).str(), queryKeyName(keyname));
  4131. }
  4132. }
  4133. //previous condition folded so always true, so keyed,opt will always be a wildcard.
  4134. if (!allowDynamicFormatChange)
  4135. ignoreKeyedExtend = true;
  4136. isImplicit = false;
  4137. }
  4138. if (matches.ordinality() && !ignoreKeyedExtend)
  4139. {
  4140. if (buildState.wildPending() && !ignoreUnkeyed)
  4141. {
  4142. buildWildKeySegment(buildState, ctx, buildState.wildOffset, buildState.curOffset-buildState.wildOffset);
  4143. buildState.clearWild();
  4144. }
  4145. HqlExprArray args;
  4146. ForEachItemIn(i, matches)
  4147. {
  4148. KeyCondition & cur = matches.item(i);
  4149. args.append(*LINK(cur.expr));
  4150. }
  4151. OwnedHqlExpr fullExpr = createBalanced(no_and, queryBoolType(), args);
  4152. BuildCtx subctx(ctx);
  4153. buildKeySegmentExpr(buildState, selectorInfo, subctx, NULL, *fullExpr, ignoreUnkeyed ? MonitorFilterSkipAll : NoMonitorFilter);
  4154. }
  4155. else
  4156. {
  4157. if (isImplicit)
  4158. {
  4159. buildState.implicitWildField.set(selector);
  4160. buildState.doneImplicitWarning = false;
  4161. }
  4162. if (buildState.wildPending() && noMergeSelects.contains(*selector))
  4163. {
  4164. buildWildKeySegment(buildState, ctx, buildState.wildOffset, buildState.curOffset-buildState.wildOffset);
  4165. buildState.clearWild();
  4166. }
  4167. if (!buildState.wildPending())
  4168. buildState.wildOffset = buildState.curOffset;
  4169. }
  4170. buildState.curOffset += selectorInfo.size;
  4171. }
  4172. void MonitorExtractor::buildArbitaryKeySegment(BuildMonitorState & buildState, BuildCtx & ctx, unsigned curSize, IHqlExpression * condition)
  4173. {
  4174. IHqlExpression * left = condition->queryChild(0);
  4175. node_operator op = condition->getOperator();
  4176. StringBuffer createMonitorText;
  4177. OwnedHqlExpr field = createField(unknownId, getExpandedFieldType(left->queryType(), NULL), NULL);
  4178. OwnedHqlExpr pseudoSelector = createSelectExpr(getActiveTableSelector(), LINK(field));
  4179. KeySelectorInfo selectorInfo(KeyedExtend, left, pseudoSelector, buildState.curOffset, curSize, false, true);
  4180. BuildCtx subctx(ctx);
  4181. buildKeySegmentExpr(buildState, selectorInfo, subctx, NULL, *condition, MonitorFilterSkipAll);
  4182. }
  4183. void MonitorExtractor::spotSegmentCSE(BuildCtx & ctx)
  4184. {
  4185. //This could make things much better, but needs some thought
  4186. HqlExprArray conditions;
  4187. ForEachItemIn(cond, keyed.conditions)
  4188. {
  4189. KeyCondition & cur = keyed.conditions.item(cond);
  4190. if (cur.expr)
  4191. conditions.append(*LINK(cur.expr));
  4192. }
  4193. HqlExprArray associated;
  4194. IHqlExpression * selector = tableExpr->queryNormalizedSelector();
  4195. translator.traceExpressions("before seg spot", conditions);
  4196. spotScalarCSE(conditions, associated, NULL, selector, translator.queryOptions().spotCseInIfDatasetConditions);
  4197. translator.traceExpressions("after seg spot", conditions);
  4198. unsigned curCond = 0;
  4199. ForEachItemIn(i, conditions)
  4200. {
  4201. IHqlExpression * cur = &conditions.item(i);
  4202. switch (cur->getOperator())
  4203. {
  4204. case no_alias:
  4205. translator.buildStmt(ctx, cur);
  4206. break;
  4207. case no_alias_scope:
  4208. translator.expandAliasScope(ctx, cur);
  4209. cur = cur->queryChild(0);
  4210. //fallthrough
  4211. default:
  4212. loop
  4213. {
  4214. if (!keyed.conditions.isItem(curCond))
  4215. throwUnexpected();
  4216. KeyCondition & keyCond = keyed.conditions.item(curCond++);
  4217. if (keyCond.expr)
  4218. {
  4219. keyCond.expr.set(cur);
  4220. break;
  4221. }
  4222. }
  4223. break;
  4224. }
  4225. }
  4226. loop
  4227. {
  4228. if (!keyed.conditions.isItem(curCond))
  4229. break;
  4230. KeyCondition & keyCond = keyed.conditions.item(curCond++);
  4231. assertex(!keyCond.expr);
  4232. }
  4233. }
  4234. void MonitorExtractor::buildSegments(BuildCtx & ctx, const char * listName, bool _ignoreUnkeyed)
  4235. {
  4236. translator.useInclude("rtlkey.hpp");
  4237. ignoreUnkeyed = _ignoreUnkeyed;
  4238. if (translator.queryOptions().spotCSE)
  4239. spotSegmentCSE(ctx);
  4240. BuildMonitorState buildState(ctx, listName);
  4241. ForEachItemIn(idx, keyableSelects)
  4242. {
  4243. IHqlExpression * selector = &keyableSelects.item(idx);
  4244. IHqlExpression * expandedSelector = &expandedSelects.item(idx);
  4245. IHqlExpression * field = selector->queryChild(1);
  4246. unsigned curSize = expandedSelector->queryType()->getSize();
  4247. assertex(curSize != UNKNOWN_LENGTH);
  4248. //MORE: Should also allow nested record structures, and allow keying on first elements.
  4249. // and field->queryType()->getSize() doesn't work for alien datatypes etc.
  4250. if(!field->hasAttribute(virtualAtom))
  4251. {
  4252. if (mergedSizes.isItem(idx))
  4253. curSize = mergedSizes.item(idx);
  4254. if (curSize)
  4255. buildKeySegment(buildState, ctx, idx, curSize);
  4256. else
  4257. {
  4258. ForEachItemIn(cond, keyed.conditions)
  4259. {
  4260. KeyCondition & cur = keyed.conditions.item(cond);
  4261. if (cur.selector == selector)
  4262. cur.generated = true;
  4263. }
  4264. }
  4265. }
  4266. }
  4267. if (buildState.wildPending() && !ignoreUnkeyed)
  4268. buildWildKeySegment(buildState, ctx, buildState.wildOffset, buildState.curOffset-buildState.wildOffset);
  4269. //These really don't work very sensibly - we would need an offset that was constant, possibly based on the crc of the expression.
  4270. //I suspect they need a complete rethink.
  4271. ForEachItemIn(i, keyed.conditions)
  4272. {
  4273. KeyCondition & cur = keyed.conditions.item(i);
  4274. if (cur.selector->isAttribute() && cur.selector->queryName() == _translated_Atom)
  4275. {
  4276. BuildCtx subctx(ctx);
  4277. IHqlExpression * curExpr = cur.expr;
  4278. unsigned curSize = curExpr->queryChild(0)->queryType()->getSize();
  4279. buildArbitaryKeySegment(buildState, subctx, curSize, curExpr);
  4280. buildState.curOffset += curSize;
  4281. cur.generated = true;
  4282. }
  4283. }
  4284. //check that all keyed entries have been matched
  4285. ForEachItemIn(cond, keyed.conditions)
  4286. {
  4287. KeyCondition & cur = keyed.conditions.item(cond);
  4288. if (!cur.generated)
  4289. translator.throwError(HQLERR_OnlyKeyFixedField);
  4290. }
  4291. }
  4292. static UniqueSequenceCounter translatedSequence;
  4293. KeyCondition * MonitorExtractor::createTranslatedCondition(IHqlExpression * cond, KeyedKind keyedKind)
  4294. {
  4295. OwnedHqlExpr seq = createSequence(no_attr, makeNullType(), _translated_Atom, translatedSequence.next());
  4296. return new KeyCondition(seq, cond, keyedKind);
  4297. }
  4298. bool MonitorExtractor::isKeySelect(IHqlExpression * select)
  4299. {
  4300. return (keyableSelects.find(*select) != NotFound);
  4301. }
  4302. bool MonitorExtractor::isEqualityFilter(IHqlExpression * search)
  4303. {
  4304. bool matched = false;
  4305. ForEachItemIn(cond, keyed.conditions)
  4306. {
  4307. KeyCondition & cur = keyed.conditions.item(cond);
  4308. if (cur.selector == search)
  4309. {
  4310. if (!cur.isWild)
  4311. {
  4312. if (matched)
  4313. return false;
  4314. matched = true;
  4315. IHqlExpression * matchExpr = cur.expr;
  4316. if (matchExpr->getOperator() != no_eq)
  4317. return false;
  4318. }
  4319. }
  4320. }
  4321. return matched;
  4322. }
  4323. bool MonitorExtractor::isEqualityFilterBefore(IHqlExpression * select)
  4324. {
  4325. ForEachItemIn(i, keyableSelects)
  4326. {
  4327. IHqlExpression & cur = keyableSelects.item(i);
  4328. if (select == &cur)
  4329. return true;
  4330. if (!isEqualityFilter(&cur))
  4331. return false;
  4332. }
  4333. throwUnexpected();
  4334. }
  4335. bool MonitorExtractor::isPrevSelectKeyed(IHqlExpression * select)
  4336. {
  4337. unsigned match = keyableSelects.find(*select);
  4338. assertex(match != NotFound);
  4339. if (match == 0)
  4340. return true;
  4341. IHqlExpression * prev = &keyableSelects.item(match-1);
  4342. ForEachItemIn(i, keyed.conditions)
  4343. {
  4344. KeyCondition & cur = keyed.conditions.item(i);
  4345. if (cur.selector == prev)
  4346. {
  4347. if (!cur.isWild && cur.isKeyed())
  4348. return true;
  4349. if (cur.wasKeyed)
  4350. return true;
  4351. }
  4352. }
  4353. return false;
  4354. }
  4355. bool MonitorExtractor::okToKey(IHqlExpression * select, KeyedKind keyedKind)
  4356. {
  4357. if (keyedKind == KeyedYes)
  4358. return true;
  4359. ForEachItemIn(i, keyed.conditions)
  4360. {
  4361. KeyCondition & cur = keyed.conditions.item(i);
  4362. if (cur.selector == select && cur.isWild)
  4363. return false;
  4364. }
  4365. return true;
  4366. }
  4367. bool MonitorExtractor::isIndexInvariant(IHqlExpression * expr)
  4368. {
  4369. if (containsAssertKeyed(expr))
  4370. return false;
  4371. HqlExprCopyArray scopeUsed;
  4372. expr->gatherTablesUsed(NULL, &scopeUsed);
  4373. IHqlExpression * search = tableExpr->queryNormalizedSelector();
  4374. ForEachItemIn(i, scopeUsed)
  4375. {
  4376. IHqlExpression * cur = &scopeUsed.item(i);
  4377. loop
  4378. {
  4379. if (cur == search || queryRoot(cur) == search)
  4380. return false;
  4381. cur = queryNextMultiLevelDataset(cur, true);
  4382. if (!cur)
  4383. break;
  4384. }
  4385. }
  4386. return true;
  4387. }
  4388. IHqlExpression * MonitorExtractor::castToFieldAndBack(IHqlExpression * left, IHqlExpression * right)
  4389. {
  4390. node_operator op = left->getOperator();
  4391. switch (op)
  4392. {
  4393. case no_cast:
  4394. case no_implicitcast:
  4395. {
  4396. IHqlExpression * uncast = left->queryChild(0);
  4397. ITypeInfo * castType = right->queryType();
  4398. ITypeInfo * uncastType = uncast->queryType();
  4399. OwnedHqlExpr castRight = ensureExprType(right, uncastType);
  4400. OwnedHqlExpr base = castToFieldAndBack(uncast, castRight);
  4401. //If this cast doesn't lose any information and child didn't change then don't bother
  4402. //casting back and forwards.
  4403. if ((base == castRight) && !castLosesInformation(uncastType, castType))
  4404. return LINK(right);
  4405. return ensureExprType(base, castType);
  4406. }
  4407. case no_select:
  4408. {
  4409. ITypeInfo * leftType = left->queryType();
  4410. ITypeInfo * rightType = right->queryType();
  4411. if (leftType == rightType || !castLosesInformation(leftType, rightType))
  4412. return LINK(right);
  4413. return ensureExprType(right, leftType);
  4414. }
  4415. case no_substring:
  4416. case no_add:
  4417. case no_sub:
  4418. return castToFieldAndBack(left->queryChild(0), right);
  4419. default:
  4420. UNIMPLEMENTED;
  4421. }
  4422. }
  4423. IHqlExpression * MonitorExtractor::invertTransforms(IHqlExpression * left, IHqlExpression * right)
  4424. {
  4425. node_operator op = left->getOperator();
  4426. switch (op)
  4427. {
  4428. case no_cast:
  4429. case no_implicitcast:
  4430. {
  4431. assertex(right->queryType()->getTypeCode() != type_set);
  4432. IHqlExpression * uncast = left->queryChild(0);
  4433. ITypeInfo * uncastType = uncast->queryType();
  4434. OwnedHqlExpr castRight = ensureExprType(right, uncastType);
  4435. return invertTransforms(uncast, castRight);
  4436. }
  4437. case no_select:
  4438. {
  4439. assertex(isKeySelect(left));
  4440. ITypeInfo * leftType = left->queryType();
  4441. ITypeInfo * rightType = right->queryType();
  4442. if (leftType == rightType || !castLosesInformation(leftType, rightType))
  4443. return LINK(right);
  4444. return ensureExprType(right, leftType);
  4445. }
  4446. case no_add:
  4447. case no_sub:
  4448. {
  4449. assertex(right->getOperator() != no_list);
  4450. OwnedHqlExpr adjusted = createValue(op == no_sub ? no_add : no_sub, right->getType(), LINK(right), LINK(left->queryChild(1)));
  4451. return invertTransforms(left->queryChild(0), adjusted);
  4452. }
  4453. case no_substring:
  4454. {
  4455. assertex(right->getOperator() != no_list);
  4456. return invertTransforms(left->queryChild(0), right);
  4457. }
  4458. default:
  4459. UNIMPLEMENTED;
  4460. }
  4461. }
  4462. IHqlExpression * MonitorExtractor::queryKeyableSelector(IHqlExpression * expr)
  4463. {
  4464. switch (expr->getOperator())
  4465. {
  4466. case no_cast:
  4467. case no_implicitcast:
  4468. case no_add:
  4469. case no_sub:
  4470. return queryKeyableSelector(expr->queryChild(0));
  4471. case no_select:
  4472. if (isKeySelect(expr))
  4473. return expr;
  4474. return NULL;
  4475. }
  4476. return NULL;
  4477. }
  4478. IHqlExpression * MonitorExtractor::isKeyableFilter(IHqlExpression * left, IHqlExpression * right, bool & duplicate, node_operator compareOp, KeyFailureInfo & reason, KeyedKind keyedKind)
  4479. {
  4480. node_operator op = left->getOperator();
  4481. switch (op)
  4482. {
  4483. case no_cast:
  4484. case no_implicitcast:
  4485. {
  4486. IHqlExpression * uncast = left->queryChild(0);
  4487. ITypeInfo * castType = left->queryType();
  4488. ITypeInfo * uncastType = uncast->queryType();
  4489. //(ty)x = y. E.g., (int1)string2field = int1value
  4490. //if more than one value of x[uncastType] corresponds to a single value in y[castType] then we can't sensibly create
  4491. //the key segment monitor. Because we will get false negatives. If it is an inverse then duplicate (see below)
  4492. bool canRemoveCast = true;
  4493. if (castLosesInformation(castType, uncastType))
  4494. {
  4495. if ((compareOp != no_ne) && (compareOp != no_notin))
  4496. canRemoveCast = false;
  4497. duplicate = true;
  4498. }
  4499. //if more than one value of y corresponds to a single value of x then need to duplicate the test condition.
  4500. //or pretest whether (ty)(tx)y == y.
  4501. //if (castLosesInformation(uncastType, castType))
  4502. //Now taken care of when the segment monitors are created
  4503. //If the comparison is non equality and the cast changes the collation sequence then you can't remove it.
  4504. switch (compareOp)
  4505. {
  4506. case no_eq:
  4507. case no_ne:
  4508. case no_in:
  4509. case no_notin:
  4510. break;
  4511. default:
  4512. if (!preservesOrder(castType, uncastType))
  4513. canRemoveCast = false;
  4514. break;
  4515. }
  4516. Linked<ITypeInfo> newType = uncastType;
  4517. if (right->queryType()->getTypeCode() == type_set)
  4518. newType.setown(makeSetType(newType.getLink()));
  4519. OwnedHqlExpr castRight = ensureExprType(right, newType);
  4520. IHqlExpression * ret = isKeyableFilter(uncast, castRight, duplicate, compareOp, reason, keyedKind);
  4521. if (canRemoveCast || !ret)
  4522. return ret;
  4523. reason.set(KFRcast, ret->queryChild(1));
  4524. return NULL;
  4525. }
  4526. case no_select:
  4527. if (isKeySelect(left) && okToKey(left, keyedKind))
  4528. {
  4529. if (isIndexInvariant(right))
  4530. return left;
  4531. reason.set(KFRtoocomplex, left);
  4532. }
  4533. else
  4534. reason.set(KFRnokey);
  4535. return NULL;
  4536. case no_substring:
  4537. {
  4538. IHqlExpression * range = left->queryChild(1);
  4539. if (range->getOperator() == no_rangeto)
  4540. {
  4541. IValue *end = range->queryChild(0)->queryValue();
  4542. if (!end)
  4543. break;
  4544. return isKeyableFilter(left->queryChild(0), right, duplicate, compareOp, reason, keyedKind);
  4545. }
  4546. else if (range->getOperator() == no_range)
  4547. {
  4548. IValue *start = range->queryChild(0)->queryValue();
  4549. IValue *end = range->queryChild(1)->queryValue();
  4550. if (!start || !end || start->getIntValue() != 1)
  4551. break;
  4552. return isKeyableFilter(left->queryChild(0), right, duplicate, compareOp, reason, keyedKind);
  4553. }
  4554. reason.set(KFRtoocomplex, right);
  4555. return NULL;
  4556. }
  4557. case no_add:
  4558. case no_sub:
  4559. if (isIndexInvariant(left->queryChild(1)))
  4560. return isKeyableFilter(left->queryChild(0), right, duplicate, compareOp, reason, keyedKind);
  4561. reason.set(KFRtoocomplex, left);
  4562. return NULL;
  4563. }
  4564. reason.set(KFRnokey);
  4565. return NULL;
  4566. }
  4567. static IHqlExpression * getCompareValue(ITypeInfo * fieldType, unsigned subStringLen, IValue * value, int whichBoundary)
  4568. {
  4569. type_t ftc = fieldType->getTypeCode();
  4570. unsigned fieldLen = fieldType->getStringLen();
  4571. unsigned lenValue = value->queryType()->getStringLen();
  4572. const void * rawValue = value->queryValue();
  4573. size32_t resultLen;
  4574. rtlDataAttr result;
  4575. if (whichBoundary < 0)
  4576. {
  4577. switch (ftc)
  4578. {
  4579. case type_qstring:
  4580. rtlCreateQStrRangeLow(resultLen, result.refstr(), fieldLen, subStringLen, lenValue, static_cast<const char *>(rawValue));
  4581. break;
  4582. case type_string:
  4583. rtlCreateStrRangeLow(resultLen, result.refstr(), fieldLen, subStringLen, lenValue, static_cast<const char *>(rawValue));
  4584. break;
  4585. case type_data:
  4586. rtlCreateDataRangeLow(resultLen, result.refdata(), fieldLen, subStringLen, lenValue, rawValue);
  4587. break;
  4588. case type_unicode:
  4589. rtlCreateUnicodeRangeLow(resultLen, result.refustr(), fieldLen, subStringLen, lenValue, static_cast<const UChar *>(rawValue));
  4590. break;
  4591. default:
  4592. //should this generate a warning/error instead?
  4593. rtlCreateRange(resultLen, result.refstr(), fieldLen, subStringLen, fieldType->getSize(), static_cast<const char *>(rawValue), 0, 0);
  4594. break;
  4595. }
  4596. }
  4597. else
  4598. {
  4599. switch (ftc)
  4600. {
  4601. case type_qstring:
  4602. rtlCreateQStrRangeHigh(resultLen, result.refstr(), fieldLen, subStringLen, lenValue, static_cast<const char *>(rawValue));
  4603. break;
  4604. case type_string:
  4605. rtlCreateStrRangeHigh(resultLen, result.refstr(), fieldLen, subStringLen, lenValue, static_cast<const char *>(rawValue));
  4606. break;
  4607. case type_data:
  4608. rtlCreateDataRangeHigh(resultLen, result.refdata(), fieldLen, subStringLen, lenValue, rawValue);
  4609. break;
  4610. case type_unicode:
  4611. rtlCreateUnicodeRangeHigh(resultLen, result.refustr(), fieldLen, subStringLen, lenValue, static_cast<const UChar *>(rawValue));
  4612. break;
  4613. default:
  4614. rtlCreateRange(resultLen, result.refstr(), fieldLen, subStringLen, fieldType->getSize(), static_cast<const char *>(rawValue), 255, 0);
  4615. break;
  4616. }
  4617. }
  4618. assertex(resultLen == fieldLen);
  4619. return createConstant(createValueFromMem(LINK(fieldType), result.getdata()));
  4620. }
  4621. IHqlExpression * MonitorExtractor::getRangeLimit(ITypeInfo * fieldType, IHqlExpression * lengthExpr, IHqlExpression * value, int whichBoundary)
  4622. {
  4623. type_t ftc = fieldType->getTypeCode();
  4624. unsigned fieldLength = fieldType->getStringLen();
  4625. IValue * constValue = value->queryValue();
  4626. if (constValue && lengthExpr->queryValue())
  4627. {
  4628. unsigned subStringLen = (unsigned)lengthExpr->queryValue()->getIntValue();
  4629. if ((int)subStringLen < 0) subStringLen = 0;
  4630. if (subStringLen > fieldLength)
  4631. translator.throwError1(HQLERR_SubstringOutOfRange, subStringLen);
  4632. return getCompareValue(fieldType, subStringLen, constValue, whichBoundary);
  4633. }
  4634. IIdAtom * func;
  4635. if (whichBoundary < 0)
  4636. {
  4637. switch (ftc)
  4638. {
  4639. case type_qstring:
  4640. func = createQStrRangeLowId;
  4641. break;
  4642. case type_string:
  4643. func = createStrRangeLowId;
  4644. break;
  4645. case type_data:
  4646. func = createDataRangeLowId;
  4647. break;
  4648. case type_unicode:
  4649. func = createUnicodeRangeLowId;
  4650. break;
  4651. default:
  4652. func = createRangeLowId;
  4653. break;
  4654. }
  4655. }
  4656. else
  4657. {
  4658. switch (ftc)
  4659. {
  4660. case type_qstring:
  4661. func = createQStrRangeHighId;
  4662. break;
  4663. case type_string:
  4664. func = createStrRangeHighId;
  4665. break;
  4666. case type_data:
  4667. func = createDataRangeHighId;
  4668. break;
  4669. case type_unicode:
  4670. func = createUnicodeRangeHighId;
  4671. break;
  4672. default:
  4673. func = createRangeHighId;
  4674. break;
  4675. }
  4676. }
  4677. HqlExprArray args;
  4678. args.append(*getSizetConstant(fieldLength));
  4679. args.append(*LINK(lengthExpr));
  4680. args.append(*LINK(value));
  4681. //Note: I can't change the return type of the function - because if fixed length then wrong call is made, and variable length is worse code.
  4682. OwnedHqlExpr call = translator.bindFunctionCall(func, args);
  4683. return createValue(no_typetransfer, LINK(fieldType), LINK(call));
  4684. }
  4685. IHqlExpression * MonitorExtractor::createRangeCompare(IHqlExpression * selector, IHqlExpression * value, IHqlExpression * lengthExpr, bool compareEqual)
  4686. {
  4687. OwnedHqlExpr foldedValue = foldHqlExpression(value);
  4688. ITypeInfo * fieldType = selector->queryType();
  4689. IHqlExpression * lowExpr = getRangeLimit(fieldType, lengthExpr, foldedValue, -1);
  4690. IHqlExpression * highExpr = getRangeLimit(fieldType, lengthExpr, foldedValue, +1);
  4691. //Could convert to two separate tests, but code is worse, and boundary conditions aren't going to happen.
  4692. return createValue(compareEqual ? no_between : no_notbetween, makeBoolType(), LINK(selector), lowExpr, highExpr);
  4693. }
  4694. static IHqlExpression * removeCastTrim(IHqlExpression * expr)
  4695. {
  4696. loop
  4697. {
  4698. if ((expr->getOperator() == no_trim) && !expr->queryChild(1))
  4699. expr = expr->queryChild(0);
  4700. else if (isLengthPreservingCast(expr))
  4701. expr = expr->queryChild(0);
  4702. else
  4703. return expr;
  4704. expr = queryNonAliased(expr);
  4705. }
  4706. }
  4707. static IHqlExpression * queryLengthFromRange(IHqlExpression * range)
  4708. {
  4709. switch (range->getOperator())
  4710. {
  4711. case no_rangefrom:
  4712. case no_rangecommon:
  4713. return NULL;
  4714. case no_rangeto:
  4715. return range->queryChild(0);
  4716. case no_range:
  4717. if (getIntValue(range->queryChild(0), 0) == 1)
  4718. return range->queryChild(1);
  4719. return NULL;
  4720. default:
  4721. if (getIntValue(range, 0) == 1)
  4722. return range;
  4723. return NULL;
  4724. }
  4725. }
  4726. static void extendRangeCheck(SharedHqlExpr & globalGuard, SharedHqlExpr & localCond, IHqlExpression * selector, IHqlExpression * lengthExpr, bool compareEqual)
  4727. {
  4728. #if 0
  4729. //This might be a good idea, but probably doesn't make a great deal of difference at runtime
  4730. //Optimize the case where you check zero length to use a wild carded range instead
  4731. //x[1..len=0] = y from x in range 0000000..FFFFFF to len==0 || x in range....
  4732. if (compareEqual)
  4733. {
  4734. if (!lengthExpr->queryValue())
  4735. {
  4736. OwnedHqlExpr testLength = createBoolExpr(no_eq, LINK(lengthExpr), ensureExprType(queryZero(), lengthExpr->queryType()));
  4737. localCond.setown(createBoolExpr(no_or, testLength.getClear(), LINK(localCond)));
  4738. }
  4739. }
  4740. #endif
  4741. //For a range check x[1..m] = y we generate
  4742. //x in range(sizeof(x), m, y, 0) and range(maxlength(x), m, y, 255)
  4743. //need to guard with condition len(trim(y)) <= m
  4744. //If x[1..length(trim(y))] == y then don't add a condition
  4745. IHqlExpression * cur = queryNonAliased(lengthExpr);
  4746. IHqlExpression * compare = removeCastTrim(queryNonAliased(selector));
  4747. if (cur->getOperator() == no_charlen)
  4748. {
  4749. cur = queryNonAliased(cur->queryChild(0));
  4750. if (cur->getOperator() == no_trim)
  4751. {
  4752. cur = queryNonAliased(cur->queryChild(0));
  4753. while (isLengthPreservingCast(cur))
  4754. cur = queryNonAliased(cur->queryChild(0));
  4755. IHqlExpression * compare = selector;
  4756. loop
  4757. {
  4758. compare = removeCastTrim(queryNonAliased(compare));
  4759. if (cur->queryBody() == compare->queryBody())
  4760. return;
  4761. //Casts between strings that reduce/increase the number of characters don't matter as long as
  4762. //they eventually match the search string
  4763. if (!isCast(compare) || !isStringType(compare->queryType()))
  4764. break;
  4765. compare = compare->queryChild(0);
  4766. if (!isStringType(compare->queryType()))
  4767. break;
  4768. }
  4769. }
  4770. }
  4771. // if x[1..n] = z[1..n] then no need for a condition
  4772. if (compare->getOperator() == no_substring)
  4773. {
  4774. IHqlExpression * range = queryLengthFromRange(compare->queryChild(1));
  4775. if (range == lengthExpr)
  4776. return;
  4777. }
  4778. // if x[1..n] = (string<m>y) where m<n then no need for condition
  4779. unsigned selectorLength = selector->queryType()->getStringLen();
  4780. if (selectorLength <= getIntValue(lengthExpr, 0))
  4781. return;
  4782. //otherwise, if x[1..y] == z then add check length(trim(z)) <= y
  4783. OwnedHqlExpr trim = createTrimExpr(selector, NULL);
  4784. OwnedHqlExpr len = createValue(no_charlen, LINK(unsignedType), LINK(trim));
  4785. ITypeInfo * lengthType = lengthExpr->queryType();
  4786. Owned<ITypeInfo> compareType = getPromotedECLCompareType(unsignedType, lengthType);
  4787. node_operator compOp = compareEqual ? no_le : no_gt;
  4788. OwnedHqlExpr positiveLen = createValue(no_maxlist, lengthExpr->getType(), createValue(no_list, makeSetType(LINK(lengthType)), LINK(lengthExpr), createConstant(lengthType->castFrom(false, I64C(0)))));
  4789. OwnedHqlExpr test = createValue(no_le, makeBoolType(), ensureExprType(len, compareType), ensureExprType(positiveLen, compareType));
  4790. test.setown(foldHqlExpression(test));
  4791. if (compareEqual)
  4792. extendConditionOwn(globalGuard, no_and, test.getClear());
  4793. else
  4794. extendConditionOwn(localCond, no_or, getInverse(test));
  4795. }
  4796. bool MonitorExtractor::matchSubstringFilter(KeyConditionInfo & matches, node_operator op, IHqlExpression * left, IHqlExpression * right, KeyedKind keyedKind, bool & duplicate)
  4797. {
  4798. LinkedHqlExpr value = right;
  4799. duplicate = false;
  4800. OwnedHqlExpr guard;
  4801. ITypeInfo * guardCastType = NULL;
  4802. if ((left->getOperator() == no_cast) || (left->getOperator() == no_implicitcast))
  4803. {
  4804. //code is extracted and simplified from isKeyableFilter() above - should be commoned up.
  4805. IHqlExpression * uncast = left->queryChild(0);
  4806. ITypeInfo * castType = left->queryType();
  4807. ITypeInfo * uncastType = uncast->queryType();
  4808. //(ty)x = y.
  4809. //if more than one value of x[uncastType] corresponds to a single value in y[castType] then we can't sensibly create
  4810. //the key segment monitor. Because we will get false negatives. If it is an inverse then duplicate (see below)
  4811. bool canRemoveCast = true;
  4812. if (castLosesInformation(castType, uncastType))
  4813. canRemoveCast = false;
  4814. //if more than one value of y corresponds to a single value of x then need to duplicate the test condition.
  4815. if (!preservesOrder(castType, uncastType))
  4816. canRemoveCast = false;
  4817. if (!canRemoveCast)
  4818. {
  4819. // reason.set(KFRcast, ret->queryChild(1));
  4820. return false;
  4821. }
  4822. if ((op != no_in) && (op != no_notin))
  4823. {
  4824. value.setown(ensureExprType(right, uncastType));
  4825. //If a simple equality test then create a global guard to check that we aren't matching a false positive
  4826. if (castLosesInformation(uncastType, castType))
  4827. guard.setown(createBoolExpr(no_eq, ensureExprType(value, castType), LINK(right)));
  4828. }
  4829. else
  4830. {
  4831. //if an IN then add guards to each comparison - generated later...
  4832. if (castLosesInformation(uncastType, castType))
  4833. guardCastType = uncastType;
  4834. else
  4835. {
  4836. Owned<ITypeInfo> targetType = makeSetType(LINK(uncastType));
  4837. value.setown(ensureExprType(right, targetType));
  4838. }
  4839. }
  4840. left = uncast;
  4841. }
  4842. if (left->getOperator() != no_substring)
  4843. return false;
  4844. if ((op == no_in) || (op == no_notin))
  4845. {
  4846. value.setown(normalizeListCasts(value));
  4847. if (value->getOperator() != no_list)
  4848. return false;
  4849. }
  4850. IHqlExpression * selector = left->queryChild(0);
  4851. if (!isKeySelect(selector) || !okToKey(selector, keyedKind))
  4852. return false;
  4853. if (!isIndexInvariant(right))
  4854. return false;
  4855. ITypeInfo * fieldType = selector->queryType();
  4856. unsigned fieldLength = fieldType->getStringLen();
  4857. if (fieldLength == UNKNOWN_LENGTH)
  4858. return false;
  4859. OwnedHqlExpr range = foldHqlExpression(left->queryChild(1));
  4860. IHqlExpression * lengthExpr = queryLengthFromRange(range);
  4861. if (!lengthExpr)
  4862. return false;
  4863. OwnedHqlExpr newTest;
  4864. if ((op == no_eq) || (op == no_ne))
  4865. {
  4866. newTest.setown(createRangeCompare(selector, value, lengthExpr, op == no_eq));
  4867. extendRangeCheck(guard, newTest, right, lengthExpr, op == no_eq);
  4868. }
  4869. else //no_in, no_notin
  4870. {
  4871. HqlExprArray compares;
  4872. ForEachChild(i, value)
  4873. {
  4874. IHqlExpression * cur = value->queryChild(i);
  4875. LinkedHqlExpr castValue = cur;
  4876. OwnedHqlExpr valueGuard;
  4877. if (guardCastType)
  4878. {
  4879. castValue.setown(ensureExprType(castValue, guardCastType));
  4880. valueGuard.setown(createBoolExpr(no_eq, ensureExprType(castValue, cur->queryType()), LINK(cur)));
  4881. extendRangeCheck(valueGuard, valueGuard, cur, lengthExpr, (op == no_in));
  4882. }
  4883. OwnedHqlExpr cond = createRangeCompare(selector, castValue, lengthExpr, (op == no_in));
  4884. if (valueGuard)
  4885. cond.setown(createValue(no_and, makeBoolType(), valueGuard.getClear(), cond.getClear()));
  4886. compares.append(*cond.getClear());
  4887. }
  4888. node_operator combineOp = (op == no_in) ? no_or : no_and;
  4889. newTest.setown(createBalanced(combineOp, queryBoolType(), compares));
  4890. }
  4891. matches.appendCondition(*new KeyCondition(selector, newTest, keyedKind));
  4892. if (guard)
  4893. matches.appendPreFilter(guard);
  4894. return true;
  4895. }
  4896. bool MonitorExtractor::extractSimpleCompareFilter(KeyConditionInfo & matches, IHqlExpression * expr, KeyedKind keyedKind)
  4897. {
  4898. OwnedHqlExpr promoted = getExplicitlyPromotedCompare(expr);
  4899. IHqlExpression * l = promoted->queryChild(0);
  4900. IHqlExpression * r = promoted->queryChild(1);
  4901. bool duplicate = false;
  4902. KeyFailureInfo reasonl, reasonr;
  4903. node_operator op = expr->getOperator();
  4904. IHqlExpression * matchedSelector = isKeyableFilter(l, r, duplicate, op, reasonl, keyedKind);
  4905. Owned<KeyCondition> result;
  4906. if (matchedSelector)
  4907. {
  4908. node_operator newOp = getModifiedOp(op, duplicate);
  4909. if (newOp != no_none)
  4910. {
  4911. OwnedHqlExpr newFilter = createValue(newOp, expr->getType(), LINK(l), LINK(r));
  4912. result.setown(new KeyCondition(matchedSelector, newFilter, keyedKind));
  4913. }
  4914. }
  4915. else
  4916. {
  4917. duplicate = false;
  4918. matchedSelector = isKeyableFilter(r, l, duplicate, op, reasonr, keyedKind);
  4919. if (matchedSelector)
  4920. {
  4921. node_operator newOp = getModifiedOp(getReverseOp(op), duplicate);
  4922. if (newOp != no_none)
  4923. {
  4924. OwnedHqlExpr newFilter = createValue(newOp, expr->getType(), LINK(r), LINK(l));
  4925. result.setown(new KeyCondition(matchedSelector, newFilter, keyedKind));
  4926. }
  4927. }
  4928. }
  4929. if (!result && allowTranslatedConds)
  4930. {
  4931. duplicate = false;
  4932. ITypeInfo * type = l->queryType();
  4933. if (isKeyableType(type))
  4934. {
  4935. bool leftHasSelects = containsTableSelects(l);
  4936. bool rightHasSelects = containsTableSelects(r);
  4937. if (leftHasSelects && !rightHasSelects)
  4938. {
  4939. result.setown(createTranslatedCondition(expr, keyedKind));
  4940. }
  4941. else if (!leftHasSelects && rightHasSelects && (op != no_in) && (op != no_notin))
  4942. {
  4943. OwnedHqlExpr newFilter = createValue(getReverseOp(op), expr->getType(), LINK(r), LINK(l));
  4944. result.setown(createTranslatedCondition(newFilter, keyedKind));
  4945. }
  4946. }
  4947. }
  4948. bool extracted = (result != NULL);
  4949. if (extracted)
  4950. {
  4951. matches.appendCondition(*result.getClear());
  4952. }
  4953. else
  4954. {
  4955. failReason.merge(reasonl);
  4956. failReason.merge(reasonr);
  4957. }
  4958. if (duplicate || !extracted)
  4959. matches.appendPostFilter(expr);
  4960. return extracted;
  4961. }
  4962. bool MonitorExtractor::extractOrFilter(KeyConditionInfo & matches, IHqlExpression * expr, KeyedKind keyedKind)
  4963. {
  4964. HqlExprArray conds;
  4965. expr->unwindList(conds, no_or);
  4966. bool validOrFilter = true;
  4967. HqlExprAttr invariant;
  4968. CIArrayOf<KeyConditionInfo> branches;
  4969. ForEachItemIn(idx, conds)
  4970. {
  4971. IHqlExpression & cur = conds.item(idx);
  4972. if (isIndexInvariant(&cur))
  4973. extendOrCondition(invariant, &cur);
  4974. else
  4975. {
  4976. KeyConditionInfo & branch = * new KeyConditionInfo;
  4977. branches.append(branch);
  4978. //Can't generate an OR with a pure post-filter
  4979. if (!extractFilters(branch, &cur, keyedKind))
  4980. validOrFilter = false;
  4981. }
  4982. }
  4983. //check all the conditions that are ORd together don't contain references to multiple fields.
  4984. KeyCondition * firstBranch = NULL;
  4985. bool multipleBranches = branches.ordinality() > 1;
  4986. bool multipleSelectors = false;
  4987. bool multipleConditions = false;
  4988. bool hasPostFilter = false;
  4989. ForEachItemIn(i1, branches)
  4990. {
  4991. KeyConditionInfo & branch = branches.item(i1);
  4992. if (branch.postFilter)
  4993. hasPostFilter = true;
  4994. ForEachItemIn(i2, branch.conditions)
  4995. {
  4996. KeyCondition & cur = branch.conditions.item(i2);
  4997. if (!firstBranch)
  4998. firstBranch = &cur;
  4999. else
  5000. {
  5001. multipleConditions = true;
  5002. if (i1 == 0)
  5003. {
  5004. if (firstBranch->selector != cur.selector)
  5005. multipleSelectors = true;
  5006. }
  5007. else
  5008. {
  5009. if ((firstBranch->selector != cur.selector) || multipleSelectors)
  5010. validOrFilter = false;
  5011. }
  5012. }
  5013. }
  5014. }
  5015. if (multipleBranches && hasPostFilter)
  5016. validOrFilter = false;
  5017. if (validOrFilter && firstBranch)
  5018. {
  5019. bool optimizeSingleBranch = true;
  5020. if (multipleSelectors || hasPostFilter || (optimizeSingleBranch && !multipleConditions))
  5021. {
  5022. //Invariant ored with a conjunction
  5023. //X or (A and B) -> (X or A) AND (X or B)
  5024. assertex(branches.ordinality() == 1);
  5025. KeyConditionInfo & branch = branches.item(0);
  5026. OwnedHqlExpr preFilter = branch.preFilter ? extendCondition(no_or, invariant, branch.preFilter) : NULL;
  5027. OwnedHqlExpr postFilter = branch.postFilter ? extendCondition(no_or, invariant, branch.postFilter) : NULL;
  5028. matches.appendPreFilter(preFilter);
  5029. matches.appendPostFilter(postFilter);
  5030. ForEachItemIn(i2, branch.conditions)
  5031. {
  5032. KeyCondition & cur = branch.conditions.item(i2);
  5033. OwnedHqlExpr filter = extendCondition(no_or, invariant, cur.expr);
  5034. matches.conditions.append(*new KeyCondition(cur.selector, filter, keyedKind));
  5035. }
  5036. }
  5037. else
  5038. {
  5039. LinkedHqlExpr combinedCondition = invariant;
  5040. ForEachItemIn(i1, branches)
  5041. {
  5042. KeyConditionInfo & branch = branches.item(i1);
  5043. OwnedHqlExpr conjunction = branch.createConjunction();
  5044. extendOrCondition(combinedCondition, conjunction);
  5045. }
  5046. matches.conditions.append(*new KeyCondition(firstBranch->selector, combinedCondition, keyedKind));
  5047. }
  5048. return true;
  5049. }
  5050. else
  5051. {
  5052. matches.appendPostFilter(expr);
  5053. KeyFailureInfo reason;
  5054. reason.set(KFRor);
  5055. failReason.merge(reason);
  5056. return false;
  5057. }
  5058. }
  5059. bool MonitorExtractor::extractIfFilter(KeyConditionInfo & matches, IHqlExpression * expr, KeyedKind keyedKind)
  5060. {
  5061. //MORE: This could generate better code, but I don't think it is worth the effort at the moment.
  5062. //Really, I should analyse left and right. Iterate each selector referenced. If there are no post conditions then
  5063. //generate IF(a, X, Y) compound expression, otherwise generate the default below.
  5064. IHqlExpression * cond = expr->queryChild(0);
  5065. if ((keyedKind != KeyedNo) && isIndexInvariant(cond))
  5066. {
  5067. //Convert IF(a, X, Y) to... IF (a, X, true) AND IF (a, true, Y) to... (NOT a OR X) AND (a OR Y)
  5068. OwnedHqlExpr inverseCond = getInverse(cond);
  5069. OwnedHqlExpr trueValue = createBoolExpr(no_or, LINK(inverseCond), LINK(expr->queryChild(1)));
  5070. OwnedHqlExpr falseValue = createBoolExpr(no_or, LINK(cond), LINK(expr->queryChild(2)));
  5071. OwnedHqlExpr combined = createBoolExpr(no_and, LINK(trueValue), LINK(falseValue));
  5072. return extractFilters(matches, combined, keyedKind);
  5073. }
  5074. matches.appendPostFilter(expr);
  5075. return false;
  5076. }
  5077. static HqlTransformerInfo selectSpotterInfo("SelectSpotter");
  5078. MonitorExtractor::SelectSpotter::SelectSpotter(const HqlExprArray & _selects) : NewHqlTransformer(selectSpotterInfo), selects(_selects)
  5079. {
  5080. hasSelects = false;
  5081. }
  5082. void MonitorExtractor::SelectSpotter::analyseExpr(IHqlExpression * expr)
  5083. {
  5084. if (hasSelects || alreadyVisited(expr))
  5085. return;
  5086. if (selects.find(*expr) != NotFound)
  5087. {
  5088. hasSelects = true;
  5089. return;
  5090. }
  5091. NewHqlTransformer::analyseExpr(expr);
  5092. }
  5093. bool MonitorExtractor::containsTableSelects(IHqlExpression * expr)
  5094. {
  5095. HqlExprCopyArray inScope;
  5096. expr->gatherTablesUsed(NULL, &inScope);
  5097. //Check that cursors for all inScope tables are already bound in the start context
  5098. return inScope.find(*tableExpr->queryNormalizedSelector()) != NotFound;
  5099. }
  5100. void MonitorExtractor::extractFilters(IHqlExpression * expr, SharedHqlExpr & extraFilter)
  5101. {
  5102. HqlExprArray conds;
  5103. expr->unwindList(conds, no_and);
  5104. extractFilters(conds, extraFilter);
  5105. }
  5106. void MonitorExtractor::extractFilters(HqlExprArray & exprs, SharedHqlExpr & extraFilter)
  5107. {
  5108. OwnedHqlExpr savedFilter = keyed.postFilter.getClear();
  5109. ForEachItemIn(i1, exprs)
  5110. {
  5111. IHqlExpression & cur = exprs.item(i1);
  5112. switch (cur.getOperator())
  5113. {
  5114. case no_assertkeyed:
  5115. case no_assertwild:
  5116. extractFilters(keyed, &cur, KeyedNo);
  5117. break;
  5118. }
  5119. }
  5120. keyedExplicitly = (keyed.conditions.ordinality() != 0);
  5121. cleanlyKeyedExplicitly = keyedExplicitly && !keyed.postFilter;
  5122. ForEachItemIn(i2, exprs)
  5123. {
  5124. IHqlExpression & cur = exprs.item(i2);
  5125. switch (cur.getOperator())
  5126. {
  5127. case no_assertkeyed:
  5128. case no_assertwild:
  5129. break;
  5130. default:
  5131. if (!keyedExplicitly)
  5132. extractFilters(keyed, &cur, KeyedNo);
  5133. else if (!cur.isAttribute() && isIndexInvariant(&cur))
  5134. keyed.appendPreFilter(&cur);
  5135. else
  5136. keyed.appendPostFilter(&cur);
  5137. break;
  5138. }
  5139. }
  5140. extraFilter.set(keyed.postFilter);
  5141. keyed.postFilter.setown(extendConditionOwn(no_and, savedFilter.getClear(), LINK(extraFilter)));
  5142. }
  5143. void MonitorExtractor::extractFiltersFromFilterDs(IHqlExpression * expr)
  5144. {
  5145. HqlExprArray conds;
  5146. HqlExprAttr dummy;
  5147. unwindFilterConditions(conds, expr);
  5148. extractFilters(conds, dummy);
  5149. }
  5150. void MonitorExtractor::extractFoldedWildFields(IHqlExpression * expr)
  5151. {
  5152. node_operator op = expr->getOperator();
  5153. switch (op)
  5154. {
  5155. case no_cast:
  5156. case no_implicitcast:
  5157. case no_add:
  5158. case no_sub:
  5159. //fields may have been transformed since folding...
  5160. extractFoldedWildFields(expr->queryChild(0));
  5161. break;
  5162. case no_select:
  5163. if (isKeySelect(expr))
  5164. {
  5165. KeyCondition * condition = new KeyCondition;
  5166. condition->selector.set(expr);
  5167. condition->isWild = true;
  5168. condition->wasKeyed = true;
  5169. keyed.conditions.append(*condition);
  5170. }
  5171. break;
  5172. }
  5173. }
  5174. bool MonitorExtractor::extractBoolFieldFilter(KeyConditionInfo & matches, IHqlExpression * selector, KeyedKind keyedKind, bool compareValue)
  5175. {
  5176. if (selector->isBoolean())
  5177. {
  5178. if (isKeySelect(selector) && okToKey(selector, keyedKind))
  5179. {
  5180. OwnedHqlExpr newFilter = createValue(no_eq, makeBoolType(), LINK(selector), createConstant(compareValue));
  5181. matches.appendCondition(*new KeyCondition(selector, newFilter, keyedKind));
  5182. return true;
  5183. }
  5184. }
  5185. return false;
  5186. }
  5187. bool MonitorExtractor::extractFilters(KeyConditionInfo & matches, IHqlExpression * expr, KeyedKind keyedKind)
  5188. {
  5189. if (!expr->isAttribute() && isIndexInvariant(expr))
  5190. {
  5191. extendAndCondition(matches.preFilter, expr);
  5192. return true;
  5193. }
  5194. IHqlExpression *l = expr->queryChild(0);
  5195. IHqlExpression *r = expr->queryChild(1);
  5196. node_operator op = expr->getOperator();
  5197. switch (op)
  5198. {
  5199. case no_and:
  5200. {
  5201. bool extracted = extractFilters(matches, l, keyedKind);
  5202. if (!extractFilters(matches, r, keyedKind)) extracted = false;
  5203. return extracted;
  5204. }
  5205. case no_or:
  5206. return extractOrFilter(matches, expr, keyedKind);
  5207. case no_attr:
  5208. case no_attr_expr:
  5209. case no_attr_link:
  5210. return true;
  5211. case no_not:
  5212. {
  5213. IHqlExpression * arg = expr->queryChild(0);
  5214. OwnedHqlExpr inverse = getInverse(arg);
  5215. if (inverse->queryBody() != expr->queryBody())
  5216. return extractFilters(matches, inverse, keyedKind);
  5217. if ((arg->getOperator() == no_select) && arg->isBoolean() &&
  5218. extractBoolFieldFilter(matches, arg, keyedKind, false))
  5219. return true;
  5220. matches.appendPostFilter(expr);
  5221. return false;
  5222. }
  5223. case no_between:
  5224. case no_notbetween:
  5225. {
  5226. //Convert this into two comparisons because that will handle weird boundary conditions much better.
  5227. OwnedHqlExpr normalized = expandBetween(expr);
  5228. return extractFilters(matches, normalized, keyedKind);
  5229. }
  5230. case no_eq:
  5231. case no_ne:
  5232. {
  5233. bool duplicate = false;
  5234. if (matchSubstringFilter(matches, op, l, r, keyedKind, duplicate) || matchSubstringFilter(matches, op, r, l, keyedKind, duplicate))
  5235. {
  5236. if (duplicate)
  5237. matches.appendPostFilter(expr);
  5238. return true;
  5239. }
  5240. return extractSimpleCompareFilter(matches, expr, keyedKind);
  5241. }
  5242. case no_in:
  5243. case no_notin:
  5244. {
  5245. bool duplicate = false;
  5246. if (matchSubstringFilter(matches, op, l, r, keyedKind, duplicate))
  5247. {
  5248. if (duplicate)
  5249. matches.appendPostFilter(expr);
  5250. return true;
  5251. }
  5252. return extractSimpleCompareFilter(matches, expr, keyedKind);
  5253. }
  5254. case no_gt:
  5255. case no_lt:
  5256. case no_ge:
  5257. case no_le:
  5258. return extractSimpleCompareFilter(matches, expr, keyedKind);
  5259. case no_assertkeyed:
  5260. {
  5261. KeyFailureInfo reason;
  5262. reason.merge(failReason);
  5263. failReason.clear();
  5264. bool extend = expr->hasAttribute(extendAtom);
  5265. if (!extractFilters(matches, l, extend ? KeyedExtend : KeyedYes))
  5266. {
  5267. if (!extend)
  5268. failReason.reportError(translator, expr);
  5269. }
  5270. IHqlExpression * original = expr->queryAttribute(_selectors_Atom);
  5271. if (original)
  5272. {
  5273. ForEachChild(i, original)
  5274. extractFoldedWildFields(original->queryChild(i));
  5275. }
  5276. failReason.merge(reason);
  5277. return true;
  5278. }
  5279. case no_assertwild:
  5280. {
  5281. if (l->getOperator() == no_all)
  5282. {
  5283. IHqlExpression * original = expr->queryAttribute(_selectors_Atom);
  5284. assertex(original);
  5285. ForEachChild(i, original)
  5286. extractFoldedWildFields(original->queryChild(i));
  5287. }
  5288. else
  5289. {
  5290. IHqlExpression * selector = queryKeyableSelector(l);
  5291. if (!selector)
  5292. {
  5293. StringBuffer keyname;
  5294. translator.throwError1(HQLERR_WildNotReferenceIndex, queryKeyName(keyname));
  5295. }
  5296. KeyCondition * condition = new KeyCondition;
  5297. condition->selector.set(selector);
  5298. condition->isWild = true;
  5299. matches.appendCondition(*condition);
  5300. }
  5301. return true;
  5302. }
  5303. case no_if:
  5304. return extractIfFilter(matches, expr, keyedKind);
  5305. case no_select:
  5306. {
  5307. if (expr->isBoolean() && extractBoolFieldFilter(matches, expr, keyedKind, true))
  5308. return true;
  5309. matches.appendPostFilter(expr);
  5310. return false;
  5311. }
  5312. default:
  5313. // Add this condition to the catchall expr
  5314. matches.appendPostFilter(expr);
  5315. return false;
  5316. }
  5317. }
  5318. void MonitorExtractor::extractAllFilters(IHqlExpression * dataset)
  5319. {
  5320. loop
  5321. {
  5322. switch (dataset->getOperator())
  5323. {
  5324. case no_newkeyindex:
  5325. return;
  5326. case no_filter:
  5327. extractAllFilters(dataset->queryChild(0));
  5328. extractFiltersFromFilterDs(dataset);
  5329. return;
  5330. case no_compound_indexread:
  5331. case no_newusertable:
  5332. case no_hqlproject:
  5333. case no_distributed:
  5334. case no_preservemeta:
  5335. case no_unordered:
  5336. case no_sorted:
  5337. case no_stepped:
  5338. case no_grouped:
  5339. case no_alias_scope:
  5340. case no_dataset_alias:
  5341. break;
  5342. default:
  5343. UNIMPLEMENTED;
  5344. }
  5345. dataset = dataset->queryChild(0);
  5346. }
  5347. }
  5348. bool MonitorExtractor::isKeyed()
  5349. {
  5350. ForEachItemIn(i, keyed.conditions)
  5351. {
  5352. if (!keyed.conditions.item(i).isWild)
  5353. return true;
  5354. }
  5355. return false;
  5356. }
  5357. bool expandFilename(StringBuffer & s, IHqlExpression * expr)
  5358. {
  5359. switch (expr->getOperator())
  5360. {
  5361. case no_constant:
  5362. expr->toString(s);
  5363. return true;
  5364. case no_getresult:
  5365. //more
  5366. break;
  5367. case no_concat:
  5368. {
  5369. bool hadString = expandFilename(s, expr->queryChild(0));
  5370. unsigned oldLength = s.length();
  5371. s.append("+");
  5372. if (expandFilename(s, expr->queryChild(1)) || hadString)
  5373. return true;
  5374. s.setLength(oldLength);
  5375. return false;
  5376. }
  5377. case no_alias:
  5378. case no_cast:
  5379. case no_implicitcast:
  5380. return expandFilename(s, expr->queryChild(0));
  5381. }
  5382. if (hasNamedSymbol(expr))
  5383. {
  5384. s.append(expr->queryName());
  5385. return true;
  5386. }
  5387. s.append("...");
  5388. return false;
  5389. }
  5390. const char * MonitorExtractor::queryKeyName(StringBuffer & s)
  5391. {
  5392. IAtom * name = tableExpr->queryName();
  5393. if (name)
  5394. s.append(" \'").append(name).append("'");
  5395. else
  5396. {
  5397. IHqlExpression * filename = queryTableFilename(tableExpr);
  5398. if (filename)
  5399. {
  5400. if (!expandFilename(s.append(' '), filename))
  5401. s.clear();
  5402. }
  5403. }
  5404. return s.str();
  5405. }
  5406. static bool isNextField(IHqlExpression * record, IHqlExpression * prevExpr, IHqlExpression * nextExpr)
  5407. {
  5408. if ((prevExpr->getOperator() != no_select) || (nextExpr->getOperator() != no_select))
  5409. return false;
  5410. IHqlExpression * prevSelect = prevExpr->queryChild(0);
  5411. if (prevSelect != nextExpr->queryChild(0))
  5412. return false;
  5413. if (!containsOnlyLeft(prevExpr))
  5414. return false;
  5415. if (prevSelect->getOperator() != no_left)
  5416. record = prevSelect->queryRecord();
  5417. IHqlExpression * prevField = prevExpr->queryChild(1);
  5418. IHqlExpression * nextField = nextExpr->queryChild(1);
  5419. //Slow, but probably doesn't matter...
  5420. ForEachChild(i, record)
  5421. {
  5422. IHqlExpression * cur = record->queryChild(i);
  5423. if (cur == prevField)
  5424. return (record->queryChild(i+1) == nextField);
  5425. }
  5426. return false;
  5427. }
  5428. IHqlExpression * MonitorExtractor::querySimpleJoinValue(IHqlExpression * selector)
  5429. {
  5430. IHqlExpression * matched = NULL;
  5431. ForEachItemIn(cond, keyed.conditions)
  5432. {
  5433. KeyCondition & cur = keyed.conditions.item(cond);
  5434. if (cur.selector == selector)
  5435. {
  5436. if (!cur.isWild)
  5437. {
  5438. if (matched)
  5439. return NULL;
  5440. IHqlExpression * matchExpr = cur.expr;
  5441. if (matchExpr->getOperator() != no_eq)
  5442. return NULL;
  5443. if (matchExpr->queryChild(0) != selector)
  5444. return NULL;
  5445. matched = matchExpr->queryChild(1);
  5446. }
  5447. }
  5448. }
  5449. return matched;
  5450. }
  5451. void MonitorExtractor::optimizeSegments(IHqlExpression * leftRecord)
  5452. {
  5453. //loop to see if we have matches for key.cpt[n] = x.field[n] and key.cpt[n+1] = x.field[n+1]
  5454. //where fields are fixed length, no casts and no modifiers.
  5455. //if so, mark the new total size,
  5456. //when generate, extend the size of the first monitor, and skip the others.
  5457. //MORE: Could also combine fixed constants, but less of an advantage.
  5458. //Don't process anything after a variable size field/something that is transformed.
  5459. unsigned i = 0;
  5460. for (; i < firstOffsetField; i++)
  5461. {
  5462. IHqlExpression * keySelector = &keyableSelects.item(i);
  5463. unsigned mergedSize = keySelector->queryType()->getSize();
  5464. IHqlExpression * prevValue = querySimpleJoinValue(keySelector);
  5465. unsigned best = i;
  5466. if (prevValue && isSameBasicType(keySelector->queryType(), prevValue->queryType()))
  5467. {
  5468. for (unsigned j = i+1; j < firstOffsetField; j++)
  5469. {
  5470. IHqlExpression * nextSelector = &keyableSelects.item(j);
  5471. if (noMergeSelects.contains(*nextSelector))
  5472. break;
  5473. IHqlExpression * nextValue = querySimpleJoinValue(nextSelector);
  5474. if (!nextValue || !isNextField(leftRecord, prevValue, nextValue) ||
  5475. !isSameBasicType(nextSelector->queryType(), nextValue->queryType()))
  5476. break;
  5477. prevValue = nextValue;
  5478. mergedSize += nextSelector->queryType()->getSize();
  5479. best = j;
  5480. }
  5481. }
  5482. mergedSizes.append(mergedSize);
  5483. for (;i < best;i++)
  5484. mergedSizes.append(0);
  5485. }
  5486. while ( i < numKeyableFields)
  5487. mergedSizes.append(expandedSelects.item(i).queryType()->getSize());
  5488. }
  5489. //---------------------------------------------------------------------------
  5490. static ABoundActivity * buildNullIndexActivity(HqlCppTranslator & translator, BuildCtx & ctx, IHqlExpression * expr)
  5491. {
  5492. while (isCompoundSource(expr))
  5493. expr = expr->queryChild(0);
  5494. return translator.buildCachedActivity(ctx, expr);
  5495. }
  5496. class IndexReadBuilderBase : public SourceBuilder
  5497. {
  5498. friend class MonitorRemovalTransformer;
  5499. public:
  5500. IndexReadBuilderBase(HqlCppTranslator & _translator, IHqlExpression *_tableExpr, IHqlExpression *_nameExpr)
  5501. : SourceBuilder(_translator, _tableExpr, _nameExpr), monitors(_tableExpr, _translator, -(int)numPayloadFields(_tableExpr), false)
  5502. {
  5503. fpos.setown(getFilepos(tableExpr, false));
  5504. lfpos.setown(getFilepos(tableExpr, true));
  5505. }
  5506. virtual void buildMembers(IHqlExpression * expr);
  5507. virtual void extractMonitors(IHqlExpression * ds, SharedHqlExpr & unkeyedFilter, HqlExprArray & conds);
  5508. protected:
  5509. virtual void buildFlagsMember(IHqlExpression * expr);
  5510. virtual void buildTransformFpos(BuildCtx & transformCtx)
  5511. {
  5512. associateFilePositions(transformCtx, "fpp", "left");
  5513. }
  5514. IHqlExpression * removeMonitors(IHqlExpression * expr);
  5515. protected:
  5516. MonitorExtractor monitors;
  5517. SourceSteppingInfo steppingInfo;
  5518. };
  5519. void IndexReadBuilderBase::buildMembers(IHqlExpression * expr)
  5520. {
  5521. //---- virtual void createSegmentMonitors(struct IIndexReadContext *) { ... } ----
  5522. BuildCtx createSegmentCtx(instance->startctx);
  5523. createSegmentCtx.addQuotedCompoundLiteral("virtual void createSegmentMonitors(IIndexReadContext *irc)");
  5524. monitors.buildSegments(createSegmentCtx, "irc", false);
  5525. buildLimits(instance->startctx, expr, instance->activityId);
  5526. if (!limitExpr && !keyedLimitExpr && !choosenValue && (instance->kind == TAKindexread || instance->kind == TAKindexnormalize) && !steppedExpr)
  5527. {
  5528. unsigned implicitLimit = translator.getDefaultImplicitIndexReadLimit();
  5529. if (translator.checkIndexReadLimit())
  5530. {
  5531. StringBuffer keyname;
  5532. if (implicitLimit)
  5533. translator.WARNINGAT2(CategoryLimit, queryLocation(expr), HQLWRN_ImplicitReadAddLimit, implicitLimit, monitors.queryKeyName(keyname));
  5534. else
  5535. translator.WARNINGAT1(CategoryLimit, queryLocation(expr), HQLWRN_ImplicitReadLimit, monitors.queryKeyName(keyname));
  5536. }
  5537. if (implicitLimit)
  5538. {
  5539. OwnedHqlExpr limit = getSizetConstant(implicitLimit);
  5540. translator.buildLimitHelpers(instance->startctx, limit, NULL, false, nameExpr, instance->activityId);
  5541. }
  5542. }
  5543. instance->addAttributeBool("preload", isPreloaded);
  5544. if (translator.getTargetClusterType() == RoxieCluster)
  5545. instance->addAttributeBool("_isIndexOpt", tableExpr->hasAttribute(optAtom));
  5546. if (monitors.queryGlobalGuard())
  5547. translator.doBuildBoolFunction(instance->startctx, "canMatchAny", monitors.queryGlobalGuard());
  5548. buildKeyedLimitHelper(expr);
  5549. translator.buildFormatCrcFunction(instance->classctx, "getFormatCrc", tableExpr, tableExpr, 1);
  5550. IHqlExpression * originalKey = queryAttributeChild(tableExpr, _original_Atom, 0);
  5551. translator.buildSerializedLayoutMember(instance->classctx, originalKey->queryRecord(), "getIndexLayout", numKeyedFields(originalKey));
  5552. //Note the helper base class contains code like the following
  5553. //IThorIndexCallback * fpp;");
  5554. //virtual void setCallback(IThorIndexCallback * _tc) { fpp = _tc; }");
  5555. }
  5556. void IndexReadBuilderBase::buildFlagsMember(IHqlExpression * expr)
  5557. {
  5558. StringBuffer flags;
  5559. if (tableExpr->hasAttribute(sortedAtom))
  5560. flags.append("|TIRsorted");
  5561. else if (!isOrdered(tableExpr))
  5562. flags.append("|TIRunordered");
  5563. if (!monitors.isFiltered())
  5564. flags.append("|TIRnofilter");
  5565. if (isPreloaded)
  5566. flags.append("|TIRpreload");
  5567. if (tableExpr->hasAttribute(optAtom))
  5568. flags.append("|TIRoptional");
  5569. if (limitExpr && limitExpr->hasAttribute(skipAtom))
  5570. flags.append("|TIRlimitskips");
  5571. if (limitExpr && limitExpr->hasAttribute(onFailAtom))
  5572. flags.append("|TIRlimitcreates");
  5573. if (generateUnfilteredTransform)
  5574. flags.append("|TIRunfilteredtransform");
  5575. if (keyedLimitExpr)
  5576. {
  5577. if (keyedLimitExpr->hasAttribute(onFailAtom))
  5578. flags.append("|TIRkeyedlimitcreates|TIRcountkeyedlimit");
  5579. else if (keyedLimitExpr->hasAttribute(skipAtom))
  5580. flags.append("|TIRkeyedlimitskips|TIRcountkeyedlimit");
  5581. else if (keyedLimitExpr->hasAttribute(countAtom))
  5582. flags.append("|TIRcountkeyedlimit");
  5583. }
  5584. IHqlExpression * firstStepped = steppingInfo.firstStepped();
  5585. if (firstStepped && monitors.isEqualityFilterBefore(firstStepped))
  5586. flags.append("|TIRstepleadequality");
  5587. if (onlyExistsAggreate) flags.append("|TIRaggregateexists");
  5588. if (monitorsForGrouping) flags.append("|TIRgroupmonitors");
  5589. if (!nameExpr->isConstant()) flags.append("|TIRvarfilename");
  5590. if (translator.hasDynamicFilename(tableExpr)) flags.append("|TIRdynamicfilename");
  5591. if (requiresOrderedMerge) flags.append("|TIRorderedmerge");
  5592. if (flags.length())
  5593. translator.doBuildUnsignedFunction(instance->classctx, "getFlags", flags.str()+1);
  5594. }
  5595. void IndexReadBuilderBase::extractMonitors(IHqlExpression * ds, SharedHqlExpr & unkeyedFilter, HqlExprArray & conds)
  5596. {
  5597. OwnedHqlExpr extraFilter;
  5598. monitors.extractFilters(conds, extraFilter);
  5599. appendFilter(unkeyedFilter, extraFilter);
  5600. }
  5601. class MonitorRemovalTransformer : public HqlMapTransformer
  5602. {
  5603. public:
  5604. MonitorRemovalTransformer(IndexReadBuilderBase & _builder) : builder(_builder) {}
  5605. virtual IHqlExpression * createTransformed(IHqlExpression * expr);
  5606. protected:
  5607. IndexReadBuilderBase & builder;
  5608. };
  5609. IHqlExpression * MonitorRemovalTransformer::createTransformed(IHqlExpression * expr)
  5610. {
  5611. switch (expr->getOperator())
  5612. {
  5613. case no_filter:
  5614. {
  5615. IHqlExpression * ds = expr->queryChild(0);
  5616. IHqlExpression * body = expr->queryBody();
  5617. unsigned match = builder.originalFilters.find(*body);
  5618. if (match == NotFound)
  5619. {
  5620. if (builder.removedFilters.find(*body))
  5621. return transform(ds);
  5622. return NewHqlTransformer::createTransformed(expr);
  5623. }
  5624. IHqlExpression & newFilter = builder.mappedFilters.item(match);
  5625. HqlExprArray args;
  5626. args.append(*transform(ds));
  5627. args.append(*transform(&newFilter));
  5628. return expr->clone(args);
  5629. }
  5630. }
  5631. return NewHqlTransformer::createTransformed(expr);
  5632. }
  5633. IHqlExpression * IndexReadBuilderBase::removeMonitors(IHqlExpression * expr)
  5634. {
  5635. MonitorRemovalTransformer mapper(*this);
  5636. mapper.setMapping(tableExpr, tableExpr);
  5637. return mapper.transformRoot(expr);
  5638. }
  5639. //---------------------------------------------------------------------------
  5640. class NewIndexReadBuilder : public IndexReadBuilderBase
  5641. {
  5642. public:
  5643. NewIndexReadBuilder(HqlCppTranslator & _translator, IHqlExpression *_tableExpr, IHqlExpression *_nameExpr)
  5644. : IndexReadBuilderBase(_translator, _tableExpr, _nameExpr)
  5645. {
  5646. }
  5647. virtual void analyseGraph(IHqlExpression * expr)
  5648. {
  5649. IndexReadBuilderBase::analyseGraph(expr);
  5650. gatherSteppingMeta(expr, steppingInfo);
  5651. if (steppedExpr && transformCanFilter && translator.queryOptions().optimizeSteppingPostfilter)
  5652. {
  5653. //If sorted, but can't step output no point in duplicating the transform
  5654. if (steppingInfo.outputStepping.exists())
  5655. generateUnfilteredTransform = true;
  5656. }
  5657. returnIfFilterFails = !isNormalize;
  5658. }
  5659. virtual void buildTransform(IHqlExpression * expr);
  5660. virtual void buildMembers(IHqlExpression * expr);
  5661. };
  5662. void NewIndexReadBuilder::buildMembers(IHqlExpression * expr)
  5663. {
  5664. if (steppingInfo.exists())
  5665. {
  5666. steppingInfo.checkKeyable(monitors);
  5667. monitors.preventMerge(steppingInfo.firstStepped());
  5668. }
  5669. buildReadMembers(expr);
  5670. if (steppingInfo.exists())
  5671. steppingInfo.generateMembers(translator, instance->classctx);
  5672. IndexReadBuilderBase::buildMembers(expr);
  5673. }
  5674. void NewIndexReadBuilder::buildTransform(IHqlExpression * expr)
  5675. {
  5676. if (true)
  5677. {
  5678. BuildCtx transformCtx(instance->startctx);
  5679. transformCtx.addQuotedCompoundLiteral("virtual size32_t transform(ARowBuilder & crSelf, const void * _left)");
  5680. translator.ensureRowAllocated(transformCtx, "crSelf");
  5681. transformCtx.addQuotedLiteral("unsigned char * left = (unsigned char *)_left;");
  5682. translator.associateBlobHelper(transformCtx, tableExpr, "fpp");
  5683. buildTransformBody(transformCtx, expr, true, false, true);
  5684. }
  5685. if (generateUnfilteredTransform)
  5686. {
  5687. BuildCtx transformCtx(instance->startctx);
  5688. transformCtx.addQuotedCompoundLiteral("virtual size32_t unfilteredTransform(ARowBuilder & crSelf, const void * _left)");
  5689. translator.ensureRowAllocated(transformCtx, "crSelf");
  5690. transformCtx.addQuotedLiteral("unsigned char * left = (unsigned char *)_left;");
  5691. translator.associateBlobHelper(transformCtx, tableExpr, "fpp");
  5692. buildTransformBody(transformCtx, expr, true, true, true);
  5693. }
  5694. }
  5695. //---------------------------------------------------------------------------
  5696. ABoundActivity * HqlCppTranslator::doBuildActivityIndexRead(BuildCtx & ctx, IHqlExpression * expr)
  5697. {
  5698. OwnedHqlExpr transformed = buildIndexFromPhysical(expr);
  5699. OwnedHqlExpr optimized = optimizeHqlExpression(queryErrorProcessor(), transformed, HOOfold);
  5700. IHqlExpression *tableExpr = queryPhysicalRootTable(optimized);
  5701. //If the filter is false, then it may get reduced to a NULL operation!
  5702. if (!tableExpr)
  5703. return buildNullIndexActivity(*this, ctx, optimized);
  5704. if (optimized->getOperator() != no_compound_indexread)
  5705. optimized.setown(createDataset(no_compound_indexread, LINK(optimized)));
  5706. traceExpression("before index read", optimized);
  5707. assertex(tableExpr->getOperator() == no_newkeyindex);
  5708. NewIndexReadBuilder info(*this, tableExpr, tableExpr->queryChild(3));
  5709. info.gatherVirtualFields(true, true);
  5710. if (info.containsStepping(optimized))
  5711. return info.buildActivity(ctx, optimized, TAKindexread, "SteppedIndexRead", NULL);
  5712. return info.buildActivity(ctx, optimized, TAKindexread, "IndexRead", NULL);
  5713. }
  5714. //---------------------------------------------------------------------------
  5715. class IndexNormalizeBuilder : public IndexReadBuilderBase
  5716. {
  5717. public:
  5718. IndexNormalizeBuilder(HqlCppTranslator & _translator, IHqlExpression *_tableExpr, IHqlExpression *_nameExpr)
  5719. : IndexReadBuilderBase(_translator, _tableExpr, _nameExpr)
  5720. {
  5721. }
  5722. virtual void buildTransform(IHqlExpression * expr);
  5723. virtual void buildMembers(IHqlExpression * expr);
  5724. protected:
  5725. virtual void analyseGraph(IHqlExpression * expr);
  5726. virtual void processTransformSelect(BuildCtx & ctx, IHqlExpression * expr)
  5727. {
  5728. doBuildNormalizeIterators(ctx, expr, false);
  5729. }
  5730. };
  5731. void IndexNormalizeBuilder::analyseGraph(IHqlExpression * expr)
  5732. {
  5733. IndexReadBuilderBase::analyseGraph(expr);
  5734. needDefaultTransform = (expr->queryNormalizedSelector()->getOperator() == no_select);
  5735. }
  5736. void IndexNormalizeBuilder::buildMembers(IHqlExpression * expr)
  5737. {
  5738. buildFilenameMember();
  5739. IndexReadBuilderBase::buildMembers(expr);
  5740. buildNormalizeHelpers(expr);
  5741. }
  5742. void IndexNormalizeBuilder::buildTransform(IHqlExpression * expr)
  5743. {
  5744. globaliterctx.setown(new BuildCtx(instance->startctx));
  5745. globaliterctx->addGroup();
  5746. BuildCtx transformCtx(instance->startctx);
  5747. transformCtx.addQuotedCompoundLiteral("virtual size32_t transform(ARowBuilder & crSelf)");
  5748. translator.ensureRowAllocated(transformCtx, "crSelf");
  5749. //Because this transform creates iterator classes for the child iterators the expression tree needs to be modified
  5750. //instead of using an inline tests. We could switch to using this all the time for indexes once I trust it!
  5751. OwnedHqlExpr simplified = removeMonitors(expr);
  5752. lastTransformer.set(queryExpression(simplified->queryDataset()->queryTable()));
  5753. useFilterMappings=false;
  5754. buildTransformBody(transformCtx, simplified, true, false, false);
  5755. }
  5756. //---------------------------------------------------------------------------
  5757. ABoundActivity * HqlCppTranslator::doBuildActivityIndexNormalize(BuildCtx & ctx, IHqlExpression * expr)
  5758. {
  5759. OwnedHqlExpr transformed = buildIndexFromPhysical(expr);
  5760. OwnedHqlExpr optimized = optimizeHqlExpression(queryErrorProcessor(), transformed, HOOfold);
  5761. traceExpression("after optimize", optimized);
  5762. IHqlExpression *tableExpr = queryPhysicalRootTable(optimized);
  5763. if (!tableExpr)
  5764. return buildNullIndexActivity(*this, ctx, optimized);
  5765. assertex(tableExpr->getOperator() == no_newkeyindex);
  5766. IndexNormalizeBuilder info(*this, tableExpr, tableExpr->queryChild(3));
  5767. info.gatherVirtualFields(true, true);
  5768. return info.buildActivity(ctx, optimized, TAKindexnormalize, "IndexNormalize", NULL);
  5769. }
  5770. //---------------------------------------------------------------------------
  5771. class IndexAggregateBuilder : public IndexReadBuilderBase
  5772. {
  5773. public:
  5774. IndexAggregateBuilder(HqlCppTranslator & _translator, IHqlExpression *_tableExpr, IHqlExpression *_nameExpr)
  5775. : IndexReadBuilderBase(_translator, _tableExpr, _nameExpr)
  5776. {
  5777. failedFilterValue.clear();
  5778. }
  5779. virtual void buildTransform(IHqlExpression * expr);
  5780. virtual void buildMembers(IHqlExpression * expr);
  5781. protected:
  5782. virtual void processTransformSelect(BuildCtx & ctx, IHqlExpression * expr)
  5783. {
  5784. doBuildAggregateSelectIterator(ctx, expr);
  5785. }
  5786. virtual void analyseGraph(IHqlExpression * expr)
  5787. {
  5788. IndexReadBuilderBase::analyseGraph(expr);
  5789. returnIfFilterFails = !isNormalize;
  5790. }
  5791. };
  5792. void IndexAggregateBuilder::buildMembers(IHqlExpression * expr)
  5793. {
  5794. StringBuffer s;
  5795. buildFilenameMember();
  5796. IndexReadBuilderBase::buildMembers(expr);
  5797. buildAggregateHelpers(expr);
  5798. //virtual void processRow(void * self, const void * src) = 0;
  5799. BuildCtx rowctx(instance->startctx);
  5800. rowctx.addQuotedCompoundLiteral("virtual void processRow(ARowBuilder & crSelf, const void * src)");
  5801. rowctx.addQuotedLiteral("doProcessRow(crSelf, (byte *)src);");
  5802. //virtual void processRows(void * self, size32_t srcLen, const void * src) = 0;
  5803. BuildCtx rowsctx(instance->startctx);
  5804. rowsctx.addQuotedCompoundLiteral("virtual void processRows(ARowBuilder & crSelf, size32_t srcLen, const void * _left)");
  5805. rowsctx.addQuotedLiteral("unsigned char * left = (unsigned char *)_left;");
  5806. OwnedHqlExpr ds = createVariable("left", makeReferenceModifier(tableExpr->getType()));
  5807. OwnedHqlExpr len = createVariable("srcLen", LINK(sizetType));
  5808. OwnedHqlExpr fullDs = createTranslated(ds, len);
  5809. Owned<IHqlCppDatasetCursor> iter = translator.createDatasetSelector(rowsctx, fullDs);
  5810. BoundRow * curRow = iter->buildIterateLoop(rowsctx, false);
  5811. s.clear().append("doProcessRow(crSelf, ");
  5812. translator.generateExprCpp(s, curRow->queryBound());
  5813. s.append(");");
  5814. rowsctx.addQuoted(s);
  5815. }
  5816. void IndexAggregateBuilder::buildTransform(IHqlExpression * expr)
  5817. {
  5818. BuildCtx transformCtx(instance->startctx);
  5819. transformCtx.addQuotedCompoundLiteral("void doProcessRow(ARowBuilder & crSelf, byte * left)");
  5820. translator.ensureRowAllocated(transformCtx, "crSelf");
  5821. translator.associateBlobHelper(transformCtx, tableExpr, "fpp");
  5822. buildTransformBody(transformCtx, expr, false, false, true);
  5823. }
  5824. //---------------------------------------------------------------------------
  5825. class IndexCountBuilder : public IndexReadBuilderBase
  5826. {
  5827. public:
  5828. IndexCountBuilder(HqlCppTranslator & _translator, IHqlExpression *_tableExpr, IHqlExpression *_nameExpr, node_operator _aggOp)
  5829. : IndexReadBuilderBase(_translator, _tableExpr, _nameExpr)
  5830. {
  5831. aggOp = _aggOp;
  5832. isCompoundCount = true;
  5833. failedFilterValue.set(queryZero());
  5834. }
  5835. virtual void buildTransform(IHqlExpression * expr);
  5836. virtual void buildMembers(IHqlExpression * expr);
  5837. virtual bool isExplicitExists() { return (aggOp == no_existsgroup); }
  5838. protected:
  5839. virtual void processTransformSelect(BuildCtx & ctx, IHqlExpression * expr)
  5840. {
  5841. doBuildAggregateSelectIterator(ctx, expr);
  5842. }
  5843. virtual void analyseGraph(IHqlExpression * expr)
  5844. {
  5845. IndexReadBuilderBase::analyseGraph(expr);
  5846. returnIfFilterFails = !isNormalize;
  5847. IHqlExpression * aggregate = expr->queryChild(0);
  5848. if (isKeyedCountAggregate(aggregate))
  5849. {
  5850. if (isNormalize)
  5851. translator.throwError(HQLERR_KeyedCountCantNormalize);
  5852. if (!monitors.isKeyedExplicitly())
  5853. translator.throwError(HQLERR_KeyedCountNotKeyed);
  5854. if (transformCanFilter)
  5855. {
  5856. ForEachItemIn(i, originalFilters)
  5857. removedFilters.append(OLINK(originalFilters.item(i)));
  5858. originalFilters.kill();
  5859. mappedFilters.kill();
  5860. transformCanFilter = false;
  5861. }
  5862. }
  5863. if (aggOp == no_existsgroup)
  5864. choosenValue.setown(getSizetConstant(1));
  5865. }
  5866. protected:
  5867. node_operator aggOp;
  5868. };
  5869. void IndexCountBuilder::buildMembers(IHqlExpression * expr)
  5870. {
  5871. buildFilenameMember();
  5872. IndexReadBuilderBase::buildMembers(expr);
  5873. buildCountHelpers(expr, false);
  5874. }
  5875. void IndexCountBuilder::buildTransform(IHqlExpression * expr)
  5876. {
  5877. if (transformCanFilter||isNormalize)
  5878. {
  5879. BuildCtx transformCtx(instance->startctx);
  5880. transformCtx.addQuotedCompoundLiteral("virtual size32_t numValid(const void * _left)");
  5881. transformCtx.addQuotedLiteral("unsigned char * left = (unsigned char *)_left;");
  5882. translator.associateBlobHelper(transformCtx, tableExpr, "fpp");
  5883. OwnedHqlExpr cnt;
  5884. if (isNormalize)
  5885. {
  5886. compoundCountVar.setown(transformCtx.getTempDeclare(sizetType, queryZero()));
  5887. cnt.set(compoundCountVar);
  5888. }
  5889. else
  5890. cnt.setown(getSizetConstant(1));
  5891. BuildCtx subctx(transformCtx);
  5892. buildTransformBody(subctx, expr, false, false, true);
  5893. transformCtx.addReturn(cnt);
  5894. }
  5895. }
  5896. //---------------------------------------------------------------------------
  5897. ABoundActivity * HqlCppTranslator::doBuildActivityIndexAggregate(BuildCtx & ctx, IHqlExpression * expr)
  5898. {
  5899. OwnedHqlExpr transformed = buildIndexFromPhysical(expr);
  5900. OwnedHqlExpr optimized = optimizeHqlExpression(queryErrorProcessor(), transformed, getSourceAggregateOptimizeFlags());
  5901. IHqlExpression *tableExpr = queryPhysicalRootTable(optimized);
  5902. if (!tableExpr)
  5903. return buildNullIndexActivity(*this, ctx, optimized);
  5904. assertex(tableExpr->getOperator() == no_newkeyindex);
  5905. node_operator aggOp = querySimpleAggregate(expr, true, false);
  5906. if (aggOp == no_countgroup || aggOp == no_existsgroup)
  5907. {
  5908. IndexCountBuilder info(*this, tableExpr, tableExpr->queryChild(3), aggOp);
  5909. info.gatherVirtualFields(true, true);
  5910. return info.buildActivity(ctx, optimized, TAKindexcount, "IndexCount", NULL);
  5911. }
  5912. else
  5913. {
  5914. IndexAggregateBuilder info(*this, tableExpr, tableExpr->queryChild(3));
  5915. info.gatherVirtualFields(true, true);
  5916. return info.buildActivity(ctx, optimized, TAKindexaggregate, "IndexAggregate", NULL);
  5917. }
  5918. }
  5919. //---------------------------------------------------------------------------
  5920. class IndexGroupAggregateBuilder : public IndexReadBuilderBase
  5921. {
  5922. public:
  5923. IndexGroupAggregateBuilder(HqlCppTranslator & _translator, IHqlExpression *_tableExpr, IHqlExpression *_nameExpr)
  5924. : IndexReadBuilderBase(_translator, _tableExpr, _nameExpr)
  5925. {
  5926. failedFilterValue.clear();
  5927. transformAccessesCallback = false;
  5928. }
  5929. virtual void buildTransform(IHqlExpression * expr);
  5930. virtual void buildMembers(IHqlExpression * expr);
  5931. protected:
  5932. void doBuildProcessCountMembers(BuildCtx & ctx, IHqlExpression * aggregate);
  5933. virtual void processTransformSelect(BuildCtx & ctx, IHqlExpression * expr)
  5934. {
  5935. doBuildAggregateSelectIterator(ctx, expr);
  5936. }
  5937. virtual void analyseGraph(IHqlExpression * expr)
  5938. {
  5939. IndexReadBuilderBase::analyseGraph(expr);
  5940. returnIfFilterFails = !isNormalize;
  5941. }
  5942. protected:
  5943. bool transformAccessesCallback;
  5944. };
  5945. void IndexGroupAggregateBuilder::buildMembers(IHqlExpression * expr)
  5946. {
  5947. transformAccessesCallback = containsOperator(expr, no_filepos) || containsOperator(expr, no_id2blob);
  5948. buildFilenameMember();
  5949. buildGroupingMonitors(expr, monitors);
  5950. IndexReadBuilderBase::buildMembers(expr);
  5951. buildGlobalGroupAggregateHelpers(expr);
  5952. if (!isNormalize && !transformCanFilter && monitorsForGrouping && !transformAccessesCallback)
  5953. {
  5954. IHqlExpression * aggregate = expr->queryChild(0);
  5955. ThorActivityKind newKind = TAKnone;
  5956. switch (querySingleAggregate(aggregate, false, true, true))
  5957. {
  5958. case no_countgroup:
  5959. newKind = TAKindexgroupcount;
  5960. break;
  5961. case no_existsgroup:
  5962. newKind = TAKindexgroupexists;
  5963. break;
  5964. }
  5965. if (newKind)
  5966. {
  5967. instance->changeActivityKind(newKind);
  5968. doBuildProcessCountMembers(instance->startctx, aggregate);
  5969. }
  5970. }
  5971. }
  5972. void IndexGroupAggregateBuilder::doBuildProcessCountMembers(BuildCtx & ctx, IHqlExpression * aggregate)
  5973. {
  5974. IHqlExpression * dataset = aggregate->queryChild(0);
  5975. IHqlExpression * tgtRecord = aggregate->queryChild(1);
  5976. IHqlExpression * transform = aggregate->queryChild(2);
  5977. OwnedHqlExpr resultDataset = createDataset(no_anon, LINK(tgtRecord));
  5978. {
  5979. BuildCtx funcctx(ctx);
  5980. funcctx.addQuotedCompoundLiteral("virtual size32_t initialiseCountGrouping(ARowBuilder & crSelf, const void * _src)");
  5981. translator.ensureRowAllocated(funcctx, "crSelf");
  5982. funcctx.addQuotedLiteral("unsigned char * src = (unsigned char *) _src;");
  5983. translator.associateBlobHelper(funcctx, tableExpr, "fpp");
  5984. BoundRow * selfCursor = translator.bindSelf(funcctx, resultDataset, "crSelf");
  5985. translator.bindTableCursor(funcctx, dataset, "src");
  5986. //Replace count() with 0, exists() with true and call as a transform - which will error if the replacement fails.
  5987. OwnedHqlExpr count = createValue(no_countgroup, LINK(defaultIntegralType));
  5988. OwnedHqlExpr exists = createValue(no_existsgroup, makeBoolType());
  5989. OwnedHqlExpr newCount = createNullExpr(count);
  5990. OwnedHqlExpr newTransform = replaceExpression(transform, count, newCount);
  5991. newTransform.setown(replaceExpression(newTransform, exists, queryBoolExpr(true)));
  5992. translator.doTransform(funcctx, newTransform, selfCursor);
  5993. translator.buildReturnRecordSize(funcctx, selfCursor);
  5994. }
  5995. {
  5996. BuildCtx funcctx(ctx);
  5997. funcctx.addQuotedCompoundLiteral("virtual size32_t processCountGrouping(ARowBuilder & crSelf, unsigned __int64 count)");
  5998. translator.ensureRowAllocated(funcctx, "crSelf");
  5999. BoundRow * selfCursor = translator.bindSelf(funcctx, resultDataset, "crSelf");
  6000. OwnedHqlExpr newCount = createTranslatedOwned(createVariable("count", LINK(defaultIntegralType)));
  6001. OwnedHqlExpr self = getSelf(aggregate);
  6002. ForEachChild(idx, transform)
  6003. {
  6004. IHqlExpression * cur = transform->queryChild(idx);
  6005. if (cur->isAttribute())
  6006. continue;
  6007. OwnedHqlExpr target = selfCursor->bindToRow(cur->queryChild(0), self);
  6008. IHqlExpression * src = cur->queryChild(1);
  6009. IHqlExpression * arg = queryRealChild(src, 0);
  6010. BuildCtx condctx(funcctx);
  6011. node_operator srcOp = src->getOperator();
  6012. switch (srcOp)
  6013. {
  6014. case no_countgroup:
  6015. {
  6016. if (arg)
  6017. translator.buildFilter(condctx, arg);
  6018. OwnedHqlExpr newValue = createValue(no_add, target->getType(), LINK(target), ensureExprType(newCount, target->queryType()));
  6019. translator.buildAssign(condctx, target, newValue);
  6020. }
  6021. break;
  6022. }
  6023. }
  6024. translator.buildReturnRecordSize(funcctx, selfCursor);
  6025. }
  6026. }
  6027. void IndexGroupAggregateBuilder::buildTransform(IHqlExpression * expr)
  6028. {
  6029. BuildCtx transformCtx(instance->startctx);
  6030. transformCtx.addQuotedCompoundLiteral("void doProcessRow(byte * left, IHThorGroupAggregateCallback * callback)");
  6031. translator.associateBlobHelper(transformCtx, tableExpr, "fpp");
  6032. buildGroupAggregateTransformBody(transformCtx, expr, isNormalize || transformAccessesCallback, true);
  6033. }
  6034. //---------------------------------------------------------------------------
  6035. ABoundActivity * HqlCppTranslator::doBuildActivityIndexGroupAggregate(BuildCtx & ctx, IHqlExpression * expr)
  6036. {
  6037. OwnedHqlExpr transformed = buildIndexFromPhysical(expr);
  6038. OwnedHqlExpr optimized = optimizeHqlExpression(queryErrorProcessor(), transformed, getSourceAggregateOptimizeFlags());
  6039. IHqlExpression *tableExpr = queryPhysicalRootTable(optimized);
  6040. if (!tableExpr)
  6041. return buildNullIndexActivity(*this, ctx, optimized);
  6042. IHqlExpression * aggregate = expr->queryChild(0);
  6043. assertex(aggregate->getOperator() == no_newaggregate || aggregate->getOperator() == no_aggregate);
  6044. ThorActivityKind tak = TAKindexgroupaggregate;
  6045. assertex(tableExpr->getOperator() == no_newkeyindex);
  6046. IndexGroupAggregateBuilder info(*this, tableExpr, tableExpr->queryChild(3));
  6047. info.gatherVirtualFields(true, true);
  6048. return info.buildActivity(ctx, optimized, tak, "IndexGroupAggregate", NULL);
  6049. }
  6050. //---------------------------------------------------------------------------
  6051. void associateVirtualCallbacks(HqlCppTranslator & translator, BuildCtx & ctx, IHqlExpression * dataset)
  6052. {
  6053. OwnedHqlExpr fpos = getFilepos(dataset, false);
  6054. OwnedHqlExpr lfpos = getFilepos(dataset, true);
  6055. Owned<IHqlExpression> fposExpr = createFileposCall(translator, getFilePositionId, "fpp", "crSelf.row()");
  6056. Owned<IHqlExpression> lfposExpr = createFileposCall(translator, getLocalFilePositionId, "fpp", "crSelf.row()");
  6057. ctx.associateExpr(fpos, fposExpr);
  6058. ctx.associateExpr(lfpos, lfposExpr);
  6059. }
  6060. void HqlCppTranslator::buildXmlReadTransform(IHqlExpression * dataset, StringBuffer & factoryName, bool & usesContents)
  6061. {
  6062. OwnedHqlExpr xmlMarker = createAttribute(xmlReadMarkerAtom, LINK(dataset->queryRecord()));
  6063. BuildCtx declarectx(*code, declareAtom);
  6064. HqlExprAssociation * match = declarectx.queryMatchExpr(xmlMarker);
  6065. if (match)
  6066. {
  6067. IHqlExpression * matchExpr = match->queryExpr();
  6068. matchExpr->queryChild(0)->queryValue()->getStringValue(factoryName);
  6069. usesContents = matchExpr->queryChild(1)->queryValue()->getBoolValue();
  6070. return;
  6071. }
  6072. StringBuffer s, id, className;
  6073. getUniqueId(id);
  6074. className.append("cx2r").append(id);
  6075. const char * interfaceName = "IXmlToRowTransformer";
  6076. StringBuffer prolog, epilog;
  6077. prolog.append("struct ").append(className).append(" : public RtlCInterface, implements ").append(interfaceName);
  6078. epilog.append(";");
  6079. GlobalClassBuilder builder(*this, declarectx, className, "CXmlToRowTransformer", interfaceName);
  6080. builder.buildClass(XmlTransformerPrio);
  6081. builder.setIncomplete(true);
  6082. BuildCtx & classctx = builder.classctx;
  6083. s.clear().append("inline ").append(className).append("(unsigned _activityId) : CXmlToRowTransformer(_activityId) {}");
  6084. classctx.addQuoted(s);
  6085. BuildCtx funcctx(classctx);
  6086. funcctx.addQuotedCompoundLiteral("virtual size32_t transform(ARowBuilder & crSelf, IColumnProvider * row, IThorDiskCallback * fpp)");
  6087. ensureRowAllocated(funcctx, "crSelf");
  6088. xmlUsesContents = false;
  6089. //MORE: If this becomes a compound activity
  6090. BoundRow * rootSelfRow = bindSelf(funcctx, dataset, "crSelf");
  6091. bindXmlTableCursor(funcctx, dataset, "row", no_none, NULL, true);
  6092. OwnedHqlExpr activityId = createVariable("activityId", LINK(sizetType));
  6093. funcctx.associateExpr(queryActivityIdMarker(), activityId);
  6094. associateVirtualCallbacks(*this, funcctx, dataset);
  6095. OwnedHqlExpr active = ensureActiveRow(dataset);
  6096. buildAssign(funcctx, rootSelfRow->querySelector(), active);
  6097. buildReturnRecordSize(funcctx, rootSelfRow);
  6098. usesContents = xmlUsesContents;
  6099. rootSelfRow = NULL;
  6100. buildMetaMember(classctx, dataset, false, "queryRecordSize");
  6101. builder.setIncomplete(false);
  6102. builder.completeClass(XmlTransformerPrio);
  6103. factoryName.append(builder.accessorName);
  6104. OwnedHqlExpr matchedValue = createAttribute(internalAtom, createConstant(factoryName.str()), createConstant(usesContents));
  6105. declarectx.associateExpr(xmlMarker, matchedValue);
  6106. }
  6107. //---------------------------------------------------------------------------
  6108. unsigned HqlCppTranslator::buildCsvReadTransform(BuildCtx & subctx, IHqlExpression * dataset, bool newInterface, IHqlExpression * csvAttr)
  6109. {
  6110. BuildCtx funcctx(subctx);
  6111. if (newInterface)
  6112. funcctx.addQuotedCompoundLiteral("virtual size32_t transform(ARowBuilder & crSelf, unsigned * lenSrc, const char * * dataSrc)");
  6113. else
  6114. funcctx.addQuotedCompoundLiteral("virtual size32_t transform(ARowBuilder & crSelf, unsigned * lenSrc, const char * * dataSrc, unsigned __int64 _fpos)");
  6115. //MORE: If this becomes a compound activity
  6116. BoundRow * rootSelfRow = bindSelf(funcctx, dataset, "crSelf");
  6117. bindCsvTableCursor(funcctx, dataset, "Src", no_none, NULL, true, queryCsvEncoding(csvAttr));
  6118. ensureRowAllocated(funcctx, rootSelfRow);
  6119. if (newInterface)
  6120. {
  6121. associateVirtualCallbacks(*this, funcctx, dataset);
  6122. }
  6123. else
  6124. {
  6125. OwnedHqlExpr fpos = getFilepos(dataset, false);
  6126. OwnedHqlExpr fposVar = createVariable("_fpos", fpos->getType());
  6127. funcctx.associateExpr(fpos, fposVar);
  6128. }
  6129. OwnedHqlExpr active = ensureActiveRow(dataset);
  6130. buildAssign(funcctx, rootSelfRow->querySelector(), active);
  6131. buildReturnRecordSize(funcctx, rootSelfRow);
  6132. rootSelfRow = NULL;
  6133. return countTotalFields(dataset->queryRecord(), false);
  6134. }
  6135. void HqlCppTranslator::buildCsvReadTransformer(IHqlExpression * dataset, StringBuffer & instanceName, IHqlExpression * optCsvAttr)
  6136. {
  6137. OwnedHqlExpr csvMarker = createAttribute(csvReadMarkerAtom, LINK(dataset->queryRecord()), LINK(optCsvAttr));
  6138. BuildCtx declarectx(*code, declareAtom);
  6139. HqlExprAssociation * match = declarectx.queryMatchExpr(csvMarker);
  6140. if (match)
  6141. {
  6142. IHqlExpression * matchExpr = match->queryExpr();
  6143. matchExpr->queryChild(0)->queryValue()->getStringValue(instanceName);
  6144. return;
  6145. }
  6146. StringBuffer id, className;
  6147. getUniqueId(id);
  6148. instanceName.append("c2r").append(id);
  6149. className.append("cc2r").append(id);
  6150. StringBuffer prolog, epilog;
  6151. prolog.append("struct ").append(className).append(" : public RtlCInterface, implements ICsvToRowTransformer");
  6152. epilog.append(" ").append(instanceName).append(";");
  6153. BuildCtx classctx(declarectx);
  6154. classctx.setNextPriority(XmlTransformerPrio);
  6155. IHqlStmt * transformClass = classctx.addQuotedCompound(prolog, epilog);
  6156. transformClass->setIncomplete(true);
  6157. transformClass->setIncluded(false); // if can't generate csv for this record, then don't generate an invalid class.
  6158. classctx.addQuotedLiteral("virtual void Link() const { RtlCInterface::Link(); }");
  6159. classctx.addQuotedLiteral("virtual bool Release() const { return RtlCInterface::Release(); }");
  6160. unsigned maxColumns = buildCsvReadTransform(classctx, dataset, false, optCsvAttr);
  6161. doBuildUnsignedFunction(classctx, "getMaxColumns", maxColumns);
  6162. buildMetaMember(classctx, dataset, false, "queryRecordSize");
  6163. buildCsvParameters(classctx, optCsvAttr, NULL, true);
  6164. transformClass->setIncomplete(false);
  6165. transformClass->setIncluded(true);
  6166. if (options.spanMultipleCpp)
  6167. {
  6168. StringBuffer helperFunc;
  6169. createAccessFunctions(helperFunc, declarectx, XmlTransformerPrio, "ICsvToRowTransformer", instanceName.str());
  6170. instanceName.clear().append(helperFunc).append("()");
  6171. }
  6172. OwnedHqlExpr matchedValue = createAttribute(internalAtom, createConstant(instanceName.str()));
  6173. declarectx.associateExpr(csvMarker, matchedValue);
  6174. }
  6175. ABoundActivity * HqlCppTranslator::doBuildActivityXmlRead(BuildCtx & ctx, IHqlExpression * expr)
  6176. {
  6177. IHqlExpression * tableExpr = expr;
  6178. IHqlExpression * filename = tableExpr->queryChild(0);
  6179. IHqlExpression * mode = tableExpr->queryChild(2);
  6180. node_operator modeType = mode->getOperator();
  6181. StringBuffer s;
  6182. ThorActivityKind kind = (modeType == no_json) ? TAKjsonread : TAKxmlread;
  6183. Owned<ActivityInstance> instance = new ActivityInstance(*this, ctx, kind, expr, "XmlRead");
  6184. buildActivityFramework(instance);
  6185. buildInstancePrefix(instance);
  6186. //MORE: Improve when we support projecting xml instead of reading all
  6187. SourceFieldUsage * fieldUsage = querySourceFieldUsage(tableExpr);
  6188. if (fieldUsage && !fieldUsage->seenAll())
  6189. fieldUsage->noteAll();
  6190. //---- virtual const char * getFileName() { return "x.d00"; } ----
  6191. buildFilenameFunction(*instance, instance->startctx, "getFileName", filename, hasDynamicFilename(tableExpr));
  6192. buildEncryptHelper(instance->startctx, tableExpr->queryAttribute(encryptAtom));
  6193. bool usesContents = false;
  6194. doBuildXmlReadMember(*instance, tableExpr, "queryTransformer", usesContents);
  6195. doBuildVarStringFunction(instance->classctx, "getXmlIteratorPath", queryRealChild(mode, 0));
  6196. buildMetaMember(instance->classctx, tableExpr, false, "queryDiskRecordSize"); // A lie, but I don't care....
  6197. //virtual unsigned getFlags() = 0;
  6198. StringBuffer flags;
  6199. if (expr->hasAttribute(_spill_Atom)) flags.append("|TDXtemporary");
  6200. if (expr->hasAttribute(unsortedAtom)) flags.append("|TDRunsorted");
  6201. if (expr->hasAttribute(optAtom)) flags.append("|TDRoptional");
  6202. if (usesContents) flags.append("|TDRusexmlcontents");
  6203. if (mode->hasAttribute(noRootAtom)) flags.append("|TDRxmlnoroot");
  6204. if (!filename->isConstant()) flags.append("|TDXvarfilename");
  6205. if (hasDynamicFilename(expr)) flags.append("|TDXdynamicfilename");
  6206. if (flags.length())
  6207. doBuildUnsignedFunction(instance->classctx, "getFlags", flags.str()+1);
  6208. //Note the helper base class contains code like the following
  6209. //IThorDiskCallback * fpp;");
  6210. //virtual void setCallback(IThorDiskCallback * _tc) { fpp = _tc; }");
  6211. buildInstanceSuffix(instance);
  6212. addFileDependency(filename, instance->queryBoundActivity());
  6213. return instance->getBoundActivity();
  6214. }
  6215. //---------------------------------------------------------------------------
  6216. ABoundActivity * HqlCppTranslator::doBuildActivityTable(BuildCtx & ctx, IHqlExpression * expr)
  6217. {
  6218. node_operator mode = expr->queryChild(2)->getOperator();
  6219. switch (mode)
  6220. {
  6221. case no_thor:
  6222. case no_flat:
  6223. case no_pipe:
  6224. case no_csv:
  6225. return doBuildActivityDiskRead(ctx, expr);
  6226. case no_xml:
  6227. case no_json:
  6228. return doBuildActivityXmlRead(ctx, expr);
  6229. default:
  6230. UNIMPLEMENTED;
  6231. }
  6232. }
  6233. //---------------------------------------------------------------------------
  6234. class FetchBuilder : public SourceBuilder
  6235. {
  6236. public:
  6237. FetchBuilder(HqlCppTranslator & _translator, IHqlExpression *_tableExpr, IHqlExpression *_nameExpr, IHqlExpression * _fetchExpr)
  6238. : SourceBuilder(_translator, _tableExpr, _nameExpr)
  6239. {
  6240. compoundExpr.set(_fetchExpr);
  6241. fetchExpr.set(queryFetch(_fetchExpr));
  6242. selSeq.set(querySelSeq(fetchExpr));
  6243. fetchRhs = fetchExpr->queryChild(1);
  6244. memoryRhsRecord = fetchRhs->queryRecord();
  6245. serializedRhsRecord.setown(getSerializedForm(memoryRhsRecord, diskAtom));
  6246. }
  6247. virtual void buildMembers(IHqlExpression * expr);
  6248. virtual void buildTransform(IHqlExpression * expr);
  6249. virtual void buildTransformFpos(BuildCtx & transformCtx);
  6250. protected:
  6251. HqlExprAttr compoundExpr;
  6252. HqlExprAttr fetchExpr;
  6253. HqlExprAttr selSeq;
  6254. HqlExprAttr serializedRhsRecord;
  6255. IHqlExpression * fetchRhs;
  6256. IHqlExpression * memoryRhsRecord;
  6257. };
  6258. void FetchBuilder::buildMembers(IHqlExpression * expr)
  6259. {
  6260. buildReadMembers(expr);
  6261. IHqlExpression * fetch = queryFetch(expr);
  6262. BuildCtx getposctx(instance->startctx);
  6263. getposctx.addQuotedCompoundLiteral("virtual unsigned __int64 extractPosition(const void * _right)");
  6264. getposctx.addQuotedLiteral("const unsigned char * right = (const unsigned char *) _right;");
  6265. translator.bindTableCursor(getposctx, fetch->queryChild(1), "right", no_right, selSeq);
  6266. translator.buildReturn(getposctx, fetch->queryChild(2));
  6267. translator.buildEncryptHelper(instance->startctx, tableExpr->queryAttribute(encryptAtom), "getFileEncryptKey");
  6268. //Fetch flags
  6269. StringBuffer flags;
  6270. if (tableExpr->hasAttribute(optAtom))
  6271. flags.append("|FFdatafileoptional");
  6272. if (!nameExpr->isConstant())
  6273. flags.append("|FFvarfilename");
  6274. if (translator.hasDynamicFilename(tableExpr))
  6275. flags.append("|FFdynamicfilename");
  6276. if (flags.length())
  6277. translator.doBuildUnsignedFunction(instance->classctx, "getFetchFlags", flags.str()+1);
  6278. if (tableExpr->hasAttribute(optAtom) && translator.targetRoxie())
  6279. instance->addAttributeBool("_isOpt", true);
  6280. buildLimits(instance->startctx, expr, instance->activityId);
  6281. switch (getDatasetKind(tableExpr))
  6282. {
  6283. case no_csv:
  6284. {
  6285. translator.buildCsvParameters(instance->nestedctx, tableExpr->queryChild(2), NULL, true);
  6286. unsigned maxColumns = getFieldCount(tableExpr->queryRecord());
  6287. StringBuffer s;
  6288. s.clear().append("virtual unsigned getMaxColumns() { return ").append(maxColumns).append("; }");
  6289. instance->classctx.addQuoted(s);
  6290. break;
  6291. }
  6292. case no_xml:
  6293. case no_json:
  6294. {
  6295. // virtual const char * getXmlIteratorPath()
  6296. translator.doBuildVarStringFunction(instance->classctx, "getXmlIteratorPath", queryRealChild(tableExpr->queryChild(2), 0));
  6297. break;
  6298. }
  6299. default:
  6300. translator.buildFormatCrcFunction(instance->classctx, "getDiskFormatCrc", physicalRecord, NULL, 0);
  6301. break;
  6302. }
  6303. if (!containsOnlyLeft(fetch->queryChild(3), true))
  6304. {
  6305. //MORE: Need to change following if we optimize it to only extract the relevant fields.
  6306. instance->classctx.addQuotedLiteral("virtual bool extractAllJoinFields() { return true; }");
  6307. BuildCtx funcctx(instance->startctx);
  6308. funcctx.addQuotedCompoundLiteral("virtual size32_t extractJoinFields(ARowBuilder & crSelf, const void * _left)");
  6309. translator.ensureRowAllocated(funcctx, "crSelf");
  6310. translator.buildRecordSerializeExtract(funcctx, memoryRhsRecord);
  6311. StringBuffer s;
  6312. MetaInstance meta(translator, serializedRhsRecord, false);
  6313. translator.buildMetaInfo(meta);
  6314. instance->classctx.addQuoted(s.clear().append("virtual IOutputMetaData * queryExtractedSize() { return &").append(meta.queryInstanceObject()).append("; }"));
  6315. }
  6316. }
  6317. void FetchBuilder::buildTransform(IHqlExpression * expr)
  6318. {
  6319. translator.xmlUsesContents = false;
  6320. BuildCtx transformCtx(instance->startctx);
  6321. switch (getDatasetKind(tableExpr))
  6322. {
  6323. case no_csv:
  6324. transformCtx.addQuotedCompoundLiteral("virtual size32_t transform(ARowBuilder & crSelf, unsigned * lenLeft, const char * * dataLeft, const void * _right, unsigned __int64 _fpos)");
  6325. transformCtx.addQuotedLiteral("unsigned char * right = (unsigned char *)_right;");
  6326. break;
  6327. case no_xml:
  6328. case no_json:
  6329. transformCtx.addQuotedCompoundLiteral("virtual size32_t transform(ARowBuilder & crSelf, IColumnProvider * xmlLeft, const void * _right, unsigned __int64 _fpos)");
  6330. transformCtx.addQuotedLiteral("unsigned char * right = (unsigned char *)_right;");
  6331. break;
  6332. default:
  6333. transformCtx.addQuotedCompoundLiteral("virtual size32_t transform(ARowBuilder & crSelf, const void * _left, const void * _right, unsigned __int64 _fpos)");
  6334. transformCtx.addQuotedLiteral("unsigned char * left = (unsigned char *)_left;");
  6335. transformCtx.addQuotedLiteral("unsigned char * right = (unsigned char *)_right;");
  6336. break;
  6337. }
  6338. translator.ensureRowAllocated(transformCtx, "crSelf");
  6339. buildTransformBody(transformCtx, expr, true, false, true);
  6340. if (translator.xmlUsesContents)
  6341. instance->classctx.addQuotedLiteral("virtual bool requiresContents() { return true; }");
  6342. }
  6343. void FetchBuilder::buildTransformFpos(BuildCtx & transformCtx)
  6344. {
  6345. fpos.setown(createVariable("_fpos", LINK(fposType)));
  6346. //NB: Because the fetch gets merged with the usertable used to project the dataset, the
  6347. //transform contains filepos(LEFT) not filepos(tableExpr)
  6348. OwnedHqlExpr leftSelect = createSelector(no_left, fetchExpr->queryChild(0), selSeq);
  6349. OwnedHqlExpr fposField = getFilepos(leftSelect, false);
  6350. transformCtx.associateExpr(fposField, fpos);
  6351. //MORE: Could possibly support virtual(filename) here
  6352. }
  6353. static HqlTransformerInfo fetchInputReplacerInfo("FetchInputReplacer");
  6354. class FetchInputReplacer : public NewHqlTransformer
  6355. {
  6356. public:
  6357. FetchInputReplacer(IHqlExpression * _newDataset, node_operator side)
  6358. : NewHqlTransformer(fetchInputReplacerInfo)
  6359. {
  6360. newDataset = _newDataset;
  6361. child = (side == no_left) ? 0 : 1;
  6362. }
  6363. virtual IHqlExpression * createTransformed(IHqlExpression * expr)
  6364. {
  6365. if (expr->getOperator() == no_fetch)
  6366. return replaceChild(expr, child, newDataset);
  6367. return NewHqlTransformer::createTransformed(expr);
  6368. }
  6369. protected:
  6370. IHqlExpression * newDataset;
  6371. unsigned child;
  6372. };
  6373. IHqlExpression * replaceFetchInput(IHqlExpression * expr, IHqlExpression * newDataset, node_operator side)
  6374. {
  6375. FetchInputReplacer replacer(newDataset, side);
  6376. return replacer.transformRoot(expr);
  6377. }
  6378. ABoundActivity * HqlCppTranslator::doBuildActivityFetch(BuildCtx & ctx, IHqlExpression * expr)
  6379. {
  6380. IHqlExpression *fetch = queryFetch(expr);
  6381. IHqlExpression *tableExpr = queryPhysicalRootTable(fetch->queryChild(0));
  6382. if (!tableExpr)
  6383. throwError(HQLERR_FetchNonDiskfile);
  6384. FetchBuilder info(*this, tableExpr, tableExpr->queryChild(0), expr);
  6385. info.gatherVirtualFields(false, true);//?needToSerializeRecord(mode)
  6386. unsigned optFlags = (options.foldOptimized ? HOOfold : 0);
  6387. if (info.recordHasVirtualsOrDeserialize())
  6388. {
  6389. OwnedHqlExpr projected = createTableWithoutVirtuals(info.fieldInfo, tableExpr);
  6390. //Nasty: We don't want to optimize the rhs, otherwise references get changed!
  6391. //so optimize everything except the rhs, and then add the rhs back in again.
  6392. IHqlExpression * fetchRhs = fetch->queryChild(1);
  6393. OwnedHqlExpr null = createDataset(no_anon, LINK(fetchRhs->queryRecord()));
  6394. OwnedHqlExpr simple = replaceFetchInput(expr, null, no_right);
  6395. OwnedHqlExpr transformed = replaceExpression(simple, tableExpr, projected);
  6396. OwnedHqlExpr optSimple = optimizeHqlExpression(queryErrorProcessor(), transformed, optFlags);
  6397. OwnedHqlExpr optimized = replaceFetchInput(optSimple, fetchRhs, no_right);
  6398. return doBuildActivityFetch(ctx, optimized);
  6399. }
  6400. if (getProjectCount(expr) > 1)
  6401. {
  6402. OwnedHqlExpr optimized = optimizeHqlExpression(queryErrorProcessor(), expr, optFlags);
  6403. return doBuildActivityFetch(ctx, optimized);
  6404. }
  6405. Owned<ABoundActivity> childActivity = buildCachedActivity(ctx, fetch->queryChild(1));
  6406. node_operator kind = getDatasetKind(tableExpr);
  6407. switch (kind)
  6408. {
  6409. case no_csv:
  6410. return info.buildActivity(ctx, expr, TAKcsvfetch, "CsvFetch", childActivity);
  6411. case no_xml:
  6412. return info.buildActivity(ctx, expr, TAKxmlfetch, "XmlFetch", childActivity);
  6413. case no_json:
  6414. //Note use of "XmlFetch" because we want the code generator to leverage existing xml classes
  6415. return info.buildActivity(ctx, expr, TAKjsonfetch, "XmlFetch", childActivity);
  6416. case no_flat:
  6417. case no_thor:
  6418. return info.buildActivity(ctx, expr, TAKfetch, "Fetch", childActivity);
  6419. }
  6420. throwError1(HQLERR_FetchNotSupportMode, getOpString(kind));
  6421. return NULL;
  6422. }