jfile.cpp 182 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712171317141715171617171718171917201721172217231724172517261727172817291730173117321733173417351736173717381739174017411742174317441745174617471748174917501751175217531754175517561757175817591760176117621763176417651766176717681769177017711772177317741775177617771778177917801781178217831784178517861787178817891790179117921793179417951796179717981799180018011802180318041805180618071808180918101811181218131814181518161817181818191820182118221823182418251826182718281829183018311832183318341835183618371838183918401841184218431844184518461847184818491850185118521853185418551856185718581859186018611862186318641865186618671868186918701871187218731874187518761877187818791880188118821883188418851886188718881889189018911892189318941895189618971898189919001901190219031904190519061907190819091910191119121913191419151916191719181919192019211922192319241925192619271928192919301931193219331934193519361937193819391940194119421943194419451946194719481949195019511952195319541955195619571958195919601961196219631964196519661967196819691970197119721973197419751976197719781979198019811982198319841985198619871988198919901991199219931994199519961997199819992000200120022003200420052006200720082009201020112012201320142015201620172018201920202021202220232024202520262027202820292030203120322033203420352036203720382039204020412042204320442045204620472048204920502051205220532054205520562057205820592060206120622063206420652066206720682069207020712072207320742075207620772078207920802081208220832084208520862087208820892090209120922093209420952096209720982099210021012102210321042105210621072108210921102111211221132114211521162117211821192120212121222123212421252126212721282129213021312132213321342135213621372138213921402141214221432144214521462147214821492150215121522153215421552156215721582159216021612162216321642165216621672168216921702171217221732174217521762177217821792180218121822183218421852186218721882189219021912192219321942195219621972198219922002201220222032204220522062207220822092210221122122213221422152216221722182219222022212222222322242225222622272228222922302231223222332234223522362237223822392240224122422243224422452246224722482249225022512252225322542255225622572258225922602261226222632264226522662267226822692270227122722273227422752276227722782279228022812282228322842285228622872288228922902291229222932294229522962297229822992300230123022303230423052306230723082309231023112312231323142315231623172318231923202321232223232324232523262327232823292330233123322333233423352336233723382339234023412342234323442345234623472348234923502351235223532354235523562357235823592360236123622363236423652366236723682369237023712372237323742375237623772378237923802381238223832384238523862387238823892390239123922393239423952396239723982399240024012402240324042405240624072408240924102411241224132414241524162417241824192420242124222423242424252426242724282429243024312432243324342435243624372438243924402441244224432444244524462447244824492450245124522453245424552456245724582459246024612462246324642465246624672468246924702471247224732474247524762477247824792480248124822483248424852486248724882489249024912492249324942495249624972498249925002501250225032504250525062507250825092510251125122513251425152516251725182519252025212522252325242525252625272528252925302531253225332534253525362537253825392540254125422543254425452546254725482549255025512552255325542555255625572558255925602561256225632564256525662567256825692570257125722573257425752576257725782579258025812582258325842585258625872588258925902591259225932594259525962597259825992600260126022603260426052606260726082609261026112612261326142615261626172618261926202621262226232624262526262627262826292630263126322633263426352636263726382639264026412642264326442645264626472648264926502651265226532654265526562657265826592660266126622663266426652666266726682669267026712672267326742675267626772678267926802681268226832684268526862687268826892690269126922693269426952696269726982699270027012702270327042705270627072708270927102711271227132714271527162717271827192720272127222723272427252726272727282729273027312732273327342735273627372738273927402741274227432744274527462747274827492750275127522753275427552756275727582759276027612762276327642765276627672768276927702771277227732774277527762777277827792780278127822783278427852786278727882789279027912792279327942795279627972798279928002801280228032804280528062807280828092810281128122813281428152816281728182819282028212822282328242825282628272828282928302831283228332834283528362837283828392840284128422843284428452846284728482849285028512852285328542855285628572858285928602861286228632864286528662867286828692870287128722873287428752876287728782879288028812882288328842885288628872888288928902891289228932894289528962897289828992900290129022903290429052906290729082909291029112912291329142915291629172918291929202921292229232924292529262927292829292930293129322933293429352936293729382939294029412942294329442945294629472948294929502951295229532954295529562957295829592960296129622963296429652966296729682969297029712972297329742975297629772978297929802981298229832984298529862987298829892990299129922993299429952996299729982999300030013002300330043005300630073008300930103011301230133014301530163017301830193020302130223023302430253026302730283029303030313032303330343035303630373038303930403041304230433044304530463047304830493050305130523053305430553056305730583059306030613062306330643065306630673068306930703071307230733074307530763077307830793080308130823083308430853086308730883089309030913092309330943095309630973098309931003101310231033104310531063107310831093110311131123113311431153116311731183119312031213122312331243125312631273128312931303131313231333134313531363137313831393140314131423143314431453146314731483149315031513152315331543155315631573158315931603161316231633164316531663167316831693170317131723173317431753176317731783179318031813182318331843185318631873188318931903191319231933194319531963197319831993200320132023203320432053206320732083209321032113212321332143215321632173218321932203221322232233224322532263227322832293230323132323233323432353236323732383239324032413242324332443245324632473248324932503251325232533254325532563257325832593260326132623263326432653266326732683269327032713272327332743275327632773278327932803281328232833284328532863287328832893290329132923293329432953296329732983299330033013302330333043305330633073308330933103311331233133314331533163317331833193320332133223323332433253326332733283329333033313332333333343335333633373338333933403341334233433344334533463347334833493350335133523353335433553356335733583359336033613362336333643365336633673368336933703371337233733374337533763377337833793380338133823383338433853386338733883389339033913392339333943395339633973398339934003401340234033404340534063407340834093410341134123413341434153416341734183419342034213422342334243425342634273428342934303431343234333434343534363437343834393440344134423443344434453446344734483449345034513452345334543455345634573458345934603461346234633464346534663467346834693470347134723473347434753476347734783479348034813482348334843485348634873488348934903491349234933494349534963497349834993500350135023503350435053506350735083509351035113512351335143515351635173518351935203521352235233524352535263527352835293530353135323533353435353536353735383539354035413542354335443545354635473548354935503551355235533554355535563557355835593560356135623563356435653566356735683569357035713572357335743575357635773578357935803581358235833584358535863587358835893590359135923593359435953596359735983599360036013602360336043605360636073608360936103611361236133614361536163617361836193620362136223623362436253626362736283629363036313632363336343635363636373638363936403641364236433644364536463647364836493650365136523653365436553656365736583659366036613662366336643665366636673668366936703671367236733674367536763677367836793680368136823683368436853686368736883689369036913692369336943695369636973698369937003701370237033704370537063707370837093710371137123713371437153716371737183719372037213722372337243725372637273728372937303731373237333734373537363737373837393740374137423743374437453746374737483749375037513752375337543755375637573758375937603761376237633764376537663767376837693770377137723773377437753776377737783779378037813782378337843785378637873788378937903791379237933794379537963797379837993800380138023803380438053806380738083809381038113812381338143815381638173818381938203821382238233824382538263827382838293830383138323833383438353836383738383839384038413842384338443845384638473848384938503851385238533854385538563857385838593860386138623863386438653866386738683869387038713872387338743875387638773878387938803881388238833884388538863887388838893890389138923893389438953896389738983899390039013902390339043905390639073908390939103911391239133914391539163917391839193920392139223923392439253926392739283929393039313932393339343935393639373938393939403941394239433944394539463947394839493950395139523953395439553956395739583959396039613962396339643965396639673968396939703971397239733974397539763977397839793980398139823983398439853986398739883989399039913992399339943995399639973998399940004001400240034004400540064007400840094010401140124013401440154016401740184019402040214022402340244025402640274028402940304031403240334034403540364037403840394040404140424043404440454046404740484049405040514052405340544055405640574058405940604061406240634064406540664067406840694070407140724073407440754076407740784079408040814082408340844085408640874088408940904091409240934094409540964097409840994100410141024103410441054106410741084109411041114112411341144115411641174118411941204121412241234124412541264127412841294130413141324133413441354136413741384139414041414142414341444145414641474148414941504151415241534154415541564157415841594160416141624163416441654166416741684169417041714172417341744175417641774178417941804181418241834184418541864187418841894190419141924193419441954196419741984199420042014202420342044205420642074208420942104211421242134214421542164217421842194220422142224223422442254226422742284229423042314232423342344235423642374238423942404241424242434244424542464247424842494250425142524253425442554256425742584259426042614262426342644265426642674268426942704271427242734274427542764277427842794280428142824283428442854286428742884289429042914292429342944295429642974298429943004301430243034304430543064307430843094310431143124313431443154316431743184319432043214322432343244325432643274328432943304331433243334334433543364337433843394340434143424343434443454346434743484349435043514352435343544355435643574358435943604361436243634364436543664367436843694370437143724373437443754376437743784379438043814382438343844385438643874388438943904391439243934394439543964397439843994400440144024403440444054406440744084409441044114412441344144415441644174418441944204421442244234424442544264427442844294430443144324433443444354436443744384439444044414442444344444445444644474448444944504451445244534454445544564457445844594460446144624463446444654466446744684469447044714472447344744475447644774478447944804481448244834484448544864487448844894490449144924493449444954496449744984499450045014502450345044505450645074508450945104511451245134514451545164517451845194520452145224523452445254526452745284529453045314532453345344535453645374538453945404541454245434544454545464547454845494550455145524553455445554556455745584559456045614562456345644565456645674568456945704571457245734574457545764577457845794580458145824583458445854586458745884589459045914592459345944595459645974598459946004601460246034604460546064607460846094610461146124613461446154616461746184619462046214622462346244625462646274628462946304631463246334634463546364637463846394640464146424643464446454646464746484649465046514652465346544655465646574658465946604661466246634664466546664667466846694670467146724673467446754676467746784679468046814682468346844685468646874688468946904691469246934694469546964697469846994700470147024703470447054706470747084709471047114712471347144715471647174718471947204721472247234724472547264727472847294730473147324733473447354736473747384739474047414742474347444745474647474748474947504751475247534754475547564757475847594760476147624763476447654766476747684769477047714772477347744775477647774778477947804781478247834784478547864787478847894790479147924793479447954796479747984799480048014802480348044805480648074808480948104811481248134814481548164817481848194820482148224823482448254826482748284829483048314832483348344835483648374838483948404841484248434844484548464847484848494850485148524853485448554856485748584859486048614862486348644865486648674868486948704871487248734874487548764877487848794880488148824883488448854886488748884889489048914892489348944895489648974898489949004901490249034904490549064907490849094910491149124913491449154916491749184919492049214922492349244925492649274928492949304931493249334934493549364937493849394940494149424943494449454946494749484949495049514952495349544955495649574958495949604961496249634964496549664967496849694970497149724973497449754976497749784979498049814982498349844985498649874988498949904991499249934994499549964997499849995000500150025003500450055006500750085009501050115012501350145015501650175018501950205021502250235024502550265027502850295030503150325033503450355036503750385039504050415042504350445045504650475048504950505051505250535054505550565057505850595060506150625063506450655066506750685069507050715072507350745075507650775078507950805081508250835084508550865087508850895090509150925093509450955096509750985099510051015102510351045105510651075108510951105111511251135114511551165117511851195120512151225123512451255126512751285129513051315132513351345135513651375138513951405141514251435144514551465147514851495150515151525153515451555156515751585159516051615162516351645165516651675168516951705171517251735174517551765177517851795180518151825183518451855186518751885189519051915192519351945195519651975198519952005201520252035204520552065207520852095210521152125213521452155216521752185219522052215222522352245225522652275228522952305231523252335234523552365237523852395240524152425243524452455246524752485249525052515252525352545255525652575258525952605261526252635264526552665267526852695270527152725273527452755276527752785279528052815282528352845285528652875288528952905291529252935294529552965297529852995300530153025303530453055306530753085309531053115312531353145315531653175318531953205321532253235324532553265327532853295330533153325333533453355336533753385339534053415342534353445345534653475348534953505351535253535354535553565357535853595360536153625363536453655366536753685369537053715372537353745375537653775378537953805381538253835384538553865387538853895390539153925393539453955396539753985399540054015402540354045405540654075408540954105411541254135414541554165417541854195420542154225423542454255426542754285429543054315432543354345435543654375438543954405441544254435444544554465447544854495450545154525453545454555456545754585459546054615462546354645465546654675468546954705471547254735474547554765477547854795480548154825483548454855486548754885489549054915492549354945495549654975498549955005501550255035504550555065507550855095510551155125513551455155516551755185519552055215522552355245525552655275528552955305531553255335534553555365537553855395540554155425543554455455546554755485549555055515552555355545555555655575558555955605561556255635564556555665567556855695570557155725573557455755576557755785579558055815582558355845585558655875588558955905591559255935594559555965597559855995600560156025603560456055606560756085609561056115612561356145615561656175618561956205621562256235624562556265627562856295630563156325633563456355636563756385639564056415642564356445645564656475648564956505651565256535654565556565657565856595660566156625663566456655666566756685669567056715672567356745675567656775678567956805681568256835684568556865687568856895690569156925693569456955696569756985699570057015702570357045705570657075708570957105711571257135714571557165717571857195720572157225723572457255726572757285729573057315732573357345735573657375738573957405741574257435744574557465747574857495750575157525753575457555756575757585759576057615762576357645765576657675768576957705771577257735774577557765777577857795780578157825783578457855786578757885789579057915792579357945795579657975798579958005801580258035804580558065807580858095810581158125813581458155816581758185819582058215822582358245825582658275828582958305831583258335834583558365837583858395840584158425843584458455846584758485849585058515852585358545855585658575858585958605861586258635864586558665867586858695870587158725873587458755876587758785879588058815882588358845885588658875888588958905891589258935894589558965897589858995900590159025903590459055906590759085909591059115912591359145915591659175918591959205921592259235924592559265927592859295930593159325933593459355936593759385939594059415942594359445945594659475948594959505951595259535954595559565957595859595960596159625963596459655966596759685969597059715972597359745975597659775978597959805981598259835984598559865987598859895990599159925993599459955996599759985999600060016002600360046005600660076008600960106011601260136014601560166017601860196020602160226023602460256026602760286029603060316032603360346035603660376038603960406041604260436044604560466047604860496050605160526053605460556056605760586059606060616062606360646065606660676068606960706071607260736074607560766077607860796080608160826083608460856086608760886089609060916092609360946095609660976098609961006101610261036104610561066107610861096110611161126113611461156116611761186119612061216122612361246125612661276128612961306131613261336134613561366137613861396140614161426143614461456146614761486149615061516152615361546155615661576158615961606161616261636164616561666167616861696170617161726173617461756176617761786179618061816182618361846185618661876188618961906191619261936194619561966197619861996200620162026203620462056206620762086209621062116212621362146215621662176218621962206221622262236224622562266227622862296230623162326233623462356236623762386239624062416242624362446245624662476248624962506251625262536254625562566257625862596260626162626263626462656266626762686269627062716272627362746275627662776278627962806281628262836284628562866287628862896290629162926293629462956296629762986299630063016302630363046305630663076308630963106311631263136314631563166317631863196320632163226323632463256326632763286329633063316332633363346335633663376338633963406341634263436344634563466347634863496350635163526353635463556356635763586359636063616362636363646365636663676368636963706371637263736374637563766377637863796380638163826383638463856386638763886389639063916392639363946395639663976398639964006401640264036404640564066407640864096410641164126413641464156416641764186419642064216422642364246425642664276428642964306431643264336434643564366437643864396440644164426443644464456446644764486449645064516452645364546455645664576458645964606461646264636464646564666467646864696470647164726473647464756476647764786479648064816482648364846485648664876488648964906491649264936494649564966497649864996500650165026503650465056506650765086509651065116512651365146515651665176518651965206521652265236524652565266527652865296530653165326533653465356536653765386539654065416542654365446545654665476548654965506551655265536554655565566557655865596560656165626563656465656566656765686569657065716572657365746575657665776578657965806581658265836584658565866587658865896590659165926593659465956596659765986599660066016602660366046605660666076608660966106611661266136614661566166617661866196620662166226623662466256626662766286629663066316632663366346635663666376638663966406641664266436644664566466647664866496650665166526653665466556656665766586659666066616662666366646665666666676668666966706671667266736674667566766677667866796680668166826683668466856686668766886689669066916692669366946695669666976698669967006701670267036704670567066707670867096710671167126713671467156716671767186719672067216722672367246725672667276728672967306731673267336734
  1. /*##############################################################################
  2. HPCC SYSTEMS software Copyright (C) 2012 HPCC Systems®.
  3. Licensed under the Apache License, Version 2.0 (the "License");
  4. you may not use this file except in compliance with the License.
  5. You may obtain a copy of the License at
  6. http://www.apache.org/licenses/LICENSE-2.0
  7. Unless required by applicable law or agreed to in writing, software
  8. distributed under the License is distributed on an "AS IS" BASIS,
  9. WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  10. See the License for the specific language governing permissions and
  11. limitations under the License.
  12. ############################################################################## */
  13. #include "platform.h"
  14. #ifdef _WIN32
  15. #include <errno.h>
  16. //#include <winsock.h> // for TransmitFile
  17. #endif
  18. #include <sys/types.h>
  19. #include <sys/stat.h>
  20. #include <algorithm>
  21. #if defined (__linux__) || defined (__APPLE__)
  22. #include <time.h>
  23. #include <dirent.h>
  24. #include <utime.h>
  25. #include <sys/syscall.h>
  26. #include <sys/mman.h>
  27. #endif
  28. #if defined (__linux__)
  29. #include <sys/vfs.h>
  30. #include <sys/sendfile.h>
  31. #endif
  32. #if defined (__APPLE__)
  33. #include <sys/mount.h>
  34. #undef MIN
  35. #endif
  36. #include "time.h"
  37. #include "jerror.hpp"
  38. #include "jlib.hpp"
  39. #include "jio.hpp"
  40. #include "jmisc.hpp"
  41. #include "jsort.hpp"
  42. #include "jmutex.hpp"
  43. #include "jfile.hpp"
  44. #include "jfile.ipp"
  45. #include <limits.h>
  46. #include "jexcept.hpp"
  47. #include "jsocket.hpp" // for IpAddress
  48. #include "jregexp.hpp"
  49. #include "portlist.h"
  50. #include "build-config.h"
  51. #include "jprop.hpp"
  52. // #define REMOTE_DISCONNECT_ON_DESTRUCTOR // enable to disconnect on IFile destructor
  53. // this should not be enabled in WindowRemoteDirectory used
  54. #ifdef _DEBUG
  55. #define ASSERTEX(e) assertex(e);
  56. #else
  57. #define ASSERTEX(e)
  58. #endif
  59. #ifdef __64BIT__
  60. #define DEFAULT_STREAM_BUFFER_SIZE 0x100000
  61. #else
  62. // Restrict buffer sizes on 32-bit systems
  63. #define DEFAULT_STREAM_BUFFER_SIZE 0x10000
  64. #endif
  65. #ifdef _WIN32
  66. #define NULLFILE INVALID_HANDLE_VALUE
  67. #else
  68. #define NULLFILE -1
  69. #endif
  70. // #define CFILEIOTRACE 1
  71. static IFile *createIFileByHook(const RemoteFilename & filename);
  72. static IFile *createContainedIFileByHook(const char *filename);
  73. static inline bool isPCFlushAllowed();
  74. static char ShareChar='$';
  75. bool isShareChar(char c)
  76. {
  77. return (c==ShareChar)||(c=='$');
  78. }
  79. char getShareChar()
  80. {
  81. return ShareChar;
  82. }
  83. void setShareChar(char c)
  84. {
  85. ShareChar = c;
  86. }
  87. StringBuffer &setPathDrive(StringBuffer &filename,unsigned drvnum)
  88. {
  89. return swapPathDrive(filename,(unsigned)-1,drvnum);
  90. }
  91. unsigned getPathDrive(const char *s)
  92. {
  93. char c = *s;
  94. if (isPathSepChar(c)&&(s[1]==c)) {
  95. s = strchr(s+2,c);
  96. if (!s)
  97. return 0;
  98. }
  99. if (isPathSepChar(c))
  100. s++;
  101. if (*s&&((s[1]==':')||(isShareChar(s[1]))))
  102. return *s-'c';
  103. return 0;
  104. }
  105. StringBuffer &swapPathDrive(StringBuffer &filename,unsigned fromdrvnum,unsigned todrvnum,const char *frommask,const char *tomask)
  106. {
  107. const char *s = filename.str();
  108. char c = *s;
  109. if (isPathSepChar(c)&&(s[1]==c)) {
  110. s = strchr(s+2,c);
  111. if (!s)
  112. return filename;
  113. }
  114. if (isPathSepChar(c))
  115. s++;
  116. if (*s&&((fromdrvnum==(unsigned)-1)||(*s==(char) (fromdrvnum+'c')))&&((s[1]==':')||(isShareChar(s[1]))))
  117. filename.setCharAt((size32_t)(s-filename.str()),todrvnum+'c');
  118. else if (frommask&&*frommask) { // OSS
  119. StringBuffer tmp;
  120. if (replaceConfigurationDirectoryEntry(filename.str(),frommask,tomask, tmp))
  121. tmp.swapWith(filename);
  122. }
  123. return filename;
  124. }
  125. const char *pathTail(const char *path)
  126. {
  127. if (!path)
  128. return NULL;
  129. const char *tail=path;
  130. const char *s = path;
  131. while (*s)
  132. if (isPathSepChar(*(s++)))
  133. tail = s;
  134. return tail;
  135. }
  136. const char * pathExtension(const char * path)
  137. {
  138. const char * tail = pathTail(path);
  139. if (tail)
  140. return strrchr(tail, '.');
  141. return NULL;
  142. }
  143. bool checkFileExists(const char * filename)
  144. {
  145. #ifdef _WIN32
  146. for (unsigned i=0;i<10;i++) {
  147. DWORD ret = (DWORD)GetFileAttributes(filename);
  148. if (ret!=(DWORD)-1)
  149. return true;
  150. DWORD err = GetLastError();
  151. if (err!=ERROR_IO_PENDING)
  152. break;
  153. Sleep(100*i);
  154. }
  155. return false;
  156. #else
  157. struct stat info;
  158. return (stat(filename, &info) == 0);
  159. #endif
  160. }
  161. bool checkDirExists(const char * filename)
  162. {
  163. #ifdef _WIN32
  164. DWORD attr = GetFileAttributes(filename);
  165. return (attr != (DWORD)-1)&&(attr & FILE_ATTRIBUTE_DIRECTORY);
  166. #else
  167. struct stat info;
  168. if (stat(filename, &info) != 0)
  169. return false;
  170. return S_ISDIR(info.st_mode);
  171. #endif
  172. }
  173. static void set_inherit(HANDLE handle, bool inherit)
  174. {
  175. #ifndef _WIN32
  176. long flag = fcntl(handle, F_GETFD);
  177. if(inherit)
  178. flag &= ~FD_CLOEXEC;
  179. else
  180. flag |= FD_CLOEXEC;
  181. fcntl(handle, F_SETFD, flag);
  182. #endif
  183. }
  184. static StringBuffer &getLocalOrRemoteName(StringBuffer &name,const RemoteFilename & filename)
  185. {
  186. if (filename.isLocal()&&!filename.isNull()) {
  187. filename.getLocalPath(name);
  188. #ifdef _WIN32
  189. // kludge to allow local linux paths on windows
  190. const char *n=name.str();
  191. if (n[0]=='/') {
  192. StringBuffer tmp;
  193. if (isShareChar(n[2])) {
  194. tmp.append(n[1]).append(':');
  195. n+=3;
  196. }
  197. while (*n) {
  198. if (*n=='/')
  199. tmp.append('\\');
  200. else
  201. tmp.append(*n);
  202. n++;
  203. }
  204. name.clear().append(tmp);
  205. }
  206. #endif
  207. }
  208. else
  209. filename.getRemotePath(name);
  210. return name;
  211. }
  212. CFile::CFile(const char * _filename)
  213. {
  214. filename.set(_filename);
  215. flags = ((unsigned)IFSHread)|((S_IRUSR|S_IWUSR|S_IRGRP|S_IWGRP|S_IROTH|S_IWOTH)<<16);
  216. }
  217. void CFile::setCreateFlags(unsigned short cflags)
  218. {
  219. flags &= 0xffff;
  220. flags |= ((unsigned)cflags<<16);
  221. }
  222. void CFile::setShareMode(IFSHmode shmode)
  223. {
  224. flags &= ~(IFSHfull|IFSHread);
  225. flags |= (unsigned)(shmode&(IFSHfull|IFSHread));
  226. }
  227. bool CFile::exists()
  228. {
  229. if (stdIoHandle(filename)>=0)
  230. return true;
  231. return checkFileExists(filename);
  232. }
  233. #ifdef _WIN32
  234. bool WindowsCreateDirectory(const char * path)
  235. {
  236. unsigned retry = 0;
  237. for (;;) {
  238. if (CreateDirectory(path, NULL))
  239. return true;
  240. DWORD err = GetLastError();
  241. if ((err==ERROR_FILE_NOT_FOUND) || (err==ERROR_PATH_NOT_FOUND) || (err==ERROR_FILE_EXISTS) || (err==ERROR_CANNOT_MAKE))
  242. break;
  243. else if (err==ERROR_ALREADY_EXISTS) {
  244. DWORD attr = GetFileAttributes(path);
  245. if ((attr != -1)&&( attr & FILE_ATTRIBUTE_DIRECTORY))
  246. return true;
  247. return false;
  248. }
  249. if ((retry++==10)|| // some or all of the following can occur when the domain controller gets busy
  250. // retrying improves chance of success
  251. ((err!=ERROR_NETNAME_DELETED)&&(err!=ERROR_DEV_NOT_EXIST)&&(err!=ERROR_GEN_FAILURE)&&(err!=ERROR_NETWORK_BUSY)&&(err!=ERROR_BAD_NET_NAME)))
  252. throw makeOsExceptionV(err,"WindowsCreateDirectory %s", path);
  253. //PROGLOG("Retrying(%d) WindowsCreateDirectory %s, err=%d",retry,filename,err);
  254. Sleep(retry*100);
  255. }
  256. return false;
  257. }
  258. #else
  259. bool LinuxCreateDirectory(const char * path)
  260. {
  261. if (!path)
  262. return false;
  263. if (CreateDirectory(path, NULL))
  264. return true;
  265. else
  266. {
  267. if (EEXIST == errno)
  268. {
  269. struct stat info;
  270. if (stat(path, &info) != 0)
  271. return false;
  272. return S_ISDIR(info.st_mode);
  273. }
  274. }
  275. return false;
  276. }
  277. #endif
  278. bool localCreateDirectory(const char *name)
  279. {
  280. if (!name)
  281. return false;
  282. size32_t l = (size32_t)strlen(name);
  283. if (l==0)
  284. return true;
  285. if (isPathSepChar(name[0])&&((l==1)||(isPathSepChar(name[1])&&!containsPathSepChar(name+2))))
  286. return true;
  287. #ifdef _WIN32
  288. if (name[1]==':') {
  289. if ((l==2)||((l==3)&&isPathSepChar(name[2])))
  290. return true;
  291. }
  292. #endif
  293. if (checkDirExists(name))
  294. return true;
  295. #ifdef _WIN32
  296. if (WindowsCreateDirectory(name))
  297. #else
  298. if (LinuxCreateDirectory(name))
  299. #endif
  300. return true;
  301. if (isPathSepChar(name[l-1])) l--;
  302. while (l&&!isPathSepChar(name[l-1]))
  303. l--;
  304. if (l<=1)
  305. return true;
  306. StringAttr parent(name,l-1);
  307. if (!localCreateDirectory(parent.get()))
  308. return false;
  309. #ifdef _WIN32
  310. return (WindowsCreateDirectory(name));
  311. #else
  312. return (LinuxCreateDirectory(name));
  313. #endif
  314. }
  315. bool CFile::createDirectory()
  316. {
  317. return localCreateDirectory(filename);
  318. }
  319. #ifdef _WIN32
  320. union TimeIntegerUnion
  321. {
  322. FILETIME ft;
  323. ULARGE_INTEGER l;
  324. };
  325. void FILETIMEtoIDateTime(CDateTime * target, const FILETIME & ft)
  326. {
  327. if (target)
  328. {
  329. TimeIntegerUnion u;
  330. memcpy(&u, &ft, sizeof(ft));
  331. unsigned __int64 hundredNanoseconds = (u.l.QuadPart % 10000000);
  332. u.l.QuadPart -= hundredNanoseconds;
  333. SYSTEMTIME systime;
  334. FileTimeToSystemTime(&u.ft, &systime);
  335. target->set(systime.wYear, systime.wMonth, systime.wDay, systime.wHour, systime.wMinute, systime.wSecond, (unsigned)(hundredNanoseconds*100));
  336. }
  337. }
  338. FILETIME * IDateTimetoFILETIME(FILETIME & ft, const CDateTime * dt)
  339. {
  340. if (!dt)
  341. return NULL;
  342. SYSTEMTIME systime;
  343. TimeIntegerUnion u;
  344. unsigned wYear, wMonth, wDay, wHour, wMinute, wSecond, nanoSeconds;
  345. dt->getDate(wYear, wMonth, wDay);
  346. dt->getTime(wHour, wMinute, wSecond, nanoSeconds);
  347. systime.wYear = wYear;
  348. systime.wMonth = wMonth;
  349. systime.wDay = wDay;
  350. systime.wHour = wHour;
  351. systime.wMinute = wMinute;
  352. systime.wSecond = wSecond;
  353. systime.wMilliseconds = 0;
  354. SystemTimeToFileTime(&systime, &u.ft);
  355. //Adjust the fractions of a second ourselves because the function above is only accurate to milli seconds.
  356. u.l.QuadPart += (nanoSeconds / 100);
  357. memcpy(&ft, &u, sizeof(ft));
  358. return &ft;
  359. }
  360. #endif
  361. bool CFile::getTime(CDateTime * createTime, CDateTime * modifiedTime, CDateTime * accessedTime)
  362. {
  363. #ifdef _WIN32
  364. //MORE could use GetFileAttributesEx() if we were allowed...
  365. FILETIME timeCreated, timeModified, timeAccessed;
  366. HANDLE handle = CreateFile(filename, GENERIC_READ, FILE_SHARE_READ, NULL, OPEN_EXISTING, 0, NULL);
  367. if (handle==(HANDLE)-1)
  368. return false;
  369. GetFileTime(handle, &timeCreated, &timeAccessed, &timeModified);
  370. CloseHandle(handle);
  371. FILETIMEtoIDateTime(createTime, timeCreated);
  372. FILETIMEtoIDateTime(modifiedTime, timeModified);
  373. FILETIMEtoIDateTime(accessedTime, timeAccessed);
  374. #else
  375. struct stat info;
  376. if (stat(filename, &info) != 0)
  377. return false;
  378. timetToIDateTime(accessedTime, info.st_atime);
  379. timetToIDateTime(createTime, info.st_ctime);
  380. timetToIDateTime(modifiedTime, info.st_mtime);
  381. #endif
  382. return true;
  383. }
  384. bool CFile::setTime(const CDateTime * createTime, const CDateTime * modifiedTime, const CDateTime * accessedTime)
  385. {
  386. #ifdef _WIN32
  387. FILETIME timeCreated, timeModified, timeAccessed;
  388. FILETIME *pTimeCreated, *pTimeModified, *pTimeAccessed;
  389. pTimeCreated = IDateTimetoFILETIME(timeCreated, createTime);
  390. pTimeModified = IDateTimetoFILETIME(timeModified, modifiedTime);
  391. pTimeAccessed = IDateTimetoFILETIME(timeAccessed, accessedTime);
  392. HANDLE handle = CreateFile(filename, GENERIC_WRITE, FILE_SHARE_READ, NULL, OPEN_EXISTING, 0, NULL);
  393. if (!handle)
  394. return false;
  395. SetFileTime(handle, pTimeCreated, pTimeAccessed, pTimeModified);
  396. CloseHandle(handle);
  397. return true;
  398. #else
  399. struct utimbuf am;
  400. if (!accessedTime||!modifiedTime) {
  401. struct stat info;
  402. if (stat(filename, &info) != 0)
  403. return false;
  404. am.actime = info.st_atime;
  405. am.modtime = info.st_mtime;
  406. }
  407. if (accessedTime)
  408. am.actime = timetFromIDateTime (accessedTime);
  409. if (modifiedTime)
  410. am.modtime = timetFromIDateTime (modifiedTime);
  411. return (utime(filename, &am)==0);
  412. #endif
  413. }
  414. fileBool CFile::isDirectory()
  415. {
  416. #ifdef _WIN32
  417. DWORD attr = GetFileAttributes(filename);
  418. if (attr == -1)
  419. return notFound;
  420. return ( attr & FILE_ATTRIBUTE_DIRECTORY) ? foundYes : foundNo;
  421. #else
  422. struct stat info;
  423. if (stat(filename, &info) != 0)
  424. return notFound;
  425. return S_ISDIR(info.st_mode) ? foundYes : foundNo;
  426. #endif
  427. }
  428. fileBool CFile::isFile()
  429. {
  430. if (stdIoHandle(filename)>=0)
  431. return foundYes;
  432. #ifdef _WIN32
  433. DWORD attr = GetFileAttributes(filename);
  434. if (attr == -1)
  435. return notFound;
  436. return ( attr & FILE_ATTRIBUTE_DIRECTORY) ? foundNo : foundYes;
  437. #else
  438. struct stat info;
  439. if (stat(filename, &info) != 0)
  440. return notFound;
  441. return S_ISREG(info.st_mode) ? foundYes : foundNo;
  442. #endif
  443. }
  444. fileBool CFile::isReadOnly()
  445. {
  446. #ifdef _WIN32
  447. DWORD attr = GetFileAttributes(filename);
  448. if (attr == -1)
  449. return notFound;
  450. return ( attr & FILE_ATTRIBUTE_READONLY) ? foundYes : foundNo;
  451. #else
  452. struct stat info;
  453. if (stat(filename, &info) != 0)
  454. return notFound;
  455. //MORE: I think this is correct, but someone with better unix knowledge should check!
  456. return (info.st_mode & (S_IWUSR|S_IWGRP|S_IWOTH)) ? foundNo : foundYes;
  457. #endif
  458. }
  459. #ifndef _WIN32
  460. static bool setShareLock(int fd,IFSHmode share)
  461. {
  462. struct flock fl;
  463. do {
  464. memset(&fl,0,sizeof(fl));
  465. if (share==IFSHnone)
  466. fl.l_type = F_WRLCK;
  467. else if (share==IFSHread)
  468. fl.l_type = F_RDLCK;
  469. else
  470. fl.l_type = F_UNLCK;
  471. fl.l_whence = SEEK_SET;
  472. fl.l_pid = getpid();
  473. if (fcntl(fd, F_SETLK, &fl) != -1)
  474. return true;
  475. } while (errno==EINTR);
  476. if ((errno!=EAGAIN)&&(errno!=EACCES))
  477. ERRLOG("setShareLock failure %d, mode %d", errno,(int)share);
  478. return (share==IFSHfull); // always return true for full
  479. }
  480. #endif
  481. HANDLE CFile::openHandle(IFOmode mode, IFSHmode sharemode, bool async, int stdh)
  482. {
  483. HANDLE handle = NULLFILE;
  484. #ifdef _WIN32
  485. if (stdh>=0) {
  486. DWORD mode;
  487. switch (stdh) {
  488. case 0: mode = STD_INPUT_HANDLE; break;
  489. case 1: mode = STD_OUTPUT_HANDLE; break;
  490. case 2: mode = STD_ERROR_HANDLE; break;
  491. default:
  492. return handle;
  493. }
  494. DuplicateHandle(GetCurrentProcess(), GetStdHandle(mode), GetCurrentProcess(), &handle , 0, FALSE, DUPLICATE_SAME_ACCESS);
  495. return handle;
  496. }
  497. DWORD share = 0;
  498. switch (sharemode) {
  499. case (IFSHfull|IFSHread):
  500. case IFSHfull:
  501. share |= FILE_SHARE_WRITE;
  502. // fall through
  503. case IFSHread:
  504. share |= FILE_SHARE_READ;
  505. }
  506. DWORD fflags = async?FILE_FLAG_OVERLAPPED:0;
  507. if (async&&(mode==IFOread))
  508. fflags |= FILE_FLAG_SEQUENTIAL_SCAN;
  509. switch (mode) {
  510. case IFOcreate:
  511. handle = CreateFile(filename, GENERIC_WRITE, share, NULL, CREATE_ALWAYS, fflags, NULL);
  512. break;
  513. case IFOread:
  514. handle = CreateFile(filename, GENERIC_READ, share, NULL, OPEN_EXISTING, fflags, NULL);
  515. break;
  516. case IFOwrite:
  517. handle = CreateFile(filename, GENERIC_WRITE, share, NULL, OPEN_ALWAYS, fflags, NULL);
  518. break;
  519. case IFOcreaterw:
  520. handle = CreateFile(filename, GENERIC_WRITE|GENERIC_READ, share, NULL, CREATE_ALWAYS, fflags, NULL);
  521. break;
  522. case IFOreadwrite:
  523. handle = CreateFile(filename, GENERIC_WRITE|GENERIC_READ, share, NULL, OPEN_ALWAYS, fflags, NULL);
  524. break;
  525. }
  526. if (handle == NULLFILE)
  527. {
  528. DWORD err = GetLastError();
  529. if ((IFOread!=mode) || ((err!=ERROR_FILE_NOT_FOUND) && (err!=ERROR_PATH_NOT_FOUND)))
  530. throw makeOsExceptionV(err,"CFile::open %s (%x, %x)", filename.get(), mode, share);
  531. return NULLFILE;
  532. }
  533. #else
  534. if (stdh>=0)
  535. return (HANDLE)dup(stdh);
  536. unsigned fileflags = (flags>>16) & (S_IRUSR|S_IWUSR|S_IXUSR|S_IRGRP|S_IWGRP|S_IXGRP|S_IROTH|S_IWOTH|S_IXOTH);
  537. unsigned openflags;
  538. switch (mode) {
  539. case IFOcreate:
  540. openflags = O_WRONLY | O_CREAT | O_TRUNC;
  541. break;
  542. case IFOread:
  543. openflags = O_RDONLY;
  544. break;
  545. case IFOwrite:
  546. openflags = O_WRONLY | O_CREAT;
  547. break;
  548. case IFOcreaterw:
  549. openflags = O_RDWR | O_CREAT | O_TRUNC;
  550. break;
  551. case IFOreadwrite:
  552. openflags = O_RDWR | O_CREAT;
  553. break;
  554. default:
  555. return NULLFILE;
  556. }
  557. handle = _lopen(filename.get(), openflags, fileflags);
  558. if (handle == -1)
  559. {
  560. if ((IFOread!=mode) || (errno != ENOENT))
  561. throw makeErrnoExceptionV(errno, "CFile::open %s", filename.get());
  562. return NULLFILE;
  563. }
  564. // check not a directory (compatible with windows)
  565. struct stat info;
  566. if (fstat(handle, &info) == -1) {
  567. int err = errno;
  568. close(handle);
  569. handle = NULLFILE;
  570. throw makeErrnoExceptionV(err, "CFile::open fstat %s", filename.get());
  571. }
  572. if (S_ISDIR(info.st_mode)) {
  573. close(handle);
  574. handle = NULLFILE;
  575. throw makeErrnoExceptionV(EISDIR, "CFile::open %s", filename.get());
  576. }
  577. #ifdef CFILEIOTRACE
  578. DBGLOG("CFile::openHandle(%s,%d) returns %d", filename.get(), mode, handle);
  579. #endif
  580. #endif
  581. return handle;
  582. }
  583. IFileIO * CFile::open(IFOmode mode,IFEflags extraFlags)
  584. {
  585. // we may want mode dependant defaults later
  586. return openShared(mode,(IFSHmode)(flags&(IFSHfull|IFSHread)),extraFlags);
  587. }
  588. IFileAsyncIO * CFile::openAsync(IFOmode mode)
  589. {
  590. HANDLE handle = openHandle(mode,IFSHread,true); // I don't think we want shared write to an async file
  591. #ifndef _WIN32
  592. set_inherit(handle, false);
  593. #endif
  594. return new CFileAsyncIO(handle,IFSHread);
  595. }
  596. const char * CFile::queryFilename()
  597. {
  598. return filename;
  599. }
  600. bool CFile::remove()
  601. {
  602. #ifdef _WIN32
  603. unsigned retry = 0;
  604. for (;;) {
  605. if (isDirectory()==foundYes) {
  606. if (RemoveDirectory(filename) != 0)
  607. return true;
  608. }
  609. else {
  610. if (DeleteFile(filename) != 0)
  611. return true;
  612. }
  613. DWORD err = GetLastError();
  614. if ( (err==ERROR_FILE_NOT_FOUND) || (err==ERROR_PATH_NOT_FOUND) )
  615. break;
  616. if ((retry++==10)|| // some or all of the following can occur when the domain controller gets busy
  617. // retrying improves chance of success
  618. ((err!=ERROR_NETNAME_DELETED)&&(err!=ERROR_DEV_NOT_EXIST)&&(err!=ERROR_GEN_FAILURE)&&(err!=ERROR_NETWORK_BUSY)&&(err!=ERROR_BAD_NET_NAME)))
  619. throw makeOsExceptionV(err, "CFile::remove %s", filename.get());
  620. //PROGLOG("Retrying(%d) DeleteFile %s, err=%d",retry,filename,err);
  621. Sleep(retry*100);
  622. }
  623. return false;
  624. #else
  625. if (isDirectory()==foundYes) {
  626. if (rmdir(filename) == 0)
  627. return true;
  628. }
  629. else {
  630. if (unlink(filename) == 0)
  631. return true;
  632. }
  633. if (ENOENT!=errno)
  634. throw makeErrnoExceptionV("CFile::remove %s", filename.get());
  635. return false;
  636. #endif
  637. }
  638. void CFile::rename(const char *newname)
  639. {
  640. // now hopefully newname is just file tail
  641. // however we do allow full paths and extract tail, warning if the directory appears to mismatch
  642. StringBuffer path;
  643. splitDirTail(filename,path);
  644. StringBuffer newdir;
  645. const char *tail = splitDirTail(newname,newdir);
  646. if (path.length()&&newdir.length()) {
  647. if (strcmp(newdir.str(),path.str())!=0)
  648. WARNLOG("CFile::rename '%s' to '%s' : directory mismatch",filename.get(),newname);
  649. }
  650. const char *dst = path.append(tail);
  651. if (isPathSepChar(dst[0])&&(dst[1]==dst[0])) { // hmm is share - convert to local path
  652. RemoteFilename rfn;
  653. rfn.setRemotePath(dst);
  654. if (rfn.isLocal())
  655. dst = rfn.getLocalPath(path.clear()).str();
  656. }
  657. if (-1 == ::rename(filename, dst))
  658. throw makeErrnoExceptionV("CFile::rename(%s, %s)", filename.get(),dst);
  659. filename.set(path);
  660. }
  661. void CFile::move(const char *newname)
  662. {
  663. if (!newname||!*newname)
  664. return;
  665. StringBuffer path;
  666. if (isPathSepChar(newname[0])&&(newname[1]==newname[0])) { // hmm is share - convert to local path
  667. RemoteFilename rfn;
  668. rfn.setRemotePath(newname);
  669. if (rfn.isLocal())
  670. newname = rfn.getLocalPath(path.clear()).str();
  671. }
  672. #ifdef _WIN32
  673. unsigned retry = 0;
  674. for (;;) {
  675. if (MoveFileEx(filename.get(),newname,0))
  676. return;
  677. DWORD err = GetLastError();
  678. if ((retry++==10)|| // some or all of the following can occur when the domain controller gets busy
  679. // retrying improves chance of success
  680. ((err!=ERROR_NETNAME_DELETED)&&(err!=ERROR_DEV_NOT_EXIST)&&(err!=ERROR_GEN_FAILURE)&&(err!=ERROR_NETWORK_BUSY)&&(err!=ERROR_BAD_NET_NAME)))
  681. throw makeOsExceptionV(err, "CFile::move(%s, %s)", filename.get(), newname);
  682. Sleep(retry*100);
  683. }
  684. #else
  685. int ret=::rename(filename.get(),newname);
  686. if (ret==-1)
  687. throw makeErrnoExceptionV("CFile::move(%s, %s)", filename.get(), newname);
  688. #endif
  689. filename.set(newname);
  690. }
  691. #ifdef _WIN32
  692. DWORD CALLBACK fastCopyProgressRoutine(
  693. LARGE_INTEGER TotalFileSize,
  694. LARGE_INTEGER TotalBytesTransferred,
  695. LARGE_INTEGER StreamSize,
  696. LARGE_INTEGER StreamBytesTransferred,
  697. DWORD dwStreamNumber,
  698. DWORD dwCallbackReason,
  699. HANDLE hSourceFile,
  700. HANDLE hDestinationFile,
  701. LPVOID lpData
  702. )
  703. {
  704. if (TotalBytesTransferred.QuadPart<=TotalFileSize.QuadPart)
  705. {
  706. CFPmode status = ((ICopyFileProgress *)lpData)->onProgress(TotalBytesTransferred.QuadPart,TotalFileSize.QuadPart);
  707. switch (status)
  708. {
  709. case CFPcontinue:
  710. return PROGRESS_CONTINUE;
  711. case CFPstop:
  712. return PROGRESS_STOP;
  713. default:
  714. return PROGRESS_CANCEL;
  715. }
  716. }
  717. return PROGRESS_CONTINUE;
  718. }
  719. #endif
  720. bool CFile::fastCopyFile(CFile &target, size32_t buffersize, ICopyFileProgress *progress)
  721. {
  722. // only currently supported for windows
  723. #ifdef _WIN32
  724. unsigned retry = 0;
  725. for (;;) {
  726. BOOL cancel=FALSE;
  727. if (CopyFileEx(queryFilename(),target.queryFilename(),progress?fastCopyProgressRoutine:NULL,progress?progress:NULL,&cancel,0))
  728. break;
  729. DWORD err = GetLastError();
  730. if ( (err==ERROR_FILE_NOT_FOUND) || (err==ERROR_PATH_NOT_FOUND) )
  731. return false;
  732. if ((retry++==10)|| // some or all of the following can occur when the domain controller gets busy
  733. // retrying improves chance of success
  734. ((err!=ERROR_NETNAME_DELETED)&&(err!=ERROR_DEV_NOT_EXIST)&&(err!=ERROR_GEN_FAILURE)&&(err!=ERROR_NETWORK_BUSY)&&(err!=ERROR_BAD_NET_NAME)))
  735. return false;
  736. Sleep(retry*100);
  737. }
  738. return true;
  739. #else
  740. return false;
  741. #endif
  742. }
  743. void CFile::copySection(const RemoteFilename &dest, offset_t toOfs, offset_t fromOfs, offset_t size, ICopyFileProgress *progress, CFflags copyFlags)
  744. {
  745. // check to see if src and target are remote
  746. Owned<IFile> target = createIFile(dest);
  747. const size32_t buffersize = DEFAULT_COPY_BLKSIZE;
  748. IFOmode omode = IFOwrite;
  749. if (toOfs==(offset_t)-1) {
  750. if (fromOfs==0) {
  751. copyFile(target,this,buffersize,progress,copyFlags);
  752. return;
  753. }
  754. omode = IFOcreate;
  755. toOfs = 0;
  756. }
  757. IFEflags tgtFlags = IFEnone;
  758. if (copyFlags & CFflush_write)
  759. tgtFlags = IFEnocache;
  760. OwnedIFileIO targetIO = target->open(omode, tgtFlags);
  761. if (!targetIO)
  762. throw MakeStringException(-1, "copyFile: target path '%s' could not be created", target->queryFilename());
  763. MemoryAttr mb;
  764. void * buffer = mb.allocate(buffersize);
  765. IFEflags srcFlags = IFEnone;
  766. if (copyFlags & CFflush_read)
  767. srcFlags = IFEnocache;
  768. OwnedIFileIO sourceIO = open(IFOread, srcFlags);
  769. if (!sourceIO)
  770. throw MakeStringException(-1, "copySection: source '%s' not found", queryFilename());
  771. offset_t offset = 0;
  772. offset_t total;
  773. try
  774. {
  775. total = sourceIO->size();
  776. if (total<fromOfs)
  777. total = 0;
  778. else
  779. total -= fromOfs;
  780. if (total>size)
  781. total = size;
  782. while (offset<total)
  783. {
  784. size32_t got = sourceIO->read(fromOfs+offset, (buffersize>total-offset)?((size32_t)(total-offset)):buffersize, buffer);
  785. if (got == 0)
  786. break;
  787. targetIO->write(offset+toOfs, got, buffer);
  788. offset += got;
  789. if (progress && progress->onProgress(offset, total) != CFPcontinue)
  790. break;
  791. }
  792. }
  793. catch (IException *e)
  794. {
  795. StringBuffer s;
  796. s.append("copyFile target=").append(target->queryFilename()).append(" source=").append(queryFilename()).append("; read/write failure").append(": ");
  797. e->errorMessage(s);
  798. IException *e2 = makeOsExceptionV(e->errorCode(), "%s", s.str());
  799. e->Release();
  800. throw e2;
  801. }
  802. }
  803. void CFile::copyTo(IFile *dest, size32_t buffersize, ICopyFileProgress *progress,bool usetmp,CFflags copyFlags)
  804. {
  805. doCopyFile(dest,this,buffersize,progress,NULL,usetmp,copyFlags);
  806. }
  807. void CFile::setReadOnly(bool ro)
  808. {
  809. #ifdef _WIN32
  810. if (!SetFileAttributes(filename, ro ? FILE_ATTRIBUTE_READONLY : FILE_ATTRIBUTE_NORMAL))
  811. {
  812. DWORD err = GetLastError();
  813. if ( (err!=ERROR_FILE_NOT_FOUND) && (err!=ERROR_PATH_NOT_FOUND) )
  814. throw makeOsExceptionV(err, "CFile::setReadOnly %s", filename.get());
  815. }
  816. #else
  817. struct stat info;
  818. if (stat(filename, &info) != 0)
  819. throw makeErrnoExceptionV("CFile::setReadOnly() %s", filename.get());
  820. // not sure correct but consistant with isReadOnly
  821. if (ro)
  822. info.st_mode &= ~(S_IWUSR|S_IWGRP|S_IWOTH);
  823. else
  824. info.st_mode |= (S_IWUSR|S_IWGRP|S_IWOTH);
  825. chmod(filename, info.st_mode);
  826. #endif
  827. }
  828. void CFile::setFilePermissions(unsigned fPerms)
  829. {
  830. #ifndef _WIN32
  831. struct stat info;
  832. if (stat(filename, &info) != 0)
  833. throw makeErrnoExceptionV("CFile::setFilePermissions() %s", filename.get());
  834. if (chmod(filename, fPerms&0777) != 0)
  835. throw makeErrnoExceptionV("CFile::setFilePermissions() %s", filename.get());
  836. #endif
  837. }
  838. offset_t CFile::size()
  839. {
  840. if (stdIoHandle(filename)>=0)
  841. return 0; // dummy value
  842. #ifdef _WIN32
  843. WIN32_FILE_ATTRIBUTE_DATA info;
  844. if (GetFileAttributesEx(filename, GetFileExInfoStandard, &info) != 0) {
  845. LARGE_INTEGER x;
  846. x.LowPart = info.nFileSizeLow;
  847. x.HighPart = info.nFileSizeHigh;
  848. return (offset_t)x.QuadPart;
  849. }
  850. #else
  851. struct stat info;
  852. if (stat(filename, &info) == 0)
  853. return info.st_size;
  854. #endif
  855. #if 0
  856. // we could try opening but I don't think needed
  857. Owned<IFileIO>io = openShared(IFOread,IFSHfull);
  858. if (io)
  859. return io->size();
  860. #endif
  861. return (offset_t)-1;
  862. }
  863. bool CFile::setCompression(bool set)
  864. {
  865. #ifdef _WIN32
  866. DWORD attr=::GetFileAttributes(filename.get());
  867. if(attr==-1)
  868. throw makeOsExceptionV(::GetLastError(), "CFile::setCompression %s", filename.get());
  869. if (((attr & FILE_ATTRIBUTE_COMPRESSED) != 0) == set)
  870. return true;
  871. HANDLE handle=::CreateFile(filename.get(),GENERIC_READ|GENERIC_WRITE,0,NULL,OPEN_EXISTING,FILE_FLAG_BACKUP_SEMANTICS,NULL);
  872. if(handle==INVALID_HANDLE_VALUE)
  873. throw makeOsExceptionV(::GetLastError(), "CFile::setCompression %s", filename.get());
  874. USHORT compression=set ? COMPRESSION_FORMAT_DEFAULT : COMPRESSION_FORMAT_NONE;
  875. DWORD bytes;
  876. if(::DeviceIoControl(handle, FSCTL_SET_COMPRESSION, &compression, sizeof(compression), NULL, 0, &bytes, NULL))
  877. {
  878. ::CloseHandle(handle);
  879. return true;
  880. }
  881. DWORD err=::GetLastError();
  882. ::CloseHandle(handle);
  883. throw makeOsExceptionV(err, "CFile::setCompression %s", filename.get());
  884. #else
  885. return false;
  886. #endif
  887. }
  888. offset_t CFile::compressedSize()
  889. {
  890. #ifdef _WIN32
  891. DWORD hi, lo=::GetCompressedFileSize(filename.get(),&hi), err;
  892. if(lo==INVALID_FILE_SIZE && (err=::GetLastError())!=NO_ERROR)
  893. {
  894. if ( (err!=ERROR_FILE_NOT_FOUND) && (err!=ERROR_PATH_NOT_FOUND) )
  895. throw makeOsExceptionV(err,"CFile::compressedSize %s", filename.get());
  896. return -1;
  897. }
  898. return makeint64(hi,lo);
  899. #else
  900. return size();
  901. #endif
  902. }
  903. class CDiscretionaryFileLock: implements IDiscretionaryLock, public CInterface
  904. {
  905. bool locked;
  906. bool excllock;
  907. Linked<IFile> file;
  908. Linked<IFileIO> fileio;
  909. HANDLE handle;
  910. public:
  911. CDiscretionaryFileLock(IFile *_file)
  912. : file(_file)
  913. {
  914. excllock = false;
  915. locked = false;
  916. #ifdef _WIN32
  917. handle=::CreateFile(file->queryFilename(),GENERIC_READ,FILE_SHARE_READ|FILE_SHARE_WRITE|FILE_SHARE_DELETE,NULL,OPEN_EXISTING,FILE_FLAG_BACKUP_SEMANTICS,NULL);
  918. if(handle==INVALID_HANDLE_VALUE)
  919. {
  920. handle = NULLFILE;
  921. throw makeOsExceptionV(GetLastError(), "CDiscretionaryFileLock::openhandle %s", file->queryFilename());
  922. }
  923. #else
  924. handle = _lopen(file->queryFilename(), O_RDONLY, 0);
  925. if (handle == -1)
  926. {
  927. handle = NULLFILE;
  928. throw makeErrnoExceptionV(errno, "CDiscretionaryFileLock::openhandle %s", file->queryFilename());
  929. }
  930. #endif
  931. }
  932. CDiscretionaryFileLock(IFileIO *_fileio)
  933. : fileio(_fileio)
  934. {
  935. excllock = false;
  936. locked = false;
  937. handle = NULLFILE;
  938. CFileIO *cfileio = QUERYINTERFACE(_fileio,CFileIO);
  939. if (cfileio)
  940. handle = cfileio->queryHandle();
  941. if (handle==NULLFILE)
  942. throw makeStringException(-1, "CDiscretionaryFileLock - invalid parameter");
  943. }
  944. ~CDiscretionaryFileLock()
  945. {
  946. if (locked)
  947. unlock();
  948. if ((handle!=NULLFILE)&&!fileio.get())
  949. #ifdef _WIN32
  950. CloseHandle(handle);
  951. #else
  952. _lclose(handle);
  953. #endif
  954. }
  955. IMPLEMENT_IINTERFACE;
  956. virtual bool isLocked() { return (handle!=NULLFILE); }
  957. virtual bool isExclusiveLocked() { return excllock; }
  958. bool lock(bool exclusive=true, unsigned timeout=INFINITE)
  959. {
  960. if (locked) {
  961. if (exclusive) {
  962. if (excllock)
  963. return true;
  964. unlock(); // have to unlock - unfortunate as means no window-less change mode
  965. }
  966. else {
  967. if (!excllock)
  968. return true;
  969. unlock(); // have to unlock - unfortunate as means no window-less change mode
  970. }
  971. }
  972. unsigned interval = 1;
  973. unsigned start = msTick();
  974. for (;;) {
  975. #ifdef _WIN32
  976. OVERLAPPED overlapped;
  977. memset(&overlapped,0,sizeof(overlapped));
  978. if (LockFileEx(handle,
  979. (exclusive?LOCKFILE_EXCLUSIVE_LOCK:0)|((timeout==INFINITE)?0:LOCKFILE_FAIL_IMMEDIATELY),
  980. 0,1,0,&overlapped))
  981. break;
  982. DWORD err = ::GetLastError();
  983. if (err!=ERROR_LOCK_VIOLATION)
  984. throw makeOsException(err, "CDiscretionaryFileLock::lock");
  985. #else
  986. if (setShareLock(handle,exclusive?IFSHnone:IFSHread))
  987. break;
  988. #endif
  989. unsigned now = msTick();
  990. if (now-start>=timeout)
  991. return false;
  992. if (interval>timeout-now+start)
  993. interval = timeout-now+start;
  994. Sleep(interval);
  995. interval *= 2;
  996. if (interval>1000)
  997. interval = 1000;
  998. }
  999. excllock = exclusive;
  1000. locked = true;
  1001. return true;
  1002. }
  1003. void unlock()
  1004. {
  1005. if (handle!=NULLFILE)
  1006. {
  1007. #ifdef _WIN32
  1008. OVERLAPPED overlapped;
  1009. memset(&overlapped,0,sizeof(overlapped));
  1010. if (!UnlockFileEx(handle,0,1,0,&overlapped))
  1011. throw makeOsException(::GetLastError(), "CDiscretionaryFileLock::unlockhandle");
  1012. #else
  1013. setShareLock(handle,IFSHfull);
  1014. #endif
  1015. }
  1016. }
  1017. };
  1018. IDiscretionaryLock *createDiscretionaryLock(IFile *file)
  1019. {
  1020. return new CDiscretionaryFileLock(file);
  1021. }
  1022. IDiscretionaryLock *createDiscretionaryLock(IFileIO *fileio)
  1023. {
  1024. return new CDiscretionaryFileLock(fileio);
  1025. }
  1026. unsigned CFile::getCRC()
  1027. {
  1028. if (stdIoHandle(filename)>=0)
  1029. return 0; // dummy value
  1030. unsigned crc=~0;
  1031. MemoryAttr ma;
  1032. void *buf = ma.allocate(0x10000);
  1033. Owned<IFileIO> fileio = open(IFOread);
  1034. if (fileio) {
  1035. offset_t pos=0;
  1036. for (;;) {
  1037. size32_t rd = fileio->read(pos,0x10000,buf);
  1038. if (!rd)
  1039. break;
  1040. crc=crc32((const char *)buf,rd,crc);
  1041. pos += rd;
  1042. }
  1043. }
  1044. return ~crc;
  1045. }
  1046. //---------------------------------------------------------------------------
  1047. static Linked<IPasswordProvider> passwordProvider;
  1048. MODULE_INIT(INIT_PRIORITY_JFILE)
  1049. {
  1050. return true;
  1051. }
  1052. MODULE_EXIT()
  1053. {
  1054. passwordProvider.clear();
  1055. }
  1056. #ifdef _WIN32
  1057. static bool parseShare(const char *filename,IpAddress &machine,StringBuffer &share)
  1058. { // windows share parsing
  1059. if (!filename||!isPathSepChar(filename[0])||(filename[0]!=filename[1]))
  1060. return false;
  1061. const char *start = filename+2;
  1062. const char * end = start;
  1063. while (*end && !isPathSepChar(*end))
  1064. end++;
  1065. if (!*end)
  1066. return false;
  1067. StringBuffer ipText(end-start, start);
  1068. machine.ipset(ipText.str());
  1069. end++;
  1070. while (*end && !isPathSepChar(*end))
  1071. end++;
  1072. start = filename;
  1073. while (start!=end) {
  1074. if (*start=='/')
  1075. share.append('\\');
  1076. else
  1077. share.append(*start);
  1078. start++;
  1079. }
  1080. return true;
  1081. }
  1082. static CriticalSection connectcrit;
  1083. static bool connectToExternalDrive(const char * const filename)
  1084. {
  1085. CriticalBlock block(connectcrit);
  1086. if (!passwordProvider)
  1087. return false;
  1088. StringBuffer share, username, password;
  1089. IpAddress ip;
  1090. if (!parseShare(filename,ip,share))
  1091. return false;
  1092. if (!passwordProvider->getPassword(ip, username, password))
  1093. return false;
  1094. // first see if connected
  1095. char buf[255];
  1096. buf[0] = 0;
  1097. DWORD len = sizeof(buf);
  1098. DWORD err = WNetGetUser(share.str(),buf,&len);
  1099. if ((err==0)&&(stricmp(username.str(),buf)==0)) {
  1100. // check can access share as well
  1101. for (unsigned i=0;i<10;i++) {
  1102. DWORD ret = (DWORD)GetFileAttributes(share.str());
  1103. DWORD err = (ret==(DWORD)-1)?GetLastError():0;
  1104. if (err!=ERROR_IO_INCOMPLETE)
  1105. break;
  1106. Sleep(100*i);
  1107. }
  1108. if (err==0)
  1109. return true;
  1110. }
  1111. for (unsigned retry=0;retry<5;retry++) {
  1112. NETRESOURCE res;
  1113. memset(&res, 0, sizeof(res));
  1114. res.dwType = RESOURCETYPE_DISK;
  1115. res.lpRemoteName = (char *)share.str();
  1116. err = WNetAddConnection2(&res, password.str(), username.str(), 0);
  1117. if (err) {
  1118. if (err==ERROR_SESSION_CREDENTIAL_CONFLICT) {
  1119. WNetCancelConnection2(res.lpRemoteName, 0, false);
  1120. err = WNetAddConnection2(&res, password.str(), username.str(), 0);
  1121. }
  1122. if (err)
  1123. ERRLOG("WNetAddConnection2(%d): connecting to %s, User: %s",err,res.lpRemoteName,username.str());
  1124. }
  1125. if (err==0)
  1126. return true;
  1127. Sleep(retry*100);
  1128. }
  1129. return false;
  1130. }
  1131. static void disconnectFromExternalDrive(const char * const filename)
  1132. {
  1133. CriticalBlock block(connectcrit);
  1134. StringBuffer share;
  1135. IpAddress ip;
  1136. if (!parseShare(filename,ip,share))
  1137. return;
  1138. if (share.length()&&isShareChar(share.charAt(share.length())))
  1139. WNetCancelConnection2((char *)share.str(), 0, 0);
  1140. }
  1141. class CWindowsRemoteFile : implements IFile, public CInterface
  1142. {
  1143. IFile *ifile;
  1144. StringAttr filename;
  1145. bool connected;
  1146. inline bool connect()
  1147. {
  1148. if (!connected)
  1149. connected = connectToExternalDrive(filename);
  1150. return connected;
  1151. }
  1152. public:
  1153. CWindowsRemoteFile(const char * _filename)
  1154. : filename(_filename)
  1155. {
  1156. connected = false;
  1157. ifile = new CFile(_filename);
  1158. }
  1159. ~CWindowsRemoteFile()
  1160. {
  1161. ifile->Release();
  1162. #ifdef REMOTE_DISCONNECT_ON_DESTRUCTOR
  1163. if (connected)
  1164. disconnectFromExternalDrive(filename);
  1165. #endif
  1166. }
  1167. IMPLEMENT_IINTERFACE
  1168. virtual bool exists()
  1169. {
  1170. connect();
  1171. return ifile->exists();
  1172. }
  1173. virtual bool getTime(CDateTime * createTime, CDateTime * modifiedTime, CDateTime * accessedTime)
  1174. {
  1175. connect();
  1176. return ifile->getTime(createTime, modifiedTime, accessedTime);
  1177. }
  1178. virtual bool setTime(const CDateTime * createTime, const CDateTime * modifiedTime, const CDateTime * accessedTime)
  1179. {
  1180. connect();
  1181. if (ifile->setTime(createTime, modifiedTime, accessedTime))
  1182. return true;
  1183. if (connected||!connect())
  1184. return false;
  1185. return ifile->setTime(createTime, modifiedTime, accessedTime);
  1186. }
  1187. virtual fileBool isDirectory()
  1188. {
  1189. connect();
  1190. fileBool ok = ifile->isDirectory();
  1191. if (ok == notFound && !connected && connect())
  1192. ok = ifile->isDirectory();
  1193. return ok;
  1194. }
  1195. virtual fileBool isFile()
  1196. {
  1197. connect();
  1198. fileBool ok = ifile->isFile();
  1199. if (ok == notFound && !connected && connect())
  1200. ok = ifile->isFile();
  1201. return ok;
  1202. }
  1203. virtual fileBool isReadOnly()
  1204. {
  1205. connect();
  1206. fileBool ok = ifile->isReadOnly();
  1207. if (ok == notFound && !connected && connect())
  1208. ok = ifile->isFile();
  1209. return ok;
  1210. }
  1211. virtual IFileIO * open(IFOmode mode,IFEflags extraFlags=IFEnone)
  1212. {
  1213. connect();
  1214. return ifile->open(mode,extraFlags);
  1215. }
  1216. virtual IFileIO * openShared(IFOmode mode,IFSHmode shared,IFEflags extraFlags=IFEnone)
  1217. {
  1218. connect();
  1219. return ifile->openShared(mode,shared,extraFlags);
  1220. }
  1221. virtual IFileAsyncIO * openAsync(IFOmode mode)
  1222. {
  1223. connect();
  1224. return ifile->openAsync(mode);
  1225. }
  1226. virtual const char * queryFilename()
  1227. {
  1228. return ifile->queryFilename();
  1229. }
  1230. virtual bool remove()
  1231. {
  1232. connect();
  1233. unsigned attempt=0; \
  1234. return ifile->remove();
  1235. }
  1236. virtual void rename(const char *newname)
  1237. {
  1238. connect();
  1239. StringBuffer path;
  1240. splitDirTail(filename,path);
  1241. StringBuffer newdir;
  1242. const char *tail = splitDirTail(newname,newdir);
  1243. if ((newname[0]=='\\')&&(newname[1]=='\\')) { // rename to remote
  1244. if (path.length()&&newdir.length()) {
  1245. if (strcmp(newdir.str(),path.str())!=0)
  1246. WARNLOG("CWindowsRemoteFile '%s' to '%s' : directory mismatch",filename.get(),newname);
  1247. }
  1248. newname = tail; // just rename using tail
  1249. }
  1250. ifile->rename(newname);
  1251. path.append(tail);
  1252. filename.set(path);
  1253. }
  1254. virtual void move(const char *newName)
  1255. {
  1256. connect();
  1257. ifile->move(newName);
  1258. filename.set(ifile->queryFilename());
  1259. }
  1260. virtual void setReadOnly(bool ro)
  1261. {
  1262. connect();
  1263. ifile->setReadOnly(ro);
  1264. }
  1265. virtual void setFilePermissions(unsigned fPerms)
  1266. {
  1267. connect();
  1268. ifile->setFilePermissions(fPerms);
  1269. }
  1270. virtual offset_t size()
  1271. {
  1272. connect();
  1273. offset_t ret=ifile->size();
  1274. if ((ret==(offset_t)-1)&&!connected&&connect())
  1275. ret=ifile->size();
  1276. return ret;
  1277. }
  1278. virtual bool setCompression(bool set)
  1279. {
  1280. connect();
  1281. return ifile->setCompression(set);
  1282. }
  1283. virtual offset_t compressedSize()
  1284. {
  1285. connect();
  1286. return ifile->compressedSize();
  1287. }
  1288. bool fastCopyFile(CWindowsRemoteFile &target, size32_t buffersize, ICopyFileProgress *progress)
  1289. {
  1290. #ifdef _WIN32
  1291. CFile *src = QUERYINTERFACE(ifile,CFile);
  1292. if (!src)
  1293. return false;
  1294. CFile *dst = QUERYINTERFACE(target.ifile,CFile);
  1295. if (!dst)
  1296. return false;
  1297. target.isFile(); // encourage connect on target
  1298. connect();
  1299. return src->fastCopyFile(*dst, buffersize, progress);
  1300. #else
  1301. return false;
  1302. #endif
  1303. }
  1304. bool fastCopyFile(CFile &target, size32_t buffersize, ICopyFileProgress *progress)
  1305. {
  1306. #ifdef _WIN32
  1307. CFile *src = QUERYINTERFACE(ifile,CFile);
  1308. if (!src)
  1309. return false;
  1310. connect();
  1311. return src->fastCopyFile(target, buffersize, progress);
  1312. #else
  1313. return false;
  1314. #endif
  1315. }
  1316. bool fastCopyFileRev(CFile &src, size32_t buffersize, ICopyFileProgress *progress)
  1317. {
  1318. #ifdef _WIN32
  1319. CFile *dst = QUERYINTERFACE(ifile,CFile);
  1320. if (!dst)
  1321. return false;
  1322. connect();
  1323. return src.fastCopyFile(*dst, buffersize, progress);
  1324. #else
  1325. return false;
  1326. #endif
  1327. }
  1328. void copyTo(IFile *dest, size32_t buffersize, ICopyFileProgress *progress, bool usetmp,CFflags copyFlags)
  1329. {
  1330. doCopyFile(dest,this,buffersize,progress,NULL,usetmp,copyFlags);
  1331. }
  1332. bool createDirectory()
  1333. {
  1334. connect();
  1335. return localCreateDirectory(filename);
  1336. }
  1337. IDirectoryIterator *directoryFiles(const char *mask,bool sub,bool includedirs)
  1338. {
  1339. connect();
  1340. return ifile->directoryFiles(mask,sub,includedirs);
  1341. }
  1342. IDirectoryDifferenceIterator *monitorDirectory(
  1343. IDirectoryIterator *prev=NULL, // in (NULL means use current as baseline)
  1344. const char *mask=NULL,
  1345. bool sub=false,
  1346. bool includedirs=false,
  1347. unsigned checkinterval=60*1000,
  1348. unsigned timeout=(unsigned)-1,
  1349. Semaphore *abortsem=NULL) // returns NULL if timed out or abortsem signalled
  1350. {
  1351. connect();
  1352. return ifile->monitorDirectory(prev,mask,sub,includedirs,checkinterval,timeout,abortsem);
  1353. }
  1354. bool getInfo(bool &isdir,offset_t &size,CDateTime &modtime)
  1355. {
  1356. connect();
  1357. return ifile->getInfo(isdir,size,modtime);
  1358. }
  1359. unsigned getCRC()
  1360. {
  1361. connect();
  1362. return ifile->getCRC();
  1363. }
  1364. void setCreateFlags(unsigned short cflags)
  1365. {
  1366. ifile->setCreateFlags(cflags);
  1367. }
  1368. void setShareMode(IFSHmode shmode)
  1369. {
  1370. ifile->setShareMode(shmode);
  1371. }
  1372. void copySection(const RemoteFilename &dest, offset_t toOfs, offset_t fromOfs, offset_t size, ICopyFileProgress *progress, CFflags copyFlags)
  1373. {
  1374. connect();
  1375. ifile->copySection(dest,toOfs,fromOfs,size,progress,copyFlags);
  1376. }
  1377. IMemoryMappedFile *openMemoryMapped(offset_t ofs, memsize_t len, bool write)
  1378. {
  1379. throw MakeStringException(-1,"Remote file cannot be memory mapped");
  1380. return NULL;
  1381. }
  1382. };
  1383. #endif
  1384. IFileIO *_createIFileIO(const void *buffer, unsigned sz, bool readOnly)
  1385. {
  1386. class CMemoryBufferIO : implements IFileIO, public CInterface
  1387. {
  1388. MemoryBuffer mb;
  1389. void *buffer;
  1390. size32_t sz;
  1391. bool readOnly;
  1392. public:
  1393. IMPLEMENT_IINTERFACE;
  1394. CMemoryBufferIO(const void *_buffer, unsigned _sz, bool _readOnly) : readOnly(_readOnly)
  1395. {
  1396. // JCSMORE - should probably have copy as option
  1397. mb.append(_sz, _buffer);
  1398. buffer = (void *)mb.toByteArray();
  1399. sz = mb.length();
  1400. }
  1401. virtual size32_t read(offset_t pos, size32_t len, void * data)
  1402. {
  1403. if (pos>sz)
  1404. throw MakeStringException(-1, "CMemoryBufferIO: read beyond end of buffer pos=%" I64F "d, len=%d, buffer length=%d", pos, len, mb.length());
  1405. if (pos+len > sz)
  1406. len = (size32_t)(sz-pos);
  1407. memcpy(data, (byte *)buffer+pos, len);
  1408. return len;
  1409. }
  1410. virtual offset_t size() { return sz; }
  1411. virtual size32_t write(offset_t pos, size32_t len, const void * data)
  1412. {
  1413. assertex(!readOnly);
  1414. if (pos+len>sz)
  1415. throw MakeStringException(-1, "CMemoryBufferIO: UNIMPLEMENTED, writing beyond buffer, pos=%" I64F "d, len=%d, buffer length=%d", pos, len, mb.length());
  1416. memcpy((byte *)buffer+pos, data, len);
  1417. return len;
  1418. }
  1419. virtual void flush() {}
  1420. virtual void close() {}
  1421. virtual unsigned __int64 getStatistic(StatisticKind kind)
  1422. {
  1423. return 0;
  1424. }
  1425. virtual void setSize(offset_t size)
  1426. {
  1427. if (size > mb.length())
  1428. throw MakeStringException(-1, "CMemoryBufferIO: UNIMPLEMENTED, setting size %" I64F "d beyond end of buffer, buffer length=%d", size, mb.length());
  1429. mb.setLength((size32_t)size);
  1430. }
  1431. offset_t appendFile(IFile *file,offset_t pos,offset_t len)
  1432. {
  1433. if (!file)
  1434. return 0;
  1435. const size32_t buffsize = 0x10000;
  1436. void * buffer = mb.reserve(buffsize);
  1437. Owned<IFileIO> fileio = file->open(IFOread);
  1438. offset_t ret=0;
  1439. while (len) {
  1440. size32_t toread = (len>=buffsize)?buffsize:(size32_t)len;
  1441. size32_t read = fileio->read(pos,toread,buffer);
  1442. if (read<buffsize)
  1443. mb.setLength(mb.length()+read-buffsize);
  1444. if (read==0)
  1445. break;
  1446. pos += read;
  1447. len -= read;
  1448. ret += read;
  1449. }
  1450. return ret;
  1451. }
  1452. };
  1453. return new CMemoryBufferIO(buffer, sz, readOnly);
  1454. }
  1455. IFileIO * createIFileI(unsigned len, const void * buffer)
  1456. {
  1457. return _createIFileIO((void *)buffer, len, true);
  1458. }
  1459. IFileIO * createIFileIO(unsigned len, void * buffer)
  1460. {
  1461. return _createIFileIO(buffer, len, false);
  1462. }
  1463. IFileIO * createIFileIO(StringBuffer & buffer)
  1464. {
  1465. return _createIFileIO((void *)buffer.str(), buffer.length(), true);
  1466. }
  1467. IFileIO * createIFileIO(MemoryBuffer & buffer)
  1468. {
  1469. return _createIFileIO(buffer.toByteArray(), buffer.length(), false);
  1470. }
  1471. //---------------------------------------------------------------------------
  1472. class jlib_decl CSequentialFileIO : public CFileIO
  1473. {
  1474. offset_t pos;
  1475. void checkPos(const char *fn,offset_t _pos)
  1476. {
  1477. if (_pos!=pos)
  1478. throw MakeStringException(-1, "CSequentialFileIO %s out of sequence (%" I64F "d,%" I64F "d)",fn,pos,_pos);
  1479. }
  1480. public:
  1481. CSequentialFileIO(HANDLE h,IFOmode _openmode,IFSHmode _sharemode,IFEflags _extraFlags)
  1482. : CFileIO(h,_openmode,_sharemode,_extraFlags)
  1483. {
  1484. pos = 0;
  1485. }
  1486. ~CSequentialFileIO()
  1487. {
  1488. }
  1489. size32_t read(offset_t _pos, size32_t len, void * data)
  1490. {
  1491. checkPos("read",_pos);
  1492. #ifdef _WIN32
  1493. // Can't use checked_read because don't have the c fileno for it
  1494. DWORD numRead;
  1495. if (ReadFile(file,data,len,&numRead,NULL) == 0) {
  1496. DWORD err = GetLastError();
  1497. if (err==ERROR_BROKEN_PIPE) // windows returns this at end of pipe
  1498. return 0;
  1499. throw makeOsException(GetLastError(),"CSequentialFileIO::read");
  1500. }
  1501. size32_t ret = (size32_t)numRead;
  1502. #else
  1503. size32_t ret = checked_read(file, data, len);
  1504. #endif
  1505. pos += ret;
  1506. return ret;
  1507. }
  1508. virtual size32_t write(offset_t _pos, size32_t len, const void * data)
  1509. {
  1510. checkPos("write",_pos);
  1511. size32_t ret;
  1512. #ifdef _WIN32
  1513. DWORD numWritten;
  1514. if (!WriteFile(file, data, len, &numWritten, NULL))
  1515. throw makeOsException(GetLastError(), "CSequentialFileIO::write");
  1516. if (numWritten != len)
  1517. throw makeOsException(DISK_FULL_EXCEPTION_CODE, "CSequentialFileIO::write");
  1518. ret = (size32_t) numWritten;
  1519. #else
  1520. ret = ::write(file,data,len);
  1521. if (ret==(size32_t)-1)
  1522. {
  1523. PrintStackReport();
  1524. ERRLOG("errno(%d): %" I64F "d %u",errno,pos,len);
  1525. throw makeErrnoException(errno, "CFileIO::write");
  1526. }
  1527. if (ret<len)
  1528. throw makeOsException(DISK_FULL_EXCEPTION_CODE, "CSequentialFileIO::write");
  1529. #endif
  1530. pos += ret;
  1531. return ret;
  1532. }
  1533. };
  1534. IFileIO * CFile::openShared(IFOmode mode,IFSHmode share,IFEflags extraFlags)
  1535. {
  1536. int stdh = stdIoHandle(filename);
  1537. HANDLE handle = openHandle(mode,share,false, stdh);
  1538. if (handle==NULLFILE)
  1539. return NULL;
  1540. #ifndef _WIN32
  1541. set_inherit(handle, false);
  1542. #endif
  1543. if (stdh>=0)
  1544. return new CSequentialFileIO(handle,mode,share,extraFlags);
  1545. return new CFileIO(handle,mode,share,extraFlags);
  1546. }
  1547. //---------------------------------------------------------------------------
  1548. extern jlib_decl IFileIO *createIFileIO(HANDLE handle,IFOmode openmode,IFEflags extraFlags)
  1549. {
  1550. return new CFileIO(handle,openmode,IFSHfull,extraFlags);
  1551. }
  1552. offset_t CFileIO::appendFile(IFile *file,offset_t pos,offset_t len)
  1553. {
  1554. if (!file)
  1555. return 0;
  1556. CriticalBlock procedure(cs);
  1557. MemoryAttr mb;
  1558. const size32_t buffsize = 0x10000;
  1559. void * buffer = mb.allocate(buffsize);
  1560. Owned<IFileIO> fileio = file->open(IFOread);
  1561. offset_t ret=0;
  1562. offset_t outp = size();
  1563. while (len) {
  1564. size32_t toread = (len>=buffsize) ? buffsize : (size32_t)len;
  1565. size32_t read = fileio->read(pos,toread,buffer);
  1566. if (read==0)
  1567. break;
  1568. size32_t wr = write(outp,read,buffer);
  1569. pos += read;
  1570. outp += wr;
  1571. len -= read;
  1572. ret += wr;
  1573. if (wr!=read)
  1574. break;
  1575. }
  1576. return ret;
  1577. }
  1578. unsigned __int64 CFileIO::getStatistic(StatisticKind kind)
  1579. {
  1580. switch (kind)
  1581. {
  1582. case StCycleDiskReadIOCycles:
  1583. return ioReadCycles.load();
  1584. case StCycleDiskWriteIOCycles:
  1585. return ioWriteCycles.load();
  1586. case StTimeDiskReadIO:
  1587. return cycle_to_nanosec(ioReadCycles.load());
  1588. case StTimeDiskWriteIO:
  1589. return cycle_to_nanosec(ioWriteCycles.load());
  1590. case StSizeDiskRead:
  1591. return ioReadBytes.load();
  1592. case StSizeDiskWrite:
  1593. return ioWriteBytes.load();
  1594. case StNumDiskReads:
  1595. return ioReads.load();
  1596. case StNumDiskWrites:
  1597. return ioWrites.load();
  1598. }
  1599. return 0;
  1600. }
  1601. #ifdef _WIN32
  1602. //-- Windows implementation -------------------------------------------------
  1603. CFileIO::CFileIO(HANDLE handle, IFOmode _openmode, IFSHmode _sharemode, IFEflags _extraFlags)
  1604. : ioReadCycles(0), ioWriteCycles(0), ioReadBytes(0), ioWriteBytes(0), ioReads(0), ioWrites(0), unflushedReadBytes(0), unflushedWriteBytes(0)
  1605. {
  1606. assertex(handle != NULLFILE);
  1607. throwOnError = false;
  1608. file = handle;
  1609. sharemode = _sharemode;
  1610. openmode = _openmode;
  1611. extraFlags = _extraFlags; // page cache flush option silently ignored on Windows for now
  1612. if (extraFlags & IFEnocache)
  1613. if (!isPCFlushAllowed())
  1614. extraFlags = static_cast<IFEflags>(extraFlags & ~IFEnocache);
  1615. }
  1616. CFileIO::~CFileIO()
  1617. {
  1618. try
  1619. {
  1620. //note this will not call the virtual close() if anyone ever derived from this class.
  1621. //the clean fix is to move this code to beforeDispose()
  1622. close();
  1623. }
  1624. catch (IException * e)
  1625. {
  1626. EXCLOG(e, "CFileIO::~CFileIO");
  1627. e->Release();
  1628. }
  1629. }
  1630. void CFileIO::close()
  1631. {
  1632. if (file != NULLFILE)
  1633. {
  1634. if (!CloseHandle(file))
  1635. throw makeOsException(GetLastError(),"CFileIO::close");
  1636. }
  1637. file = NULLFILE;
  1638. }
  1639. void CFileIO::flush()
  1640. {
  1641. if (!FlushFileBuffers(file))
  1642. throw makeOsException(GetLastError(),"CFileIO::flush");
  1643. }
  1644. offset_t CFileIO::size()
  1645. {
  1646. LARGE_INTEGER pos;
  1647. pos.LowPart = GetFileSize(file, (unsigned long *)&pos.HighPart);
  1648. if (pos.LowPart==-1) {
  1649. DWORD err = GetLastError();
  1650. if (err!=0)
  1651. throw makeOsException(err,"CFileIO::size");
  1652. }
  1653. return pos.QuadPart;
  1654. }
  1655. size32_t CFileIO::read(offset_t pos, size32_t len, void * data)
  1656. {
  1657. CriticalBlock procedure(cs);
  1658. CCycleTimer timer;
  1659. DWORD numRead;
  1660. setPos(pos);
  1661. if (ReadFile(file,data,len,&numRead,NULL) == 0)
  1662. throw makeOsException(GetLastError(),"CFileIO::read");
  1663. ioReadCycles.fetch_add(timer.elapsedCycles());
  1664. ioReadBytes.fetch_add(numRead);
  1665. ++ioReads;
  1666. return (size32_t)numRead;
  1667. }
  1668. void CFileIO::setPos(offset_t newPos)
  1669. {
  1670. LARGE_INTEGER tempPos;
  1671. tempPos.QuadPart = newPos;
  1672. tempPos.LowPart = SetFilePointer(file, tempPos.LowPart, &tempPos.HighPart, FILE_BEGIN);
  1673. }
  1674. size32_t CFileIO::write(offset_t pos, size32_t len, const void * data)
  1675. {
  1676. CriticalBlock procedure(cs);
  1677. CCycleTimer timer;
  1678. DWORD numWritten;
  1679. setPos(pos);
  1680. if (!WriteFile(file,data,len,&numWritten,NULL))
  1681. throw makeOsException(GetLastError(),"CFileIO::write");
  1682. if (numWritten != len)
  1683. throw makeOsException(DISK_FULL_EXCEPTION_CODE,"CFileIO::write");
  1684. ioWriteCycles.fetch_add(timer.elapsedCycles());
  1685. ioWriteBytes.fetch_add(numWritten);
  1686. ++ioWrites;
  1687. return (size32_t)numWritten;
  1688. }
  1689. void CFileIO::setSize(offset_t pos)
  1690. {
  1691. CriticalBlock procedure(cs);
  1692. setPos(pos);
  1693. if (!SetEndOfFile(file))
  1694. throw makeOsException(GetLastError(), "CFileIO::setSize");
  1695. }
  1696. #else
  1697. //-- Unix implementation ----------------------------------------------------
  1698. // More errorno checking TBD
  1699. CFileIO::CFileIO(HANDLE handle, IFOmode _openmode, IFSHmode _sharemode, IFEflags _extraFlags)
  1700. : ioReadCycles(0), ioWriteCycles(0), ioReadBytes(0), ioWriteBytes(0), ioReads(0), ioWrites(0), unflushedReadBytes(0), unflushedWriteBytes(0)
  1701. {
  1702. assertex(handle != NULLFILE);
  1703. throwOnError = false;
  1704. file = handle;
  1705. sharemode = _sharemode;
  1706. openmode = _openmode;
  1707. extraFlags = _extraFlags;
  1708. if (extraFlags & IFEnocache)
  1709. if (!isPCFlushAllowed())
  1710. extraFlags = static_cast<IFEflags>(extraFlags & ~IFEnocache);
  1711. #ifdef CFILEIOTRACE
  1712. DBGLOG("CFileIO::CfileIO(%d,%d,%d,%d)", handle, _openmode, _sharemode, _extraFlags);
  1713. #endif
  1714. }
  1715. CFileIO::~CFileIO()
  1716. {
  1717. try
  1718. {
  1719. close();
  1720. }
  1721. catch (IException * e)
  1722. {
  1723. EXCLOG(e, "CFileIO::~CFileIO");
  1724. e->Release();
  1725. }
  1726. }
  1727. void CFileIO::close()
  1728. {
  1729. if (file != NULLFILE)
  1730. {
  1731. #ifdef CFILEIOTRACE
  1732. DBGLOG("CFileIO::close(%d), extraFlags = %d", file, extraFlags);
  1733. #endif
  1734. if (extraFlags & IFEnocache)
  1735. {
  1736. if (openmode != IFOread)
  1737. {
  1738. #ifdef F_FULLFSYNC
  1739. fcntl(file, F_FULLFSYNC);
  1740. #else
  1741. fdatasync(file);
  1742. #endif
  1743. }
  1744. #ifdef POSIX_FADV_DONTNEED
  1745. posix_fadvise(file, 0, 0, POSIX_FADV_DONTNEED);
  1746. #endif
  1747. }
  1748. if (::close(file) < 0)
  1749. throw makeErrnoException(errno, "CFileIO::close");
  1750. file=NULLFILE;
  1751. }
  1752. }
  1753. void CFileIO::flush()
  1754. {
  1755. CriticalBlock procedure(cs);
  1756. #ifdef F_FULLFSYNC
  1757. if (fcntl(file, F_FULLFSYNC) != 0)
  1758. #else
  1759. if (fdatasync(file) != 0)
  1760. #endif
  1761. throw makeOsException(DISK_FULL_EXCEPTION_CODE, "CFileIO::flush");
  1762. #ifdef POSIX_FADV_DONTNEED
  1763. if (extraFlags & IFEnocache)
  1764. posix_fadvise(file, 0, 0, POSIX_FADV_DONTNEED);
  1765. #endif
  1766. }
  1767. offset_t CFileIO::size()
  1768. {
  1769. CriticalBlock procedure(cs);
  1770. offset_t savedPos = lseek(file,0,SEEK_CUR);
  1771. offset_t length = lseek(file,0,SEEK_END);
  1772. setPos(savedPos);
  1773. return length;
  1774. }
  1775. size32_t CFileIO::read(offset_t pos, size32_t len, void * data)
  1776. {
  1777. if (0==len) return 0;
  1778. CCycleTimer timer;
  1779. size32_t ret = checked_pread(file, data, len, pos);
  1780. ioReadCycles.fetch_add(timer.elapsedCycles());
  1781. ioReadBytes.fetch_add(ret);
  1782. ++ioReads;
  1783. if ( (extraFlags & IFEnocache) && (ret > 0) )
  1784. {
  1785. if (unflushedReadBytes.add_fetch(ret) >= PGCFLUSH_BLKSIZE)
  1786. {
  1787. unflushedReadBytes.store(0);
  1788. #ifdef POSIX_FADV_DONTNEED
  1789. posix_fadvise(file, 0, 0, POSIX_FADV_DONTNEED);
  1790. #endif
  1791. }
  1792. }
  1793. return ret;
  1794. }
  1795. void CFileIO::setPos(offset_t newPos)
  1796. {
  1797. if (file != NULLFILE)
  1798. _llseek(file,newPos,SEEK_SET);
  1799. }
  1800. static void sync_file_region(int fd, offset_t offset, offset_t nbytes)
  1801. {
  1802. #if defined(__linux__) && defined(__NR_sync_file_range)
  1803. (void)syscall(__NR_sync_file_range, fd, offset,
  1804. nbytes, SYNC_FILE_RANGE_WAIT_BEFORE | SYNC_FILE_RANGE_WRITE);
  1805. #elif defined(F_FULLFSYNC)
  1806. fcntl(fd, F_FULLFSYNC);
  1807. #else
  1808. fdatasync(fd);
  1809. #endif
  1810. }
  1811. size32_t CFileIO::write(offset_t pos, size32_t len, const void * data)
  1812. {
  1813. CCycleTimer timer;
  1814. size32_t ret = pwrite(file,data,len,pos);
  1815. ioWriteCycles.fetch_add(timer.elapsedCycles());
  1816. ioWriteBytes.fetch_add(ret);
  1817. ++ioWrites;
  1818. if (ret==(size32_t)-1)
  1819. throw makeErrnoException(errno, "CFileIO::write");
  1820. if (ret<len)
  1821. throw makeOsException(DISK_FULL_EXCEPTION_CODE, "CFileIO::write");
  1822. if ( (extraFlags & IFEnocache) && (ret > 0) )
  1823. {
  1824. if (unflushedWriteBytes.add_fetch(ret) >= PGCFLUSH_BLKSIZE)
  1825. {
  1826. unflushedWriteBytes.store(0);
  1827. // [possibly] non-blocking request to write-out dirty pages
  1828. sync_file_region(file, 0, 0);
  1829. #ifdef POSIX_FADV_DONTNEED
  1830. // flush written-out pages
  1831. posix_fadvise(file, 0, 0, POSIX_FADV_DONTNEED);
  1832. #endif
  1833. }
  1834. }
  1835. return ret;
  1836. }
  1837. void CFileIO::setSize(offset_t pos)
  1838. {
  1839. if (0 != ftruncate(file, pos))
  1840. throw makeErrnoException(errno, "CFileIO::setSize");
  1841. }
  1842. #endif
  1843. //---------------------------------------------------------------------------
  1844. CFileRangeIO::CFileRangeIO(IFileIO * _io, offset_t _headerSize, offset_t _maxLength)
  1845. {
  1846. io.set(_io);
  1847. headerSize = _headerSize;
  1848. maxLength = _maxLength;
  1849. }
  1850. size32_t CFileRangeIO::read(offset_t pos, size32_t len, void * data)
  1851. {
  1852. if (pos + len > maxLength)
  1853. {
  1854. if (pos > maxLength)
  1855. pos = maxLength;
  1856. len = (size32_t)(maxLength - pos);
  1857. }
  1858. return io->read(pos+headerSize, len, data);
  1859. }
  1860. offset_t CFileRangeIO::size()
  1861. {
  1862. return maxLength;
  1863. }
  1864. size32_t CFileRangeIO::write(offset_t pos, size32_t len, const void * data)
  1865. {
  1866. if (pos + len > maxLength)
  1867. {
  1868. if (pos > maxLength)
  1869. pos = maxLength;
  1870. len = (size32_t)(maxLength - pos);
  1871. }
  1872. return io->write(pos+headerSize, len, data);
  1873. }
  1874. //--------------------------------------------------------------------------
  1875. CFileAsyncIO::~CFileAsyncIO()
  1876. {
  1877. try
  1878. {
  1879. close();
  1880. }
  1881. catch (IException * e)
  1882. {
  1883. EXCLOG(e, "CFileAsyncIO::~CFileAsyncIO");
  1884. e->Release();
  1885. }
  1886. }
  1887. void CFileAsyncIO::flush()
  1888. {
  1889. // wait for all outstanding results
  1890. CriticalBlock block(cs);
  1891. ForEachItemInRev(i,results) {
  1892. size32_t dummy;
  1893. results.item(i).getResult(dummy,true);
  1894. }
  1895. }
  1896. offset_t CFileAsyncIO::appendFile(IFile *file,offset_t pos,offset_t len)
  1897. {
  1898. // will implemented if needed
  1899. UNIMPLEMENTED;
  1900. }
  1901. unsigned __int64 CFileAsyncIO::getStatistic(StatisticKind kind)
  1902. {
  1903. //MORE: Could implement - but I don't think this class is currently used
  1904. return 0;
  1905. }
  1906. #ifdef _WIN32
  1907. //-- Windows implementation -------------------------------------------------
  1908. class CFileAsyncResult: implements IFileAsyncResult, public CInterface
  1909. {
  1910. protected: friend class CFileAsyncIO;
  1911. OVERLAPPED overlapped;
  1912. DWORD value;
  1913. DWORD wrsize;
  1914. CFileAsyncIO *parent;
  1915. public:
  1916. IMPLEMENT_IINTERFACE;
  1917. CFileAsyncResult(offset_t offset,size32_t _wrsize)
  1918. {
  1919. parent = NULL;
  1920. memset(&overlapped,0,sizeof(overlapped));
  1921. overlapped.hEvent = CreateEvent(NULL, TRUE, FALSE, NULL);
  1922. overlapped.Offset = (DWORD)offset;
  1923. overlapped.OffsetHigh = (DWORD)(offset>>32);
  1924. value = (size32_t)-1;
  1925. wrsize = _wrsize;
  1926. }
  1927. ~CFileAsyncResult()
  1928. {
  1929. size32_t dummy;
  1930. getResult(dummy,true);
  1931. CloseHandle(overlapped.hEvent);
  1932. }
  1933. bool getResult(size32_t &ret,bool wait)
  1934. {
  1935. if (value==(size32_t)-1) {
  1936. if (parent) {
  1937. if (GetOverlappedResult(parent->file,&overlapped,&value,wait)==0) {
  1938. int err = GetLastError();
  1939. if (err==ERROR_IO_INCOMPLETE)
  1940. return false;
  1941. if (err!=ERROR_HANDLE_EOF) {
  1942. CriticalBlock block(parent->cs);
  1943. parent->results.zap(*this,true); // don't delete as array does not own
  1944. parent = NULL;
  1945. throw makeOsException(err,"CFileAsyncResult::getResult");
  1946. }
  1947. value = 0;
  1948. }
  1949. CriticalBlock block(parent->cs);
  1950. parent->results.zap(*this,true); // don't delete as array does not own
  1951. parent = NULL;
  1952. }
  1953. else
  1954. return false;
  1955. }
  1956. ret = value;
  1957. if (value<wrsize)
  1958. throw makeOsException(DISK_FULL_EXCEPTION_CODE,"CFileAsyncResult::getResult");
  1959. return true;
  1960. }
  1961. };
  1962. CFileAsyncIO::CFileAsyncIO(HANDLE handle, IFSHmode _sharemode)
  1963. {
  1964. assertex(handle != NULLFILE);
  1965. throwOnError = false;
  1966. file = handle;
  1967. sharemode = _sharemode;
  1968. }
  1969. void CFileAsyncIO::close()
  1970. {
  1971. flush();
  1972. // wait for all outstanding results
  1973. if (file != NULLFILE)
  1974. {
  1975. if (!CloseHandle(file))
  1976. throw makeOsException(GetLastError(),"CFileAsyncIO::close");
  1977. }
  1978. file = NULLFILE;
  1979. }
  1980. offset_t CFileAsyncIO::size()
  1981. {
  1982. LARGE_INTEGER pos;
  1983. pos.LowPart = GetFileSize(file, (unsigned long *)&pos.HighPart);
  1984. if (pos.LowPart==-1) {
  1985. DWORD err = GetLastError();
  1986. if (err!=0)
  1987. throw makeOsException(GetLastError(),"CFileAsyncIO::size");
  1988. }
  1989. return pos.QuadPart;
  1990. }
  1991. size32_t CFileAsyncIO::read(offset_t pos, size32_t len, void * data)
  1992. {
  1993. Owned<IFileAsyncResult> res = readAsync(pos,len,data);
  1994. size32_t ret;
  1995. res->getResult(ret,true);
  1996. return ret;
  1997. }
  1998. size32_t CFileAsyncIO::write(offset_t pos, size32_t len, const void * data)
  1999. {
  2000. Owned<IFileAsyncResult> res = writeAsync(pos,len,data);
  2001. size32_t ret;
  2002. res->getResult(ret,true);
  2003. return ret;
  2004. }
  2005. void CFileAsyncIO::setSize(offset_t size)
  2006. {
  2007. LARGE_INTEGER tempPos;
  2008. tempPos.QuadPart = size;
  2009. tempPos.LowPart = SetFilePointer(file, tempPos.LowPart, &tempPos.HighPart, FILE_BEGIN);
  2010. if (!SetEndOfFile(file))
  2011. throw makeOsException(GetLastError(), "CFileIO::setSize");
  2012. }
  2013. IFileAsyncResult *CFileAsyncIO::readAsync(offset_t pos, size32_t len, void * data)
  2014. {
  2015. CFileAsyncResult *res = new CFileAsyncResult(pos,0);
  2016. DWORD val;
  2017. if (ReadFile(file,data,len,&val,&res->overlapped) == 0) {
  2018. int err = GetLastError();
  2019. if (err == ERROR_HANDLE_EOF) { // bit weird
  2020. res->value = 0; // won't need to wait
  2021. }
  2022. else if (err == ERROR_IO_PENDING) {
  2023. CriticalBlock block(cs);
  2024. res->parent = this;
  2025. results.append(*res);
  2026. }
  2027. else
  2028. throw makeOsException(GetLastError(),"CFileIO::readAsync");
  2029. }
  2030. else {
  2031. res->value = val; // won't need to wait
  2032. }
  2033. return res;
  2034. }
  2035. IFileAsyncResult *CFileAsyncIO::writeAsync(offset_t pos, size32_t len, const void * data)
  2036. {
  2037. CFileAsyncResult *res = new CFileAsyncResult(pos,len);
  2038. DWORD val;
  2039. if (WriteFile(file,data,len,&val,&res->overlapped) == 0) {
  2040. int err = GetLastError();
  2041. if (err != ERROR_IO_PENDING)
  2042. throw makeOsException(GetLastError(),"CFileIO::writeAsync");
  2043. CriticalBlock block(cs);
  2044. res->parent = this;
  2045. results.append(*res);
  2046. }
  2047. else {
  2048. res->value = val; // wont need to wait
  2049. }
  2050. return res;
  2051. }
  2052. #else
  2053. //-- Unix implementation ----------------------------------------------------
  2054. class CFileAsyncResult: implements IFileAsyncResult, public CInterface
  2055. {
  2056. protected:
  2057. friend class CFileAsyncIO;
  2058. DWORD value;
  2059. DWORD wrsize;
  2060. aiocb cb;
  2061. public:
  2062. IMPLEMENT_IINTERFACE;
  2063. CFileAsyncResult(offset_t offset,size32_t _wrsize)
  2064. {
  2065. value = (size32_t)-1;
  2066. wrsize = _wrsize;
  2067. }
  2068. ~CFileAsyncResult()
  2069. {
  2070. size32_t dummy;
  2071. getResult(dummy,true);
  2072. }
  2073. bool getResult(size32_t &ret,bool wait)
  2074. {
  2075. if (value==(size32_t)-1) {
  2076. for (;;) {
  2077. int aio_errno = aio_error(&cb);
  2078. if (aio_errno==ECANCELED)
  2079. return false;
  2080. if (aio_errno != EINPROGRESS)
  2081. {
  2082. if (aio_errno)
  2083. throw makeErrnoException(aio_errno, "CFileAsyncResult::getResult");
  2084. value = aio_return(&cb);
  2085. if (value<wrsize)
  2086. throw makeOsException(DISK_FULL_EXCEPTION_CODE, "CFileAsyncResult::getResult");
  2087. break;
  2088. }
  2089. if (!wait)
  2090. return false;
  2091. for (;;) {
  2092. struct timespec timeout;
  2093. timeout.tv_sec = 60*60*24; // a long time
  2094. timeout.tv_nsec = 0;
  2095. aiocb * cb_list[1];
  2096. cb_list[0] = &cb;
  2097. int rc = aio_suspend(cb_list, 1, &timeout);
  2098. if ((rc == 0)||(errno != EAGAIN))
  2099. break;
  2100. if (rc==ECANCELED)
  2101. return false;
  2102. }
  2103. }
  2104. }
  2105. ret = value;
  2106. return true;
  2107. }
  2108. };
  2109. CFileAsyncIO::CFileAsyncIO(HANDLE handle, IFSHmode _sharemode)
  2110. {
  2111. assertex(handle != NULLFILE);
  2112. throwOnError = false;
  2113. file = handle;
  2114. sharemode = _sharemode;
  2115. }
  2116. void CFileAsyncIO::close()
  2117. {
  2118. if (file != NULLFILE)
  2119. {
  2120. aio_cancel(file,NULL);
  2121. if (_lclose(file) < 0)
  2122. throw makeErrnoException(errno, "CFileAsyncIO::close");
  2123. }
  2124. file=NULLFILE;
  2125. }
  2126. offset_t CFileAsyncIO::size()
  2127. {
  2128. CriticalBlock procedure(cs);
  2129. offset_t savedPos = _llseek(file,0,SEEK_CUR);
  2130. offset_t length = _llseek(file,0,SEEK_END);
  2131. _llseek(file, savedPos, SEEK_SET);
  2132. return length;
  2133. }
  2134. size32_t CFileAsyncIO::read(offset_t pos, size32_t len, void * data)
  2135. {
  2136. CriticalBlock procedure(cs);
  2137. _llseek(file,pos,SEEK_SET);
  2138. size32_t ret = _lread(file,data,len);
  2139. if (ret==(size32_t)-1)
  2140. throw makeErrnoException(errno, "CFileAsyncIO::read");
  2141. return ret;
  2142. }
  2143. size32_t CFileAsyncIO::write(offset_t pos, size32_t len, const void * data)
  2144. {
  2145. CriticalBlock procedure(cs);
  2146. _llseek(file,pos,SEEK_SET);
  2147. size32_t ret = _lwrite(file,data,len);
  2148. if (ret==(size32_t)-1)
  2149. throw makeErrnoException(errno, "CFileAsyncIO::write");
  2150. return ret;
  2151. }
  2152. void CFileAsyncIO::setSize(offset_t pos)
  2153. {
  2154. CriticalBlock procedure(cs);
  2155. if ((file != NULLFILE)&&(0 != ftruncate(file, pos)))
  2156. throw makeErrnoException(errno, "CFileIO::setSize");
  2157. }
  2158. IFileAsyncResult *CFileAsyncIO::readAsync(offset_t pos, size32_t len, void * data)
  2159. {
  2160. CFileAsyncResult *res = new CFileAsyncResult(pos,0);
  2161. bzero( &(res->cb), sizeof (struct aiocb));
  2162. res->cb.aio_fildes = file;
  2163. res->cb.aio_offset = pos;
  2164. res->cb.aio_buf = data;
  2165. res->cb.aio_nbytes = len;
  2166. res->cb.aio_sigevent.sigev_notify = SIGEV_NONE;
  2167. int retval = aio_read(&(res->cb));
  2168. if (retval==-1)
  2169. throw makeErrnoException(errno, "CFileAsyncIO::readAsync");
  2170. return res;
  2171. }
  2172. IFileAsyncResult *CFileAsyncIO::writeAsync(offset_t pos, size32_t len, const void * data)
  2173. {
  2174. CFileAsyncResult *res = new CFileAsyncResult(pos,len);
  2175. bzero( &(res->cb), sizeof (struct aiocb));
  2176. res->cb.aio_fildes = file;
  2177. res->cb.aio_offset = pos;
  2178. res->cb.aio_buf = (void*)data;
  2179. res->cb.aio_nbytes = len;
  2180. res->cb.aio_sigevent.sigev_signo = SIGUSR1;
  2181. res->cb.aio_sigevent.sigev_notify = SIGEV_NONE;
  2182. res->cb.aio_sigevent.sigev_value.sival_ptr = (void*)res;
  2183. int retval = aio_write(&(res->cb));
  2184. if (retval==-1)
  2185. throw makeErrnoException(errno, "CFileAsyncIO::writeAsync");
  2186. return res;
  2187. }
  2188. #endif
  2189. //---------------------------------------------------------------------------
  2190. CFileIOStream::CFileIOStream(IFileIO * _io)
  2191. {
  2192. io.set(_io);
  2193. curOffset = 0;
  2194. }
  2195. void CFileIOStream::flush()
  2196. {
  2197. }
  2198. size32_t CFileIOStream::read(size32_t len, void * data)
  2199. {
  2200. size32_t numRead = io->read(curOffset, len, data);
  2201. curOffset += numRead;
  2202. return numRead;
  2203. }
  2204. void CFileIOStream::seek(offset_t pos, IFSmode origin)
  2205. {
  2206. switch (origin)
  2207. {
  2208. case IFScurrent:
  2209. curOffset += pos;
  2210. break;
  2211. case IFSend:
  2212. curOffset = io->size() + pos;
  2213. break;
  2214. case IFSbegin:
  2215. curOffset = pos;
  2216. break;
  2217. }
  2218. }
  2219. offset_t CFileIOStream::size()
  2220. {
  2221. return io->size();
  2222. }
  2223. offset_t CFileIOStream::tell()
  2224. {
  2225. return curOffset;
  2226. }
  2227. size32_t CFileIOStream::write(size32_t len, const void * data)
  2228. {
  2229. size32_t numWritten = io->write(curOffset, len, data);
  2230. curOffset += numWritten;
  2231. return numWritten;
  2232. }
  2233. //---------------------------------------------------------------------------
  2234. class CBufferedFileIOStreamBase : public CBufferedIOStreamBase, implements IFileIOStream
  2235. {
  2236. protected:
  2237. virtual offset_t directSize() = 0;
  2238. offset_t curOffset;
  2239. public:
  2240. IMPLEMENT_IINTERFACE;
  2241. CBufferedFileIOStreamBase(unsigned bufSize) : CBufferedIOStreamBase(bufSize), curOffset(0) { }
  2242. virtual void flush() { doflush(); }
  2243. void seek(offset_t pos, IFSmode origin)
  2244. {
  2245. offset_t newOffset = 0;
  2246. switch (origin)
  2247. {
  2248. case IFScurrent:
  2249. newOffset = tell() + pos;
  2250. break;
  2251. case IFSend:
  2252. newOffset = size() + pos;
  2253. break;
  2254. case IFSbegin:
  2255. newOffset = pos;
  2256. break;
  2257. default:
  2258. throwUnexpected();
  2259. }
  2260. if (reading)
  2261. {
  2262. // slightly weird but curoffset is end of buffer when reading
  2263. if ((newOffset >= curOffset-numInBuffer) && (newOffset <= curOffset))
  2264. {
  2265. curBufferOffset = (size32_t)(newOffset - (curOffset-numInBuffer));
  2266. return;
  2267. }
  2268. }
  2269. else
  2270. {
  2271. if ((newOffset >= curOffset) && (newOffset <= curOffset + numInBuffer))
  2272. {
  2273. curBufferOffset = (size32_t)(newOffset - curOffset);
  2274. return;
  2275. }
  2276. flush();
  2277. }
  2278. curOffset = newOffset;
  2279. numInBuffer = 0;
  2280. curBufferOffset = 0;
  2281. }
  2282. offset_t size()
  2283. {
  2284. offset_t curSize = directSize();
  2285. if (!reading)
  2286. curSize = std::max(curSize, curOffset + numInBuffer);
  2287. return curSize;
  2288. }
  2289. offset_t tell()
  2290. {
  2291. if (reading)
  2292. return curOffset - numInBuffer + curBufferOffset;
  2293. return curOffset + curBufferOffset;
  2294. }
  2295. size32_t read(size32_t len, void * data)
  2296. {
  2297. return CBufferedIOStreamBase::doread(len, data);
  2298. }
  2299. size32_t write(size32_t len, const void * data)
  2300. {
  2301. if (reading)
  2302. curOffset -= bytesRemaining();
  2303. return CBufferedIOStreamBase::dowrite(len, data);
  2304. }
  2305. };
  2306. class CBufferedFileIOStream : public CBufferedFileIOStreamBase
  2307. {
  2308. public:
  2309. CBufferedFileIOStream(IFileIO * _io, unsigned _bufferSize) : CBufferedFileIOStreamBase(_bufferSize), io(_io)
  2310. {
  2311. buffer = new byte[_bufferSize];
  2312. }
  2313. ~CBufferedFileIOStream()
  2314. {
  2315. flush();
  2316. delete [] buffer;
  2317. }
  2318. protected:
  2319. virtual void doflush()
  2320. {
  2321. if (!reading && numInBuffer)
  2322. {
  2323. try {
  2324. io->write(curOffset, numInBuffer, buffer);
  2325. }
  2326. catch (IException *) {
  2327. // if we get exception, clear buffer so doen't reoccur on destructor as well
  2328. numInBuffer = 0;
  2329. curBufferOffset = 0;
  2330. throw;
  2331. }
  2332. curOffset += curBufferOffset;
  2333. numInBuffer = 0;
  2334. curBufferOffset = 0;
  2335. }
  2336. }
  2337. virtual bool fillBuffer()
  2338. {
  2339. reading = true;
  2340. numInBuffer = io->read(curOffset, bufferSize, buffer);
  2341. curOffset += numInBuffer;
  2342. curBufferOffset = 0;
  2343. return numInBuffer!=0;
  2344. }
  2345. virtual size32_t directRead(size32_t len, void * data)
  2346. {
  2347. size32_t sz = io->read(curOffset,len,data);
  2348. curOffset += sz;
  2349. return sz;
  2350. }
  2351. virtual size32_t directWrite(size32_t len, const void * data)
  2352. {
  2353. size32_t sz = io->write(curOffset,len,data);
  2354. curOffset += sz;
  2355. return sz;
  2356. }
  2357. virtual offset_t directSize() { return io->size(); }
  2358. protected:
  2359. IFileIOAttr io;
  2360. };
  2361. //---------------------------------------------------------------------------
  2362. class CBufferedAsyncIOStream: public CBufferedFileIOStreamBase
  2363. {
  2364. Linked<IFileAsyncIO> io;
  2365. byte * blk1;
  2366. byte * blk2;
  2367. IFileAsyncResult *readasyncres;
  2368. IFileAsyncResult *writeasyncres;
  2369. bool readeof;
  2370. public:
  2371. CBufferedAsyncIOStream(IFileAsyncIO * _io, size32_t _bufferSize)
  2372. : CBufferedFileIOStreamBase(_bufferSize/2), io(_io)
  2373. {
  2374. blk1 = new byte[bufferSize];
  2375. blk2 = new byte[bufferSize];
  2376. buffer = blk1;
  2377. readasyncres = NULL;
  2378. writeasyncres = NULL;
  2379. readeof = false;
  2380. minDirectSize = (size32_t)-1; // async always writes using buffer
  2381. }
  2382. ~CBufferedAsyncIOStream()
  2383. {
  2384. flush();
  2385. waitAsyncWrite();
  2386. waitAsyncRead();
  2387. delete [] blk1;
  2388. delete [] blk2;
  2389. }
  2390. void waitAsyncWrite()
  2391. {
  2392. if (writeasyncres) {
  2393. size32_t res;
  2394. writeasyncres->getResult(res,true);
  2395. writeasyncres->Release();
  2396. writeasyncres = NULL;
  2397. }
  2398. }
  2399. size32_t waitAsyncRead()
  2400. {
  2401. size32_t res = 0;
  2402. if (readasyncres) {
  2403. readasyncres->getResult(res,true);
  2404. readasyncres->Release();
  2405. readasyncres = NULL;
  2406. }
  2407. return res;
  2408. }
  2409. void primeAsyncRead(offset_t pos,size32_t size, void *dst)
  2410. {
  2411. assertex(!readasyncres);
  2412. readasyncres = io->readAsync(pos, size, dst);
  2413. }
  2414. void primeAsyncWrite(offset_t pos,size32_t size, const void *src)
  2415. {
  2416. assertex(!writeasyncres);
  2417. writeasyncres = io->writeAsync(pos, size, src);
  2418. }
  2419. // CBufferedFileIOStream overloads
  2420. virtual bool fillBuffer()
  2421. {
  2422. if (!reading) {
  2423. waitAsyncWrite();
  2424. reading = true;
  2425. }
  2426. if (readeof) {
  2427. numInBuffer = 0;
  2428. curBufferOffset = 0;
  2429. return false;
  2430. }
  2431. buffer=(buffer==blk1)?blk2:blk1;
  2432. if (readasyncres==NULL) // first time
  2433. primeAsyncRead(curOffset, bufferSize, buffer);
  2434. numInBuffer = waitAsyncRead();
  2435. curOffset += numInBuffer;
  2436. curBufferOffset = 0;
  2437. if (numInBuffer)
  2438. primeAsyncRead(curOffset, bufferSize, (buffer==blk1)?blk2:blk1);
  2439. else
  2440. readeof = true;
  2441. return !readeof;
  2442. }
  2443. virtual void doflush()
  2444. {
  2445. if (!reading && numInBuffer)
  2446. {
  2447. waitAsyncWrite();
  2448. primeAsyncWrite(curOffset, numInBuffer, buffer);
  2449. buffer=(buffer==blk1)?blk2:blk1;
  2450. curOffset += curBufferOffset;
  2451. numInBuffer = 0;
  2452. curBufferOffset = 0;
  2453. }
  2454. }
  2455. virtual size32_t directRead(size32_t len, void * data) { assertex(false); return 0; } // shouldn't get called
  2456. virtual size32_t directWrite(size32_t len, const void * data) { assertex(false); return 0; } // shouldn't get called
  2457. virtual offset_t directSize() { waitAsyncWrite(); return io->size(); }
  2458. };
  2459. //-- Helper routines --------------------------------------------------------
  2460. enum GblFlushEnum { FLUSH_INIT, FLUSH_DISALLOWED, FLUSH_ALLOWED };
  2461. static GblFlushEnum gbl_flush_allowed = FLUSH_INIT;
  2462. static CriticalSection flushsect;
  2463. static inline bool isPCFlushAllowed()
  2464. {
  2465. CriticalBlock block(flushsect);
  2466. if (gbl_flush_allowed == FLUSH_INIT)
  2467. {
  2468. Owned<IProperties> conf = createProperties(CONFIG_DIR PATHSEPSTR "environment.conf", true);
  2469. if (conf->getPropBool("allow_pgcache_flush", true))
  2470. gbl_flush_allowed = FLUSH_ALLOWED;
  2471. else
  2472. gbl_flush_allowed = FLUSH_DISALLOWED;
  2473. }
  2474. if (gbl_flush_allowed == FLUSH_ALLOWED)
  2475. return true;
  2476. return false;
  2477. }
  2478. static inline size32_t doread(IFileIOStream * stream,void *dst, size32_t size)
  2479. {
  2480. size32_t toread=size;
  2481. while (toread)
  2482. {
  2483. int read = stream->read(toread, dst);
  2484. if (!read)
  2485. return size-toread;
  2486. toread -= read;
  2487. dst = (char *) dst + read;
  2488. }
  2489. return size;
  2490. }
  2491. CIOStreamReadWriteSeq::CIOStreamReadWriteSeq(IFileIOStream * _stream, offset_t _offset, size32_t _size)
  2492. {
  2493. stream.set(_stream);
  2494. // stream->setThrowOnError(true);
  2495. size = _size;
  2496. offset = _offset; // assumption that stream at correct location already
  2497. }
  2498. void CIOStreamReadWriteSeq::put(const void *src)
  2499. {
  2500. stream->write(size, src);
  2501. }
  2502. void CIOStreamReadWriteSeq::putn(const void *src, unsigned n)
  2503. {
  2504. stream->write(size*n, src);
  2505. }
  2506. void CIOStreamReadWriteSeq::flush()
  2507. {
  2508. stream->flush();
  2509. }
  2510. offset_t CIOStreamReadWriteSeq::getPosition()
  2511. {
  2512. return stream->tell();
  2513. }
  2514. bool CIOStreamReadWriteSeq::get(void *dst)
  2515. {
  2516. return doread(stream,dst,size)==size;
  2517. }
  2518. unsigned CIOStreamReadWriteSeq::getn(void *dst, unsigned n)
  2519. {
  2520. return doread(stream,dst,size*n)/size;
  2521. }
  2522. void CIOStreamReadWriteSeq::reset()
  2523. {
  2524. stream->seek(offset, IFSbegin);
  2525. }
  2526. //-- Helper routines --------------------------------------------------------
  2527. size32_t read(IFileIO * in, offset_t pos, size32_t len, MemoryBuffer & buffer)
  2528. {
  2529. const size32_t checkLengthLimit = 0x1000;
  2530. if (len >= checkLengthLimit)
  2531. {
  2532. //Don't allocate a stupid amount of memory....
  2533. offset_t fileLength = in->size();
  2534. if (pos > fileLength)
  2535. pos = fileLength;
  2536. if ((len == (size32_t)-1) || (pos + len > fileLength))
  2537. len = (size32_t)(fileLength - pos);
  2538. }
  2539. void * data = buffer.reserve(len);
  2540. size32_t lenRead = in->read(pos, len, data);
  2541. if (lenRead != len)
  2542. buffer.rewrite(buffer.length() - (len - lenRead));
  2543. return lenRead;
  2544. }
  2545. void renameFile(const char *target, const char *source, bool overwritetarget)
  2546. {
  2547. OwnedIFile src = createIFile(source);
  2548. if (!src)
  2549. throw MakeStringException(-1, "renameFile: source '%s' not found", source);
  2550. if (!src->isFile())
  2551. throw MakeStringException(-1, "renameFile: source '%s' is not a valid file", source);
  2552. if (src->isReadOnly())
  2553. throw MakeStringException(-1, "renameFile: source '%s' is readonly", source);
  2554. OwnedIFile tgt = createIFile(target);
  2555. if (!tgt)
  2556. throw MakeStringException(-1, "renameFile: target path '%s' could not be created", target);
  2557. if (tgt->exists() && !overwritetarget)
  2558. throw MakeStringException(-1, "renameFile: target file already exists: '%s' will not overwrite", target);
  2559. src->rename(target);
  2560. }
  2561. void copyFile(const char *target, const char *source, size32_t buffersize, ICopyFileProgress *progress, CFflags copyFlags)
  2562. {
  2563. OwnedIFile src = createIFile(source);
  2564. if (!src)
  2565. throw MakeStringException(-1, "copyFile: source '%s' not found", source);
  2566. OwnedIFile tgt = createIFile(target);
  2567. if (!tgt)
  2568. throw MakeStringException(-1, "copyFile: target path '%s' could not be created", target);
  2569. copyFile(tgt, src, buffersize, progress, copyFlags);
  2570. }
  2571. void copyFile(IFile * target, IFile * source, size32_t buffersize, ICopyFileProgress *progress, CFflags copyFlags)
  2572. {
  2573. source->copyTo(target, buffersize, progress, false, copyFlags);
  2574. }
  2575. void doCopyFile(IFile * target, IFile * source, size32_t buffersize, ICopyFileProgress *progress, ICopyFileIntercept *copyintercept, bool usetmp, CFflags copyFlags)
  2576. {
  2577. if (!buffersize)
  2578. buffersize = DEFAULT_COPY_BLKSIZE;
  2579. #ifdef _WIN32
  2580. if (!usetmp) {
  2581. CFile *src = QUERYINTERFACE(source,CFile);
  2582. CFile *dst = QUERYINTERFACE(target,CFile);
  2583. if (src) {
  2584. if (dst) {
  2585. if (src->fastCopyFile(*dst, buffersize, progress))
  2586. return;
  2587. }
  2588. CWindowsRemoteFile *dst2 = QUERYINTERFACE(target,CWindowsRemoteFile);
  2589. if (dst2) {
  2590. if (dst2->fastCopyFileRev(*src, buffersize, progress))
  2591. return;
  2592. }
  2593. }
  2594. CWindowsRemoteFile *src2 = QUERYINTERFACE(source,CWindowsRemoteFile);
  2595. if (src2) {
  2596. if (dst) {
  2597. if (src2->fastCopyFile(*dst,buffersize, progress))
  2598. return;
  2599. }
  2600. CWindowsRemoteFile *dst2 = QUERYINTERFACE(target,CWindowsRemoteFile);
  2601. if (dst2) {
  2602. if (src2->fastCopyFile(*dst2, buffersize, progress))
  2603. return;
  2604. }
  2605. }
  2606. }
  2607. #endif
  2608. IFEflags srcFlags = IFEnone;
  2609. if (copyFlags & CFflush_read)
  2610. srcFlags = IFEnocache;
  2611. OwnedIFileIO sourceIO = source->open(IFOread, srcFlags);
  2612. if (!sourceIO)
  2613. throw MakeStringException(-1, "copyFile: source '%s' not found", source->queryFilename());
  2614. #ifdef __linux__
  2615. // this is not really needed in windows - if it is we will have to
  2616. // test the file extenstion - .exe, .bat
  2617. struct stat info;
  2618. if (stat(source->queryFilename(), &info) == 0) // cannot fail - exception would have been thrown above
  2619. target->setCreateFlags(info.st_mode&(S_IRUSR|S_IRGRP|S_IROTH|S_IWUSR|S_IWGRP|S_IWOTH|S_IXUSR|S_IXGRP|S_IXOTH));
  2620. #endif
  2621. Owned<IFileIO> targetIO;
  2622. Owned<IFile> tmpfile;
  2623. IFile *dest;
  2624. if (usetmp) {
  2625. StringBuffer tmpname;
  2626. makeTempCopyName(tmpname,target->queryFilename());
  2627. tmpfile.setown(createIFile(tmpname.str()));
  2628. dest = tmpfile;
  2629. }
  2630. else
  2631. dest = target;
  2632. IFEflags tgtFlags = IFEnone;
  2633. if (copyFlags & CFflush_write)
  2634. tgtFlags = IFEnocache;
  2635. targetIO.setown(dest->open(IFOcreate, tgtFlags));
  2636. if (!targetIO)
  2637. throw MakeStringException(-1, "copyFile: target path '%s' could not be created", dest->queryFilename());
  2638. MemoryAttr mb;
  2639. void * buffer = copyintercept?NULL:mb.allocate(buffersize);
  2640. offset_t offset = 0;
  2641. offset_t total = 0;
  2642. Owned<IException> exc;
  2643. try
  2644. {
  2645. if (progress)
  2646. total = sourceIO->size(); // only needed for progress
  2647. for (;;)
  2648. {
  2649. size32_t got;
  2650. if (copyintercept) {
  2651. got = (size32_t)copyintercept->copy(sourceIO,targetIO,offset,buffersize);
  2652. if (got == 0)
  2653. break;
  2654. }
  2655. else {
  2656. got = sourceIO->read(offset, buffersize, buffer);
  2657. if (got == 0)
  2658. break;
  2659. targetIO->write(offset, got, buffer);
  2660. }
  2661. offset += got;
  2662. if (progress && progress->onProgress(offset, total) != CFPcontinue)
  2663. break;
  2664. }
  2665. targetIO.clear();
  2666. if (usetmp) {
  2667. StringAttr tail(pathTail(target->queryFilename()));
  2668. target->remove();
  2669. dest->rename(tail);
  2670. }
  2671. }
  2672. catch (IException *e)
  2673. {
  2674. // try to delete partial copy
  2675. StringBuffer s;
  2676. s.append("copyFile target=").append(dest->queryFilename()).append(" source=").append(source->queryFilename()).appendf("; read/write failure (%d): ",e->errorCode());
  2677. exc.setown(MakeStringException(e->errorCode(), "%s", s.str()));
  2678. e->Release();
  2679. EXCLOG(exc, "doCopyFile");
  2680. }
  2681. if (exc.get()) {
  2682. try {
  2683. sourceIO.clear();
  2684. }
  2685. catch (IException *e) {
  2686. EXCLOG(e, "doCopyFile closing source");
  2687. e->Release();
  2688. }
  2689. try {
  2690. targetIO.clear();
  2691. }
  2692. catch (IException *e) {
  2693. EXCLOG(e, "doCopyFile closing dest");
  2694. e->Release();
  2695. }
  2696. try {
  2697. dest->remove();
  2698. }
  2699. catch (IException *e) {
  2700. StringBuffer s;
  2701. EXCLOG(e, s.clear().append("Removing partial copy file: ").append(dest->queryFilename()).str());
  2702. e->Release();
  2703. }
  2704. throw exc.getClear();
  2705. }
  2706. CDateTime createTime, modifiedTime;
  2707. if (source->getTime(&createTime, &modifiedTime, NULL))
  2708. target->setTime(&createTime, &modifiedTime, NULL);
  2709. }
  2710. void makeTempCopyName(StringBuffer &tmpname,const char *destname)
  2711. {
  2712. // simple for the moment (maybe used uid later)
  2713. tmpname.append(destname).append("__");
  2714. #ifdef _WIN32
  2715. genUUID(tmpname,true);
  2716. #else
  2717. genUUID(tmpname);
  2718. #endif
  2719. tmpname.append(".tmp");
  2720. }
  2721. //---------------------------------------------------------------------------
  2722. #ifndef _WIN32
  2723. /// This code is dangerous - pass it arrays smaller than _MAX_xxx and it will write off the end even if you KNOW your strings would fit
  2724. // Should NOT use strncpy here
  2725. void _splitpath(const char *path, char *drive, char *dir, char *fname, char *ext)
  2726. {
  2727. strncpy(dir, path, (_MAX_DIR-1));
  2728. char* last = strrchr(dir, '.');
  2729. if (last != NULL)
  2730. {
  2731. if (strrchr(last, PATHSEPCHAR)==NULL)
  2732. {
  2733. strncpy(ext, last, (_MAX_EXT-1));
  2734. *last = '\0';
  2735. }
  2736. else
  2737. {
  2738. *ext = '\0';
  2739. }
  2740. }
  2741. else
  2742. {
  2743. *ext = '\0';
  2744. }
  2745. last = strrchr(dir, PATHSEPCHAR);
  2746. if (last != NULL)
  2747. {
  2748. strncpy(fname, ++last, (_MAX_FNAME-1));
  2749. *last = '\0';
  2750. }
  2751. else
  2752. {
  2753. strncpy(fname, dir, (_MAX_FNAME-1));
  2754. *dir = '\0';
  2755. }
  2756. *drive = '\0';
  2757. }
  2758. #endif
  2759. void splitFilename(const char * filename, StringBuffer * drive, StringBuffer * path, StringBuffer * tail, StringBuffer * ext, bool longExt)
  2760. {
  2761. char tdrive[_MAX_DRIVE];
  2762. char tdir[_MAX_DIR];
  2763. char ttail[_MAX_FNAME];
  2764. char text[_MAX_EXT];
  2765. ::_splitpath(filename, tdrive, tdir, ttail, text);
  2766. char *longExtStart = longExt ? strchr(ttail, '.') : NULL;
  2767. if (drive)
  2768. drive->append(tdrive);
  2769. if (path)
  2770. path->append(tdir);
  2771. if (tail)
  2772. tail->append(longExtStart ? longExtStart-ttail : strlen(ttail), ttail);
  2773. if (ext)
  2774. {
  2775. if (longExtStart)
  2776. ext->append(longExtStart);
  2777. ext->append(text);
  2778. }
  2779. }
  2780. StringBuffer &createUNCFilename(const char * filename, StringBuffer &UNC, bool useHostNames)
  2781. {
  2782. char buf[255];
  2783. #ifdef _WIN32
  2784. char *dummy;
  2785. GetFullPathName(filename, sizeof(buf), buf, &dummy);
  2786. if (buf[1]==':')
  2787. {
  2788. // MORE - assumes it's a local drive not a mapped one
  2789. UNC.append("\\\\");
  2790. if (useHostNames)
  2791. UNC.append(GetCachedHostName());
  2792. else
  2793. queryHostIP().getIpText(UNC);
  2794. UNC.append("\\").append((char)tolower(buf[0])).append(getShareChar()).append(buf+2);
  2795. }
  2796. else
  2797. {
  2798. assertex(buf[0]=='\\' && buf[1]=='\\');
  2799. UNC.append(buf);
  2800. }
  2801. return UNC;
  2802. #else
  2803. if (filename[0]=='/' && filename[1]=='/')
  2804. UNC.append(filename);
  2805. else
  2806. {
  2807. UNC.append("//");
  2808. if (useHostNames)
  2809. UNC.append(GetCachedHostName());
  2810. else
  2811. queryHostIP().getIpText(UNC);
  2812. if (*filename != '/')
  2813. {
  2814. if (!GetCurrentDirectory(sizeof(buf), buf)) {
  2815. ERRLOG("createUNCFilename: Current directory path too big, bailing out");
  2816. throwUnexpected();
  2817. }
  2818. UNC.append(buf).append("/");
  2819. }
  2820. UNC.append(filename);
  2821. }
  2822. return UNC;
  2823. #endif
  2824. }
  2825. bool splitUNCFilename(const char * filename, StringBuffer * machine, StringBuffer * path, StringBuffer * tail, StringBuffer * ext)
  2826. {
  2827. if (!filename || !isPathSepChar(filename[0]) || !isPathSepChar(filename[1]))
  2828. return false;
  2829. const char * cur = filename+2;
  2830. while (*cur && !isPathSepChar(*cur))
  2831. cur++;
  2832. if (!*cur)
  2833. return false;
  2834. const char * startPath = cur;
  2835. const char * lastExt = NULL;
  2836. const char * startTail = NULL;
  2837. char next;
  2838. while ((next = *cur) != 0)
  2839. {
  2840. if (isPathSepChar(next))
  2841. {
  2842. lastExt = NULL;
  2843. startTail = cur+1;
  2844. }
  2845. else if (next == '.')
  2846. lastExt = cur;
  2847. cur++;
  2848. }
  2849. assertex(startTail);
  2850. if (machine)
  2851. machine->append(startPath-filename, filename);
  2852. if (path)
  2853. path->append(startTail - startPath, startPath);
  2854. if (lastExt)
  2855. {
  2856. if (tail)
  2857. tail->append(lastExt - startTail, startTail);
  2858. if (ext)
  2859. ext->append(lastExt);
  2860. }
  2861. else
  2862. {
  2863. if (tail)
  2864. tail->append(startTail);
  2865. }
  2866. return true;
  2867. }
  2868. /**
  2869. * Ensure the filename has desired extension.
  2870. * If it has no extension, add the desired extension, return true.
  2871. * If it has an extension different from the desiredExtension, return false.
  2872. * Otherwise, return true.
  2873. */
  2874. bool ensureFileExtension(StringBuffer& filename, const char* desiredExtension)
  2875. {
  2876. char drive[_MAX_DRIVE];
  2877. char dir[_MAX_DIR];
  2878. char fname[_MAX_FNAME];
  2879. char ext[_MAX_EXT];
  2880. _splitpath(filename.str(), drive, dir, fname, ext);
  2881. if (ext[0]==0)
  2882. {
  2883. filename.append(desiredExtension);
  2884. return true;
  2885. }
  2886. if (strcmp(ext,desiredExtension)!=0)
  2887. return false;
  2888. return true;
  2889. }
  2890. /* Get full file name. If noExtension is true, the extension (if any) will be trimmed */
  2891. StringBuffer& getFullFileName(StringBuffer& filename, bool noExtension)
  2892. {
  2893. char drive[_MAX_DRIVE];
  2894. char dir[_MAX_DIR];
  2895. char fname[_MAX_FNAME];
  2896. char ext[_MAX_EXT];
  2897. _splitpath(filename.str(), drive, dir, fname, ext);
  2898. filename.clear();
  2899. filename.append(drive).append(dir).append(fname);
  2900. if (!noExtension)
  2901. filename.append(ext);
  2902. return filename;
  2903. }
  2904. /* Get the file name only. If noExtension is true, the extension (if any) will be trimmed */
  2905. StringBuffer& getFileNameOnly(StringBuffer& filename, bool noExtension)
  2906. {
  2907. char drive[_MAX_DRIVE];
  2908. char dir[_MAX_DIR];
  2909. char fname[_MAX_FNAME];
  2910. char ext[_MAX_EXT];
  2911. _splitpath(filename.str(), drive, dir, fname, ext);
  2912. filename.clear();
  2913. filename.append(fname);
  2914. if (!noExtension)
  2915. filename.append(ext);
  2916. return filename;
  2917. }
  2918. //---------------------------------------------------------------------------
  2919. class CNullDirectoryIterator : implements IDirectoryIterator, public CInterface
  2920. {
  2921. public:
  2922. IMPLEMENT_IINTERFACE;
  2923. bool first()
  2924. {
  2925. return false;
  2926. }
  2927. bool next()
  2928. {
  2929. return false;
  2930. }
  2931. StringBuffer &getName(StringBuffer &buf)
  2932. {
  2933. return buf;
  2934. }
  2935. __int64 getFileSize()
  2936. {
  2937. return -1;
  2938. }
  2939. bool getModifiedTime(CDateTime &ret)
  2940. {
  2941. return false;
  2942. }
  2943. virtual bool isValid() { return false; }
  2944. virtual IFile & query() { return *(IFile *)NULL; }
  2945. virtual bool isDir() { return false; }
  2946. };
  2947. extern jlib_decl IDirectoryIterator *createNullDirectoryIterator()
  2948. {
  2949. return new CNullDirectoryIterator;
  2950. }
  2951. class CDirectoryIterator : implements IDirectoryIterator, public CInterface
  2952. {
  2953. public:
  2954. CDirectoryIterator(const char * _path, const char * _mask, bool _sub, bool _includedir)
  2955. {
  2956. StringBuffer tmp;
  2957. if (!_path || !*_path)
  2958. _path = "." PATHSEPSTR;
  2959. else if (_path[strlen(_path)-1] != PATHSEPCHAR)
  2960. _path = tmp.append(_path).append(PATHSEPCHAR);
  2961. path.set(_path);
  2962. mask.set(_mask);
  2963. sub = _sub;
  2964. includedir = _includedir;
  2965. subidx = 0;
  2966. curisdir = false;
  2967. }
  2968. IMPLEMENT_IINTERFACE
  2969. virtual bool first()=0;
  2970. virtual bool next()=0;
  2971. virtual bool isValid() { return cur != NULL; }
  2972. virtual IFile & query() { return *cur; }
  2973. virtual StringBuffer &getName(StringBuffer &buf)=0;
  2974. virtual bool isDir() { return curisdir; }
  2975. protected:
  2976. Owned<IFile> cur;
  2977. bool curisdir;
  2978. StringAttr path;
  2979. StringAttr mask;
  2980. bool includedir;
  2981. bool sub; // TBD
  2982. StringAttrArray subpaths;
  2983. unsigned subidx; // -1 to index subpaths
  2984. };
  2985. #ifdef _WIN32
  2986. class CWindowsDirectoryIterator : public CDirectoryIterator
  2987. {
  2988. WIN32_FIND_DATA info;
  2989. HANDLE handle;
  2990. bool setCurrent()
  2991. {
  2992. if (strcmp(info.cFileName, ".") == 0 || strcmp(info.cFileName, "..") == 0)
  2993. return false;
  2994. bool match = (!mask.length() || WildMatch(info.cFileName, mask, true));
  2995. curisdir = (info.dwFileAttributes&FILE_ATTRIBUTE_DIRECTORY)!=0;
  2996. if (!match&&!curisdir)
  2997. return false;
  2998. StringBuffer f(path);
  2999. if (subidx)
  3000. f.append(subpaths.item(subidx-1).text).append('\\');
  3001. f.append(info.cFileName);
  3002. if (curisdir) {
  3003. if (sub) {
  3004. const char *s = f.str()+path.length();
  3005. unsigned i;
  3006. for (i=0;i<subpaths.ordinality();i++)
  3007. if (stricmp(subpaths.item(i).text,s)==0)
  3008. break;
  3009. if (i==subpaths.ordinality())
  3010. subpaths.append(*new StringAttrItem(s));
  3011. }
  3012. if (!includedir)
  3013. return false;
  3014. }
  3015. if (!match)
  3016. return false;
  3017. cur.setown(createIFile(f.str()));
  3018. return true;
  3019. }
  3020. bool open()
  3021. {
  3022. close();
  3023. while (subidx<=subpaths.ordinality()) {
  3024. StringBuffer location(path);
  3025. if (subidx)
  3026. location.append(subpaths.item(subidx-1).text).append('\\');
  3027. location.append("*");
  3028. handle = FindFirstFile(location.str(), &info);
  3029. if (handle != INVALID_HANDLE_VALUE)
  3030. return true;
  3031. subidx++;
  3032. }
  3033. return false;
  3034. }
  3035. void close()
  3036. {
  3037. cur.clear();
  3038. if (handle != INVALID_HANDLE_VALUE) {
  3039. FindClose(handle);
  3040. handle = INVALID_HANDLE_VALUE;
  3041. }
  3042. };
  3043. public:
  3044. CWindowsDirectoryIterator(const char * _path, const char * _mask, bool _sub, bool _includedir)
  3045. : CDirectoryIterator(_path,_mask,_sub,_includedir)
  3046. {
  3047. handle = INVALID_HANDLE_VALUE;
  3048. }
  3049. ~CWindowsDirectoryIterator()
  3050. {
  3051. close();
  3052. }
  3053. bool first()
  3054. {
  3055. subpaths.kill();
  3056. subidx = 0;
  3057. if (!open())
  3058. return false;
  3059. if (setCurrent())
  3060. return true;
  3061. return next();
  3062. }
  3063. bool next()
  3064. {
  3065. for (;;) {
  3066. for (;;) {
  3067. if (!FindNextFile(handle, &info))
  3068. break;
  3069. if (setCurrent())
  3070. return true;
  3071. }
  3072. subidx++;
  3073. if (!open())
  3074. break;
  3075. if (setCurrent())
  3076. return true;
  3077. }
  3078. return false;
  3079. }
  3080. StringBuffer &getName(StringBuffer &buf)
  3081. {
  3082. if (subidx)
  3083. buf.append(subpaths.item(subidx-1).text).append('\\');
  3084. return buf.append(info.cFileName);
  3085. }
  3086. __int64 getFileSize()
  3087. {
  3088. if (curisdir)
  3089. return -1;
  3090. LARGE_INTEGER x;
  3091. x.LowPart = info.nFileSizeLow;
  3092. x.HighPart = info.nFileSizeHigh;
  3093. return x.QuadPart;
  3094. }
  3095. bool getModifiedTime(CDateTime &ret)
  3096. {
  3097. FILETIMEtoIDateTime(&ret, info.ftLastWriteTime);
  3098. return true;
  3099. }
  3100. };
  3101. IDirectoryIterator * createDirectoryIterator(const char * path, const char * mask)
  3102. {
  3103. if (mask&&!*mask) // only NULL is wild
  3104. return new CNullDirectoryIterator;
  3105. if (!path || !*path) // cur directory so no point in checking for remote etc.
  3106. return new CWindowsDirectoryIterator(path, mask,false,true);
  3107. OwnedIFile iFile = createIFile(path);
  3108. if (!iFile||(iFile->isDirectory()!=foundYes))
  3109. return new CNullDirectoryIterator;
  3110. return iFile->directoryFiles(mask, false, true);
  3111. }
  3112. IDirectoryIterator *CFile::directoryFiles(const char *mask,bool sub,bool includedirs)
  3113. {
  3114. if ((mask&&!*mask)|| // only NULL is wild
  3115. (isDirectory()!=foundYes))
  3116. return new CNullDirectoryIterator;
  3117. return new CWindowsDirectoryIterator(filename, mask,sub,includedirs);
  3118. }
  3119. bool CFile::getInfo(bool &isdir,offset_t &size,CDateTime &modtime)
  3120. {
  3121. WIN32_FILE_ATTRIBUTE_DATA info;
  3122. if (GetFileAttributesEx(filename, GetFileExInfoStandard, &info) != 0) {
  3123. LARGE_INTEGER x;
  3124. x.LowPart = info.nFileSizeLow;
  3125. x.HighPart = info.nFileSizeHigh;
  3126. size = (offset_t)x.QuadPart;
  3127. isdir = (info.dwFileAttributes != (DWORD)-1)&&(info.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY);
  3128. FILETIMEtoIDateTime(&modtime, info.ftLastWriteTime);
  3129. return true;
  3130. }
  3131. size = 0;
  3132. modtime.clear();
  3133. isdir = false;
  3134. return false;
  3135. }
  3136. #else
  3137. class CLinuxDirectoryIterator : public CDirectoryIterator
  3138. {
  3139. StringAttr tail;
  3140. DIR * handle;
  3141. struct stat st;
  3142. bool gotst;
  3143. CriticalSection sect;
  3144. bool loadst()
  3145. {
  3146. if (!gotst&&cur)
  3147. gotst = (stat(cur->queryFilename(), &st) == 0);
  3148. return gotst;
  3149. }
  3150. public:
  3151. CLinuxDirectoryIterator(const char * _path, const char * _mask, bool _sub,bool _includedir)
  3152. : CDirectoryIterator(_path,_mask,_sub,_includedir)
  3153. {
  3154. handle = NULL;
  3155. gotst = false;
  3156. }
  3157. ~CLinuxDirectoryIterator()
  3158. {
  3159. close();
  3160. }
  3161. bool open()
  3162. {
  3163. close();
  3164. while (subidx<=subpaths.ordinality()) {
  3165. StringBuffer location(path);
  3166. if (subidx)
  3167. location.append(subpaths.item(subidx-1).text);
  3168. // not sure if should remove trailing '/'
  3169. handle = ::opendir(location.str());
  3170. // better error handling here?
  3171. if (handle)
  3172. return true;
  3173. subidx++;
  3174. }
  3175. return false;
  3176. }
  3177. void close()
  3178. {
  3179. cur.clear();
  3180. if (handle) {
  3181. closedir(handle);
  3182. handle = NULL;
  3183. }
  3184. }
  3185. bool first()
  3186. {
  3187. subpaths.kill();
  3188. subidx = 0;
  3189. if (open())
  3190. return next();
  3191. return false;
  3192. }
  3193. bool next()
  3194. {
  3195. for (;;) {
  3196. struct dirent *entry;
  3197. for (;;) {
  3198. gotst = false;
  3199. CriticalBlock b(sect);
  3200. entry = readdir(handle);
  3201. // need better checking here?
  3202. if (!entry)
  3203. break;
  3204. if (strcmp(entry->d_name, ".") == 0 || strcmp(entry->d_name, "..") == 0)
  3205. continue;
  3206. bool match = (!mask.length() || WildMatch(entry->d_name, mask, false));
  3207. curisdir = (entry->d_type==DT_DIR);
  3208. bool islnk = (entry->d_type==DT_LNK);
  3209. bool isunknown = (entry->d_type==DT_UNKNOWN); // to work around xfs bug
  3210. if (match||curisdir||islnk||isunknown) {
  3211. StringBuffer f(path);
  3212. if (subidx)
  3213. f.append(subpaths.item(subidx-1).text).append('/');
  3214. f.append(entry->d_name);
  3215. if (islnk||isunknown) {
  3216. struct stat info;
  3217. if (stat(f.str(), &info) == 0) // will follow link
  3218. curisdir = S_ISDIR(info.st_mode);
  3219. else
  3220. curisdir = false;
  3221. }
  3222. if (curisdir) {
  3223. if (sub) {
  3224. const char *s = f.str()+path.length();
  3225. unsigned i;
  3226. for (i=0;i<subpaths.ordinality();i++)
  3227. if (strcmp(subpaths.item(i).text,s)==0)
  3228. break;
  3229. if (i==subpaths.ordinality())
  3230. subpaths.append(*new StringAttrItem(s));
  3231. }
  3232. if (!includedir)
  3233. match = false;
  3234. }
  3235. if (match) {
  3236. tail.set(entry->d_name);
  3237. cur.setown(createIFile(f.str()));
  3238. return true;
  3239. }
  3240. }
  3241. }
  3242. subidx++;
  3243. if (!open())
  3244. break;
  3245. }
  3246. return false;
  3247. }
  3248. StringBuffer &getName(StringBuffer &buf)
  3249. {
  3250. if (subidx)
  3251. buf.append(subpaths.item(subidx-1).text).append('/');
  3252. return buf.append(tail);
  3253. }
  3254. __int64 getFileSize()
  3255. {
  3256. if (curisdir)
  3257. return -1;
  3258. if (!loadst())
  3259. return -1;
  3260. return st.st_size ;
  3261. }
  3262. bool getModifiedTime(CDateTime &ret)
  3263. {
  3264. if (!loadst())
  3265. return false;
  3266. timetToIDateTime(&ret,st.st_mtime);
  3267. return true;
  3268. }
  3269. };
  3270. IDirectoryIterator * createDirectoryIterator(const char * path, const char * mask)
  3271. {
  3272. if (mask&&!*mask) // only NULL is wild
  3273. return new CNullDirectoryIterator;
  3274. if (!path || !*path) // no point in checking for remote etc.
  3275. return new CLinuxDirectoryIterator(path, mask,false,true);
  3276. OwnedIFile iFile = createIFile(path);
  3277. if (!iFile||(iFile->isDirectory()!=foundYes))
  3278. return new CNullDirectoryIterator;
  3279. return iFile->directoryFiles(mask, false, true);
  3280. }
  3281. IDirectoryIterator *CFile::directoryFiles(const char *mask,bool sub,bool includedirs)
  3282. {
  3283. if ((mask&&!*mask)|| // only NULL is wild
  3284. (isDirectory()!=foundYes))
  3285. return new CNullDirectoryIterator;
  3286. return new CLinuxDirectoryIterator(filename, mask,sub,includedirs);
  3287. }
  3288. bool CFile::getInfo(bool &isdir,offset_t &size,CDateTime &modtime)
  3289. {
  3290. struct stat info;
  3291. if (stat(filename, &info) == 0) {
  3292. size = (offset_t)info.st_size;
  3293. isdir = S_ISDIR(info.st_mode);
  3294. timetToIDateTime(&modtime, info.st_mtime);
  3295. return true;
  3296. }
  3297. size = 0;
  3298. modtime.clear();
  3299. isdir = false;
  3300. return false;
  3301. }
  3302. #endif
  3303. class CDirEntry: extends CInterface
  3304. {
  3305. public:
  3306. StringAttr name;
  3307. Owned<IFile> file;
  3308. __int64 size;
  3309. CDateTime modified;
  3310. byte flags; // IDDI*
  3311. bool isdir;
  3312. CDirEntry(IDirectoryIterator *iter)
  3313. {
  3314. StringBuffer n;
  3315. name.set(iter->getName(n).str());
  3316. size = iter->getFileSize();
  3317. iter->getModifiedTime(modified);
  3318. file.set(&iter->query());
  3319. flags = 0;
  3320. isdir = iter->isDir();
  3321. }
  3322. bool match(byte f)
  3323. {
  3324. return (flags==0)||((f&flags)!=0);
  3325. }
  3326. int compare(const CDirEntry *e)
  3327. {
  3328. return strcmp(name.get(),e->name.get());
  3329. }
  3330. int compareProp(const CDirEntry *e)
  3331. {
  3332. int c = compare(e);
  3333. if (c)
  3334. return c;
  3335. if (isdir!=e->isdir)
  3336. return isdir?-1:1;
  3337. if (size!=e->size)
  3338. return (size<e->size)?-1:1;
  3339. return modified.compare(e->modified,false);
  3340. }
  3341. };
  3342. class CDirectoryDifferenceIterator : public CIArrayOf<CDirEntry>, extends CInterface, implements IDirectoryDifferenceIterator
  3343. {
  3344. static int compare(CInterface * const *_a, CInterface * const *_b)
  3345. {
  3346. CDirEntry *a = *(CDirEntry **)_a;
  3347. CDirEntry *b = *(CDirEntry **)_b;
  3348. return a->compare(b);
  3349. }
  3350. unsigned idx;
  3351. byte mask;
  3352. public:
  3353. IMPLEMENT_IINTERFACE;
  3354. CDirectoryDifferenceIterator(IDirectoryIterator *iter, CDirectoryDifferenceIterator *cmp)
  3355. {
  3356. mask = IDDIstandard;
  3357. idx = 0;
  3358. ForEach(*iter)
  3359. append(*new CDirEntry(iter));
  3360. CIArrayOf<CDirEntry>::sort(compare);
  3361. if (cmp) { // assumes cmp sorted
  3362. unsigned i = 0;
  3363. unsigned ni = ordinality();
  3364. unsigned j = 0;
  3365. unsigned nj = cmp->ordinality();
  3366. for (;;) {
  3367. CDirEntry *a = NULL;
  3368. CDirEntry *b = NULL;
  3369. if (i>=ni) {
  3370. if (j>=nj)
  3371. break;
  3372. b = &cmp->item(j++);
  3373. }
  3374. else if (j>=nj) {
  3375. a = &item(i++);
  3376. }
  3377. else {
  3378. a = &item(i);
  3379. b = &cmp->item(j);
  3380. int c = a->compare(b);
  3381. if (c==0) {
  3382. if (a->compareProp(b)==0) {
  3383. a->flags = IDDIunchanged;
  3384. a = NULL;
  3385. b = NULL;
  3386. }
  3387. i++;
  3388. j++;
  3389. }
  3390. else if (c<0) {
  3391. b = NULL;
  3392. i++;
  3393. }
  3394. else {
  3395. a = NULL;
  3396. j++;
  3397. }
  3398. }
  3399. if (a) {
  3400. if (b)
  3401. a->flags|=IDDImodified;
  3402. else
  3403. a->flags|=IDDIadded;
  3404. }
  3405. else if (b) {
  3406. b->Link();
  3407. b->flags = IDDIdeleted;
  3408. add(*b,i);
  3409. i++;
  3410. ni++;
  3411. }
  3412. }
  3413. }
  3414. }
  3415. virtual bool first()
  3416. {
  3417. idx = 0;
  3418. while (idx<ordinality()) {
  3419. if (item(idx).match(mask))
  3420. return true;
  3421. idx++;
  3422. }
  3423. return false;
  3424. }
  3425. virtual bool next()
  3426. {
  3427. idx++;
  3428. while (idx<ordinality()) {
  3429. if (item(idx).match(mask))
  3430. return true;
  3431. idx++;
  3432. }
  3433. return false;
  3434. }
  3435. virtual bool isValid()
  3436. {
  3437. return (idx<ordinality());
  3438. }
  3439. virtual IFile & query()
  3440. {
  3441. if (isValid())
  3442. return *item(idx).file;
  3443. return *(IFile *) NULL;
  3444. }
  3445. virtual StringBuffer &getName(StringBuffer &buf)
  3446. {
  3447. if (isValid())
  3448. buf.append(item(idx).name);
  3449. return buf;
  3450. }
  3451. virtual bool isDir()
  3452. {
  3453. if (isValid())
  3454. return item(idx).isdir;
  3455. return false;
  3456. }
  3457. __int64 getFileSize()
  3458. {
  3459. if (isValid())
  3460. return item(idx).isdir?0:item(idx).size;
  3461. return 0;
  3462. }
  3463. virtual bool getModifiedTime(CDateTime &ret)
  3464. {
  3465. if (isValid()) {
  3466. ret.set(item(idx).modified);
  3467. }
  3468. return false;
  3469. }
  3470. virtual void setMask(unsigned _mask)
  3471. {
  3472. mask = (byte)_mask;
  3473. }
  3474. virtual unsigned getFlags()
  3475. {
  3476. if (isValid())
  3477. return item(idx).flags;
  3478. return 0;
  3479. }
  3480. unsigned numChanges()
  3481. {
  3482. unsigned ret=0;
  3483. ForEachItemIn(i,*this) {
  3484. byte f = item(i).flags;
  3485. if ((f!=0)&&(f!=IDDIunchanged))
  3486. ret++;
  3487. }
  3488. return ret;
  3489. }
  3490. };
  3491. IDirectoryDifferenceIterator *CFile::monitorDirectory(IDirectoryIterator *_prev, // in
  3492. const char *mask,
  3493. bool sub,
  3494. bool includedirs,
  3495. unsigned checkinterval,
  3496. unsigned timeout,
  3497. Semaphore *abortsem)
  3498. {
  3499. Linked<IDirectoryIterator> prev;
  3500. if (_prev)
  3501. prev.set(_prev);
  3502. else
  3503. prev.setown(directoryFiles(mask,sub,includedirs));
  3504. if (!prev)
  3505. return NULL;
  3506. Owned<CDirectoryDifferenceIterator> base = new CDirectoryDifferenceIterator(prev,NULL);
  3507. prev.clear(); // not needed now
  3508. unsigned start=msTick();
  3509. for (;;) {
  3510. if (abortsem) {
  3511. if (abortsem->wait(checkinterval))
  3512. break;
  3513. }
  3514. else
  3515. Sleep(checkinterval);
  3516. Owned<IDirectoryIterator> current = directoryFiles(mask,sub,includedirs);
  3517. if (!current)
  3518. break;
  3519. Owned<CDirectoryDifferenceIterator> cmp = new CDirectoryDifferenceIterator(current,base);
  3520. current.clear();
  3521. if (cmp->numChanges())
  3522. return cmp.getClear();
  3523. if (msTick()-start>timeout)
  3524. break;
  3525. }
  3526. return NULL; // timed out
  3527. }
  3528. //---------------------------------------------------------------------------
  3529. class FixedPasswordProvider : implements IPasswordProvider, public CInterface
  3530. {
  3531. public:
  3532. FixedPasswordProvider(const char * _username, const char * _password) { username.set(_username); password.set(_password); }
  3533. IMPLEMENT_IINTERFACE;
  3534. virtual bool getPassword(const IpAddress & ip, StringBuffer & _username, StringBuffer & _password)
  3535. {
  3536. _username.append(username.get());
  3537. _password.append(password.get());
  3538. return true;
  3539. }
  3540. protected:
  3541. StringAttr username;
  3542. StringAttr password;
  3543. };
  3544. IPasswordProvider * queryPasswordProvider()
  3545. {
  3546. return passwordProvider;
  3547. }
  3548. void setPasswordProvider(IPasswordProvider * provider)
  3549. {
  3550. passwordProvider.set(provider);
  3551. }
  3552. void setDefaultUser(const char * username,const char *password)
  3553. {
  3554. Owned<IPasswordProvider> provider = new FixedPasswordProvider(username, password);
  3555. setPasswordProvider(provider);
  3556. }
  3557. //---------------------------------------------------------------------------
  3558. bool recursiveCreateDirectory(const char * path)
  3559. {
  3560. Owned<IFile> file = createIFile(path);
  3561. return file->createDirectory();
  3562. }
  3563. bool recursiveCreateDirectoryForFile(const char *fullFileName)
  3564. {
  3565. StringBuffer path;
  3566. splitFilename(fullFileName, &path, &path, NULL, NULL);
  3567. return recursiveCreateDirectory(path.str());
  3568. }
  3569. //---------------------------------------------------------------------------
  3570. size32_t DirectBufferI::read(offset_t pos, size32_t len, void * data)
  3571. {
  3572. if (pos + len > buffLen)
  3573. {
  3574. if (pos > buffLen)
  3575. pos = buffLen;
  3576. len = (size32_t)(buffLen - pos);
  3577. }
  3578. memcpy(data, buffer+pos, len);
  3579. return len;
  3580. }
  3581. size32_t DirectBufferI::write(offset_t pos, size32_t len, const void * data)
  3582. {
  3583. UNIMPLEMENTED;
  3584. }
  3585. size32_t DirectBufferIO::write(offset_t pos, size32_t len, const void * data)
  3586. {
  3587. if (pos + len > buffLen)
  3588. {
  3589. if (pos > buffLen)
  3590. pos = buffLen;
  3591. len = (size32_t)(buffLen - pos);
  3592. }
  3593. memcpy(buffer+pos, data, len);
  3594. return len;
  3595. }
  3596. //---------------------------------------------------------------------------
  3597. IFile * createIFile(const char * filename)
  3598. {
  3599. if (!filename)
  3600. return NULL;
  3601. IFile *ret = createContainedIFileByHook(filename);
  3602. if (ret)
  3603. return ret;
  3604. bool linremote=(memcmp(filename,"//",2)==0);
  3605. if (!linremote&&(memcmp(filename,"\\\\",2)!=0)) // see if remote looking
  3606. return new CFile(filename);
  3607. RemoteFilename rfn;
  3608. rfn.setRemotePath(filename);
  3609. if (rfn.isNull())
  3610. throw MakeStringException(-1, "CreateIFile cannot resolve %s", filename);
  3611. if (rfn.isLocal()) { // ignore dafilesrv request if local and standard port
  3612. StringBuffer tmplocal;
  3613. return new CFile(rfn.getLocalPath(tmplocal).str());
  3614. }
  3615. #ifdef _WIN32
  3616. StringBuffer tmplocal;
  3617. if (linremote||(rfn.queryEndpoint().port!=0)) {
  3618. ret = createIFileByHook(rfn); // use daliservix in preference
  3619. if (ret)
  3620. return ret;
  3621. while (*filename) { // no daliservix so swap '/' for '\' and hope for best
  3622. if (*filename=='/')
  3623. tmplocal.append('\\');
  3624. else
  3625. tmplocal.append(*filename);
  3626. filename++;
  3627. }
  3628. filename =tmplocal.str();
  3629. }
  3630. return new CWindowsRemoteFile(filename);
  3631. #else
  3632. #ifdef USE_SAMBA
  3633. if(strncmp(filename, "smb://", 6) == 0)
  3634. return new CSambaRemoteFile(filename);
  3635. if(memcmp(filename, "//", 2) == 0) {
  3636. StringBuffer smbfile("smb:");
  3637. smbfile.append(filename);
  3638. return new CSambaRemoteFile(smbfile.str());
  3639. }
  3640. if(memcmp(filename, "\\\\", 2) == 0) {
  3641. StringBuffer smbfile("smb:");
  3642. int i = 0;
  3643. while(filename[i]) {
  3644. if(filename[i] == '\\')
  3645. smbfile.append('/');
  3646. else
  3647. smbfile.append(filename[i]);
  3648. i++;
  3649. }
  3650. return new CSambaRemoteFile(smbfile.str());
  3651. }
  3652. #else
  3653. if (memcmp(filename,"smb://",6)==0) // don't support samba - try remote
  3654. return createIFile(filename+4);
  3655. ret = createIFileByHook(rfn);
  3656. if (!ret)
  3657. throw MakeStringException(-1, "CreateIFile::cannot attach to %s. (remote.so not linked?)", filename);
  3658. return ret;
  3659. #endif
  3660. return new CFile(filename);
  3661. #endif
  3662. }
  3663. IFileIOStream * createIOStream(IFileIO * file)
  3664. {
  3665. return new CFileIOStream(file);
  3666. }
  3667. IFileIO * createIORange(IFileIO * io, offset_t header, offset_t length)
  3668. {
  3669. return new CFileRangeIO(io, header, length);
  3670. }
  3671. IFileIOStream * createBufferedIOStream(IFileIO * io, unsigned bufsize)
  3672. {
  3673. if (bufsize == (unsigned)-1)
  3674. bufsize = DEFAULT_BUFFER_SIZE;
  3675. return new CBufferedFileIOStream(io, bufsize);
  3676. }
  3677. IFileIOStream * createBufferedAsyncIOStream(IFileAsyncIO * io, unsigned bufsize)
  3678. {
  3679. if (bufsize == (unsigned)-1)
  3680. bufsize = DEFAULT_BUFFER_SIZE*2;
  3681. return new CBufferedAsyncIOStream(io, bufsize);
  3682. }
  3683. IReadSeq *createReadSeq(IFileIOStream * stream, offset_t offset, size32_t size)
  3684. {
  3685. return new CIOStreamReadWriteSeq(stream, offset, size);
  3686. }
  3687. IWriteSeq *createWriteSeq(IFileIOStream * stream, size32_t size)
  3688. {
  3689. return new CIOStreamReadWriteSeq(stream, 0, size);
  3690. }
  3691. extern jlib_decl offset_t filesize(const char *name)
  3692. {
  3693. CFile f(name);
  3694. return f.size();
  3695. }
  3696. extern jlib_decl offset_t getFreeSpace(const char* name)
  3697. {
  3698. offset_t freeBytesToCaller;
  3699. #ifdef _WIN32
  3700. offset_t totalBytes;
  3701. offset_t freeBytes;
  3702. int fResult = GetDiskFreeSpaceEx (name,
  3703. (PULARGE_INTEGER)&freeBytesToCaller,
  3704. (PULARGE_INTEGER)&totalBytes,
  3705. (PULARGE_INTEGER)&freeBytes);
  3706. if (fResult == 0) // error
  3707. {
  3708. return 0;
  3709. }
  3710. #elif defined (__linux__) || defined (__APPLE__)
  3711. struct statfs buf;
  3712. int fResult = statfs(name, &buf);
  3713. if (fResult == -1) // error
  3714. {
  3715. return 0;
  3716. }
  3717. // from "man statfs the def of f_bavail and f_bsize...
  3718. // mult "free blocks avail to non-superuser" by "optimal transfer block size" to get the available size
  3719. freeBytesToCaller = (offset_t)buf.f_bavail * buf.f_bsize;
  3720. #else
  3721. UNIMPLEMENTED;
  3722. #endif
  3723. return freeBytesToCaller;
  3724. }
  3725. extern jlib_decl void createHardLink(const char* fileName, const char* existingFileName)
  3726. {
  3727. #ifdef _WIN32
  3728. // requirements...need to make sure that the directory already exists
  3729. // and the 2 directories need to be on the same drive
  3730. if (!CreateHardLink(fileName, existingFileName, NULL))
  3731. {
  3732. LPVOID lpMsgBuf;
  3733. FormatMessage(
  3734. FORMAT_MESSAGE_ALLOCATE_BUFFER |
  3735. FORMAT_MESSAGE_FROM_SYSTEM |
  3736. FORMAT_MESSAGE_IGNORE_INSERTS,
  3737. NULL,
  3738. GetLastError(),
  3739. MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT), // Default language
  3740. (LPTSTR) &lpMsgBuf,
  3741. 0,
  3742. NULL
  3743. );
  3744. StringBuffer err;
  3745. err.appendf("Failed to create log alias %s for %s: %s", fileName, existingFileName, lpMsgBuf);
  3746. LocalFree( lpMsgBuf );
  3747. throw MakeStringException(-1, "createHardLink:: %s.",err.str());
  3748. }
  3749. #else
  3750. if (link(existingFileName, fileName) != 0) // error
  3751. throw MakeStringException(-1, "Failed to create log alias %s for %s: error code = %d", fileName, existingFileName, errno);
  3752. #endif
  3753. }
  3754. #if 0
  3755. void testDirectory()
  3756. {
  3757. //setDefaultUser("administrator", "password");
  3758. Owned<IFile> dir = createIFile("\\test1\\gavin\\home\\base");
  3759. dir->createDirectory();
  3760. Owned<IFile> dir2 = createIFile("\\test1\\gavin\\home\\base2\\");
  3761. dir2->createDirectory();
  3762. dir2->createDirectory();
  3763. Owned<IFile> dir3 = createIFile("c:\\test1\\gavin\\home2\\base");
  3764. dir3->createDirectory();
  3765. Owned<IFile> dir4 = createIFile("\\\\ghalliday\\c$\\test1\\gavin\\home3\\base");
  3766. dir4->createDirectory();
  3767. Owned<IFile> dir5 = createIFile("\\\\192.168.1.3\\d$\\test2\\gavin\\home3\\base");
  3768. dir5->createDirectory();
  3769. StringBuffer x;
  3770. x.clear();
  3771. splitUNCFilename("\\\\127.0.0.1\\gavin\\abc.ext", &x, &x, &x, &x);
  3772. x.clear();
  3773. splitUNCFilename("\\\\127.0.0.1\\gavin\\abc.ext\\etc", &x, &x, &x, &x);
  3774. x.clear();
  3775. splitUNCFilename("\\\\127.0.0.1\\", &x, &x, &x, &x);
  3776. }
  3777. #endif
  3778. // -- RemoteFilename class (file location encapsulation)
  3779. //#define _TRACERFN
  3780. void RemoteFilename::badFilename(const char * filename)
  3781. {
  3782. throw MakeStringException(-1, "Badly formatted file entry %s", filename);
  3783. }
  3784. bool RemoteFilename::equals(const RemoteFilename & other) const
  3785. {
  3786. if (isNull())
  3787. return other.isNull();
  3788. if (other.isNull())
  3789. return false;
  3790. if (!ep.ipequals(other.ep)) // only use ip for compare
  3791. return false;
  3792. if (isUnixPath()) { // case sensitive
  3793. if (localhead.length()&&other.localhead.length()) {
  3794. if (strcmp(localhead.get(),other.localhead.get())!=0)
  3795. return false;
  3796. }
  3797. else if (sharehead.length()&&other.sharehead.length()) {
  3798. if (strcmp(sharehead.get(),other.sharehead.get())!=0)
  3799. return false;
  3800. }
  3801. else if (sharehead.length()!=other.sharehead.length())
  3802. return false;
  3803. if (strcmp(tailpath.get(),other.tailpath.get())!=0)
  3804. return false;
  3805. }
  3806. else {
  3807. if (localhead.length()&&other.localhead.length()) {
  3808. if (stricmp(localhead.get(),other.localhead.get())!=0)
  3809. return false;
  3810. }
  3811. else if (sharehead.length()&&other.sharehead.length()) {
  3812. if (stricmp(sharehead.get(),other.sharehead.get())!=0)
  3813. return false;
  3814. }
  3815. else if (sharehead.length()!=other.sharehead.length())
  3816. return false;
  3817. if (stricmp(tailpath.get(),other.tailpath.get())!=0)
  3818. return false;
  3819. }
  3820. return true;
  3821. }
  3822. bool RemoteFilename::isNull() const
  3823. {
  3824. return (ep.isNull()||(tailpath.length()==0));
  3825. }
  3826. void RemoteFilename::clear()
  3827. {
  3828. ep.set(NULL, 0);
  3829. localhead.clear();
  3830. sharehead.clear();
  3831. tailpath.clear();
  3832. }
  3833. void RemoteFilename::serialize(MemoryBuffer & out)
  3834. {
  3835. ep.serialize(out);
  3836. out.append(localhead);
  3837. out.append(sharehead);
  3838. out.append(tailpath);
  3839. }
  3840. void RemoteFilename::deserialize(MemoryBuffer & in)
  3841. {
  3842. ep.deserialize(in);
  3843. in.read(localhead);
  3844. in.read(sharehead);
  3845. in.read(tailpath);
  3846. }
  3847. StringBuffer & RemoteFilename::getTail(StringBuffer &name) const
  3848. {
  3849. split(NULL, NULL, &name, &name);
  3850. return name;
  3851. }
  3852. StringBuffer & RemoteFilename::getPath(StringBuffer & name) const
  3853. {
  3854. if (isLocal())
  3855. return getLocalPath(name);
  3856. else
  3857. return getRemotePath(name);
  3858. }
  3859. bool RemoteFilename::isUnixPath() const // bit arbitrary
  3860. {
  3861. if (tailpath.length()!=0) {
  3862. char c = tailpath[0];
  3863. if (c=='/')
  3864. return true;
  3865. if (c=='\\')
  3866. return false;
  3867. }
  3868. if (sharehead.length()!=0) {
  3869. char c = sharehead[0];
  3870. if (c=='/')
  3871. return true;
  3872. if (c=='\\')
  3873. return false;
  3874. }
  3875. if (localhead.length()!=0) {
  3876. const char *s=localhead;
  3877. if (*s=='/')
  3878. return true;
  3879. if ((s[1]==':')&&(s[2]=='/'))
  3880. return false;
  3881. // those are the expected cases, otherwise look for first separator (legacy)
  3882. while (*s) {
  3883. if (*s=='\\')
  3884. return false;
  3885. s++;
  3886. if (*s=='/')
  3887. return true;
  3888. }
  3889. }
  3890. #ifdef _WIN32 // assume local on starting machine
  3891. return false;
  3892. #else
  3893. return true;
  3894. #endif
  3895. }
  3896. char RemoteFilename::getPathSeparator() const
  3897. {
  3898. return isUnixPath()?'/':'\\';
  3899. }
  3900. StringBuffer & RemoteFilename::getLocalPath(StringBuffer & out) const
  3901. {
  3902. if (tailpath.length()!=0) {
  3903. out.append(localhead).append(tailpath);
  3904. }
  3905. #ifdef _TRACERFN
  3906. StringBuffer eps;
  3907. PROGLOG("getLocalPath '%s' '%s' '%s' -> '%s'",sharehead.get()?sharehead.get():"NULL",localhead.get()?localhead.get():"NULL",tailpath.get()?tailpath.get():"NULL",out.str());
  3908. #endif
  3909. return out;
  3910. }
  3911. StringBuffer & RemoteFilename::getRemotePath(StringBuffer & out) const
  3912. { // this creates a name that can be used by windows or linux
  3913. // note - no longer loses port
  3914. char c=getPathSeparator();
  3915. out.append(c).append(c);
  3916. ep.getUrlStr(out);
  3917. const char *fn;
  3918. StringBuffer loc;
  3919. if (sharehead.length())
  3920. fn = loc.append(sharehead).append(tailpath).str();
  3921. else // try and guess from just tail (may likely fail other than for windows)
  3922. fn=getLocalPath(loc).str();
  3923. if ((c=='\\')&&(fn[1]==':')) { // windows \\d$
  3924. out.append((char)tolower(c)).append(*fn).append(getShareChar());
  3925. fn+=2;
  3926. }
  3927. out.append(fn);
  3928. #ifdef _TRACERFN
  3929. StringBuffer eps;
  3930. PROGLOG("getRemotePath '%s' '%s' '%s' -> '%s'",sharehead.get()?sharehead.get():"NULL",localhead.get()?localhead.get():"NULL",tailpath.get()?tailpath.get():"NULL",out.str());
  3931. #endif
  3932. return out;
  3933. }
  3934. bool RemoteFilename::isLocal() const
  3935. {
  3936. if (ep.port&&(ep.port!=DAFILESRV_PORT && ep.port!=SECURE_DAFILESRV_PORT))
  3937. return false; // treat non-dafilesrv port as remote
  3938. return ep.isLocal() || ep.isNull();
  3939. }
  3940. void RemoteFilename::set(const RemoteFilename & other)
  3941. {
  3942. ep.set(other.ep);
  3943. localhead.set(other.localhead);
  3944. sharehead.set(other.sharehead);
  3945. tailpath.set(other.tailpath);
  3946. }
  3947. void RemoteFilename::setExtension(const char * newext)
  3948. {
  3949. const char * dot = NULL;
  3950. const char * start = tailpath;
  3951. const char * cur = start;
  3952. const char pathSep=getPathSeparator();
  3953. while (*cur)
  3954. {
  3955. // if it is a "." inside the path then skip it.
  3956. if (dot && (*cur == pathSep))
  3957. dot = NULL;
  3958. else if (*cur == '.')
  3959. dot = cur;
  3960. cur++;
  3961. }
  3962. StringBuffer newtail;
  3963. if (dot)
  3964. newtail.append(dot-start,start);
  3965. else
  3966. newtail.append(start);
  3967. if (newext)
  3968. {
  3969. if (newext[0] != '.')
  3970. newtail.append('.');
  3971. newtail.append(newext);
  3972. }
  3973. tailpath.set(newtail);
  3974. }
  3975. void RemoteFilename::setPath(const SocketEndpoint & _ep, const char * _filename)
  3976. {
  3977. const char * filename=_filename;
  3978. StringBuffer full;
  3979. ep.set(_ep);
  3980. localhead.clear();
  3981. sharehead.clear();
  3982. if (filename&&*filename) {
  3983. if (isSpecialPath(filename)) {
  3984. tailpath.set(filename);
  3985. return;
  3986. }
  3987. if (isLocal()&&!isAbsolutePath(filename)) {
  3988. char dir[_MAX_PATH];
  3989. if (!GetCurrentDirectory(sizeof(dir), dir)) {
  3990. ERRLOG("RemoteFilename::setPath: Current directory path too big, bailing out");
  3991. throwUnexpected();
  3992. }
  3993. if (*filename==PATHSEPCHAR) {
  3994. #ifdef _WIN32
  3995. if (*dir && (dir[1]==':'))
  3996. dir[2] = 0;
  3997. #endif
  3998. filename++;
  3999. }
  4000. full.append(dir);
  4001. if (full.charAt(full.length()-1) != PATHSEPCHAR)
  4002. full.append(PATHSEPCHAR);
  4003. full.append(filename);
  4004. filename = full.str();
  4005. }
  4006. bool isunix;
  4007. if (filename[0]=='/')
  4008. isunix=true;
  4009. else if (filename[0]=='\\')
  4010. isunix=false;
  4011. else if ((filename[1]==':')&&(filename[2]=='\\'))
  4012. isunix = false;
  4013. else // invalid call really as supposed to be full local path!
  4014. #ifdef _WIN32
  4015. isunix = false;
  4016. #else
  4017. isunix = true;
  4018. #endif
  4019. StringBuffer tmpxlat;
  4020. char sepchar = isunix?'/':'\\';
  4021. char altchar = isunix?'\\':'/';
  4022. // fix mixed separator path
  4023. if (strchr(filename,altchar)) {
  4024. while (*filename) {
  4025. if (*filename==altchar)
  4026. tmpxlat.append(sepchar);
  4027. else
  4028. tmpxlat.append(*filename);
  4029. filename++;
  4030. }
  4031. filename = tmpxlat.str();
  4032. }
  4033. if (isunix) {
  4034. if (filename[0]=='/') {
  4035. const char *tail;
  4036. tail=strchr(filename+1,'/');
  4037. if (tail) {
  4038. sharehead.set(filename,tail-filename); // we don't know share so guess same as leading dir
  4039. localhead.set(filename,tail-filename);
  4040. filename = tail;
  4041. }
  4042. }
  4043. }
  4044. else { // windows can guess anonymous share
  4045. StringBuffer sharestr;
  4046. if ((filename[1]==':')&&(filename[2]=='\\')) { // this should be always true if true full windows path
  4047. localhead.set(filename,2);
  4048. sharestr.append('\\').append((char)tolower(filename[0])).append(getShareChar());
  4049. filename += 2;
  4050. }
  4051. else if (filename[0]=='\\') {
  4052. const char *tail = strchr(filename+1,'\\');
  4053. if (tail) {
  4054. sharestr.append(tail-filename,filename);
  4055. localhead.set(filename,tail-filename);
  4056. filename = tail;
  4057. }
  4058. else {
  4059. localhead.set("c:");
  4060. sharestr.append("\\c").append(getShareChar());
  4061. }
  4062. }
  4063. sharehead.set(sharestr);
  4064. }
  4065. tailpath.set(filename);
  4066. }
  4067. else
  4068. tailpath.clear();
  4069. #ifdef _TRACERFN
  4070. StringBuffer eps;
  4071. PROGLOG("setPath (%s,%s) -> '%s' '%s' '%s'",ep.getUrlStr(eps).str(),_filename?_filename:"NULL",sharehead.get()?sharehead.get():"NULL",localhead.get()?localhead.get():"NULL",tailpath.get()?tailpath.get():"NULL");
  4072. #endif
  4073. }
  4074. void RemoteFilename::setRemotePath(const char * _url,const char *localpath)
  4075. {
  4076. // url should be full (share) path including ep
  4077. const char *url=_url;
  4078. // if ep not included then assume local (bit weird though)
  4079. char sep = url?url[0]:0;
  4080. if (isPathSepChar(sep)&&(sep==url[1])) {
  4081. url+=2;
  4082. const char *end = findPathSepChar(url);
  4083. if (end) {
  4084. StringBuffer eps(end-url,url);
  4085. ep.set(eps.str());
  4086. url = end;
  4087. }
  4088. else {
  4089. ep.set(NULL,0);
  4090. url--; // don't know what is going on!
  4091. }
  4092. if (isPathSepChar(*url)||!*url)
  4093. sep = *url;
  4094. }
  4095. else
  4096. ep.setLocalHost(0);
  4097. if (localpath&&*localpath)
  4098. setPath(ep,localpath);
  4099. else {
  4100. localhead.clear();
  4101. sharehead.clear();
  4102. tailpath.clear();
  4103. }
  4104. if (sep&&*url) {
  4105. // url should point to the share now
  4106. const char *tail=findPathSepChar(url+1);
  4107. if (tail) { // hopefully must be!
  4108. sharehead.set(url,tail-url);
  4109. url = tail;
  4110. }
  4111. if (localhead.length()==0) { // we don't know so guess
  4112. if (sharehead.length()!=0) {
  4113. const char *share=sharehead.get();
  4114. if (sep=='\\') {
  4115. StringBuffer locstr;
  4116. if (sharehead.length()>2) {
  4117. if (isShareChar(share[2])) {
  4118. locstr.append(share[1]).append(':');
  4119. share+=3;
  4120. }
  4121. else // we haven't a clue!
  4122. locstr.append("c:");
  4123. }
  4124. else if (sharehead[1]!='$') // data format
  4125. locstr.append("c:");
  4126. locstr.append(share);
  4127. localhead.set(locstr);
  4128. }
  4129. else { // we haven't a clue so assume same as share
  4130. localhead.set(sharehead);
  4131. }
  4132. }
  4133. }
  4134. }
  4135. if (tailpath.length()==0)
  4136. tailpath.set(url);
  4137. #ifdef _TRACERFN
  4138. StringBuffer eps;
  4139. PROGLOG("setRemotePath ('%s','%s') -> '%s' '%s' '%s'",_url,localpath?localpath:"NULL",sharehead.get()?sharehead.get():"NULL",localhead.get()?localhead.get():"NULL",tailpath.get()?tailpath.get():"NULL");
  4140. #endif
  4141. }
  4142. void RemoteFilename::setLocalPath(const char *filename)
  4143. {
  4144. SocketEndpoint localEP;
  4145. localEP.setLocalHost(0);
  4146. setPath(localEP, filename);
  4147. }
  4148. void RemoteFilename::split(StringBuffer * drive, StringBuffer * path, StringBuffer * tail, StringBuffer * ext) const
  4149. {
  4150. StringBuffer localname;
  4151. const char *s = getLocalPath(localname).str();
  4152. char sep;
  4153. if (isUnixPath())
  4154. sep = '/';
  4155. else {
  4156. sep = '\\';
  4157. if (s[1]==':') {
  4158. if (drive)
  4159. drive->append(2,s);
  4160. s+=2;
  4161. }
  4162. }
  4163. const char *t=NULL;
  4164. const char *e=s;
  4165. while (*e) {
  4166. if (*e==sep)
  4167. t = e+1;
  4168. e++;
  4169. }
  4170. if (t) {
  4171. if (path)
  4172. path->append(t-s, s);
  4173. s=t;
  4174. }
  4175. if (!tail&&!ext)
  4176. return;
  4177. t = NULL;
  4178. e=s;
  4179. while (*e) {
  4180. if (*e=='.') // NB final extension
  4181. t = e;
  4182. e++;
  4183. }
  4184. if (t) {
  4185. if (tail)
  4186. tail->append(t-s,s);
  4187. if (ext)
  4188. ext->append(e-t,t);
  4189. }
  4190. else if (tail)
  4191. tail->append(e-s,s);
  4192. }
  4193. void RemoteMultiFilename::append(const char *mpath,const char *defaultdir)
  4194. {
  4195. StringArray paths;
  4196. expand(mpath, paths);
  4197. StringBuffer fullpath;
  4198. StringBuffer deftmp;
  4199. if (paths.ordinality()&&(defaultdir==NULL)&&isAbsolutePath(paths.item(0))) {
  4200. splitDirTail(paths.item(0),deftmp);
  4201. defaultdir = deftmp.str();
  4202. }
  4203. ForEachItemIn(i,paths) {
  4204. const char *filename = paths.item(i);
  4205. RemoteFilename rfn;
  4206. if (isPathSepChar(*filename)&&isPathSepChar(filename[1])) // full URL
  4207. rfn.setRemotePath(filename);
  4208. else {
  4209. if (defaultdir&&!isAbsolutePath(filename))
  4210. filename = addPathSepChar(fullpath.clear().append(defaultdir)).append(filename).str();
  4211. rfn.setPath(ep,filename);
  4212. }
  4213. append(rfn);
  4214. }
  4215. }
  4216. void RemoteMultiFilename::append(const RemoteFilename &inrfn)
  4217. {
  4218. RemoteFilename rfn;
  4219. rfn.set(inrfn);
  4220. if (ep.isNull())
  4221. ep = rfn.queryEndpoint();
  4222. else if (rfn.queryIP().isNull()) {
  4223. rfn.setEp(ep);
  4224. }
  4225. else if (!rfn.queryIP().ipequals(ep)) {
  4226. StringBuffer path;
  4227. rfn.getRemotePath(path);
  4228. throw MakeStringException(-1, "Component file IP does not match: %s", path.str());
  4229. }
  4230. RemoteFilenameArray::append(rfn);
  4231. }
  4232. void RemoteMultiFilename::deserialize(MemoryBuffer & in)
  4233. {
  4234. clear();
  4235. ep.deserialize(in);
  4236. unsigned n;
  4237. in.read(n);
  4238. StringBuffer last;
  4239. StringBuffer filename;
  4240. RemoteFilename rfn;
  4241. for (unsigned i=0;i<n;i++) {
  4242. byte l;
  4243. in.read(l);
  4244. filename.clear();
  4245. if (l)
  4246. filename.append((size32_t)l,last.str());
  4247. StringAttr s;
  4248. in.read(s);
  4249. filename.append(s);
  4250. rfn.setPath(ep,filename.str());
  4251. RemoteFilenameArray::append(rfn);
  4252. last.swapWith(filename);
  4253. }
  4254. }
  4255. void RemoteMultiFilename::serialize(MemoryBuffer & out)
  4256. {
  4257. ep.serialize(out);
  4258. unsigned n=ordinality();
  4259. out.append(n);
  4260. // do simple compression
  4261. StringBuffer last;
  4262. StringBuffer filename;
  4263. for (unsigned i=0;i<n;i++) {
  4264. item(i).getLocalPath(filename.clear());
  4265. const char *s1 = last.str();
  4266. const char *s2 = filename.str();
  4267. byte l=0;
  4268. while ((l<255)&&*s1&&*s2&&(*s1==*s2)) {
  4269. s1++;
  4270. s2++;
  4271. l++;
  4272. }
  4273. out.append(l);
  4274. out.append(s2);
  4275. last.swapWith(filename);
  4276. }
  4277. }
  4278. bool RemoteMultiFilename::isWild(unsigned idx) const
  4279. {
  4280. if (idx==(unsigned)-1) {
  4281. ForEachItem(i)
  4282. if (isWild(i))
  4283. return true;
  4284. }
  4285. else {
  4286. StringBuffer s;
  4287. item(idx).getTail(s);
  4288. if (containsFileWildcard(s.str()))
  4289. return true;
  4290. }
  4291. return false;
  4292. }
  4293. void RemoteMultiFilename::expandWild()
  4294. {
  4295. bool anywild = false;
  4296. BoolArray iswild;
  4297. ForEachItem(i1) {
  4298. if (isWild(i1)) {
  4299. anywild = true;
  4300. iswild.append(true);
  4301. }
  4302. else
  4303. iswild.append(false);
  4304. }
  4305. if (!anywild)
  4306. return; // nothing to do
  4307. // first cache old values (bit long winded but want to preserve order)
  4308. RemoteFilenameArray tmpa;
  4309. Int64Array tmpsz;
  4310. ForEachItem(i2) {
  4311. RemoteFilename rfn(item(i2));
  4312. tmpa.append(rfn);
  4313. if (i2<sizescache.ordinality())
  4314. tmpsz.append(sizescache.item(i2));
  4315. else
  4316. tmpsz.append(-1);
  4317. }
  4318. RemoteFilenameArray::kill();
  4319. sizescache.kill();
  4320. ForEachItemIn(i3,tmpa) {
  4321. RemoteFilename rfn(tmpa.item(i3));
  4322. if (iswild.item(i3)) {
  4323. StringBuffer name;
  4324. rfn.getLocalPath(name);
  4325. const char *s=name.str();
  4326. const char *t=s;
  4327. for (;;) {
  4328. const char *sep=findPathSepChar(t);
  4329. if (!sep)
  4330. break;
  4331. t = sep+1;
  4332. }
  4333. StringAttr tail(t);
  4334. name.setLength(t-s);
  4335. rfn.setPath(rfn.queryEndpoint(),name);
  4336. Owned<IFile> dir = createIFile(rfn);
  4337. Owned<IDirectoryIterator> iter = dir->directoryFiles(tail.get());
  4338. ForEach(*iter) {
  4339. append(iter->query().queryFilename());
  4340. sizescache.append(iter->getFileSize());
  4341. }
  4342. }
  4343. else {
  4344. append(rfn);
  4345. sizescache.append(tmpsz.item(i3));
  4346. }
  4347. }
  4348. }
  4349. offset_t RemoteMultiFilename::getSize(unsigned i)
  4350. {
  4351. __int64 ret = (i<sizescache.ordinality())?sizescache.item(i):-1;
  4352. if (ret==-1) {
  4353. RemoteFilename rfn(item(i));
  4354. Owned<IFile> file = createIFile(rfn);
  4355. ret = file->size();
  4356. }
  4357. return (offset_t)ret;
  4358. }
  4359. void RemoteMultiFilename::setIp(const IpAddress & ip)
  4360. {
  4361. ep.ipset(ip);
  4362. ForEachItem(i)
  4363. element(i).setIp(ip);
  4364. }
  4365. void RemoteMultiFilename::setEp(const SocketEndpoint & _ep)
  4366. {
  4367. ep.set(_ep);
  4368. ForEachItem(i)
  4369. element(i).setEp(_ep);
  4370. }
  4371. void RemoteMultiFilename::setPort(unsigned short port)
  4372. {
  4373. ep.port = port;
  4374. ForEachItem(i)
  4375. element(i).setPort(port);
  4376. }
  4377. void RemoteMultiFilename::set(const RemoteMultiFilename & other)
  4378. {
  4379. clear();
  4380. ep.set(other.ep);
  4381. ForEachItemIn(i,other) {
  4382. append(other.item(i));
  4383. if (i<other.sizescache.ordinality())
  4384. sizescache.append(other.sizescache.item(i));
  4385. }
  4386. }
  4387. bool RemoteMultiFilename::equals(const RemoteMultiFilename & other)
  4388. {
  4389. if (!ep.equals(other.ep))
  4390. return false;
  4391. if (ordinality()!=other.ordinality())
  4392. return false;
  4393. ForEachItem(i)
  4394. if (!item(i).equals(other.item(i)))
  4395. return false;
  4396. return true;
  4397. }
  4398. void RemoteMultiFilename::expand(const char *mpath, StringArray &array)
  4399. {
  4400. StringBuffer path;
  4401. StringBuffer fullpath;
  4402. for (;;) {
  4403. while (isspace(*mpath))
  4404. mpath++;
  4405. if (!*mpath)
  4406. break;
  4407. bool inquote=false;
  4408. while (*mpath) {
  4409. if (*mpath=='"') {
  4410. mpath++;
  4411. if (inquote) {
  4412. if (*mpath!='"') {
  4413. inquote = false;
  4414. continue;
  4415. }
  4416. }
  4417. else {
  4418. inquote = true;
  4419. continue;
  4420. }
  4421. }
  4422. if ((*mpath==',')&&!inquote) {
  4423. mpath++;
  4424. break;
  4425. }
  4426. path.append(*mpath);
  4427. mpath++;
  4428. }
  4429. path.clip();
  4430. if (path.length()) {
  4431. array.append(path.str());
  4432. path.clear();
  4433. }
  4434. }
  4435. }
  4436. void RemoteMultiFilename::tostr(StringArray &array,StringBuffer &out)
  4437. {
  4438. ForEachItemIn(i,array) {
  4439. const char *s = array.item(i);
  4440. if (!s||!*s)
  4441. continue;
  4442. if (i!=0)
  4443. out.append(',');
  4444. bool needquote=false;
  4445. for (const char *e=s;*e;e++)
  4446. if (isspace(*e)||(*e==',')) {
  4447. needquote = true;
  4448. break;
  4449. }
  4450. if (needquote)
  4451. out.append('"');
  4452. out.append(s);
  4453. if (needquote)
  4454. out.append('"');
  4455. }
  4456. }
  4457. //===================================================================================================
  4458. static IArrayOf<IContainedFileHook> containedFileHooks;
  4459. static ReadWriteLock containedFileHookLock;
  4460. void addContainedFileHook(IContainedFileHook *hook)
  4461. {
  4462. if (hook)
  4463. {
  4464. hook->Link();
  4465. WriteLockBlock block(containedFileHookLock);
  4466. containedFileHooks.append(*hook);
  4467. }
  4468. }
  4469. void removeContainedFileHook(IContainedFileHook *hook)
  4470. {
  4471. WriteLockBlock block(containedFileHookLock);
  4472. containedFileHooks.zap(*hook);
  4473. }
  4474. static IFile *createContainedIFileByHook(const char *filename)
  4475. {
  4476. ReadLockBlock block(containedFileHookLock);
  4477. ForEachItemIn(i, containedFileHooks)
  4478. {
  4479. IFile * ret = containedFileHooks.item(i).createIFile(filename);
  4480. if (ret)
  4481. return ret;
  4482. }
  4483. return NULL;
  4484. }
  4485. static IArrayOf<IRemoteFileCreateHook> remoteFileHooks;
  4486. static ReadWriteLock remoteFileHookLock;
  4487. void addIFileCreateHook(IRemoteFileCreateHook *hook)
  4488. {
  4489. if (hook)
  4490. {
  4491. hook->Link();
  4492. WriteLockBlock block(remoteFileHookLock);
  4493. remoteFileHooks.append(*hook);
  4494. }
  4495. }
  4496. void removeIFileCreateHook(IRemoteFileCreateHook *hook)
  4497. {
  4498. WriteLockBlock block(remoteFileHookLock);
  4499. remoteFileHooks.zap(*hook);
  4500. }
  4501. static IFile *createIFileByHook(const RemoteFilename & filename)
  4502. {
  4503. ReadLockBlock block(remoteFileHookLock);
  4504. ForEachItemIn(i, remoteFileHooks)
  4505. {
  4506. IFile * ret = remoteFileHooks.item(i).createIFile(filename);
  4507. if (ret)
  4508. return ret;
  4509. }
  4510. return NULL;
  4511. }
  4512. IFile * createIFile(const RemoteFilename & filename)
  4513. {
  4514. IFile * ret = createIFileByHook(filename);
  4515. if (ret)
  4516. return ret;
  4517. StringBuffer name;
  4518. return createIFile(getLocalOrRemoteName(name,filename).str());
  4519. }
  4520. StringBuffer &makePathUniversal(const char *path, StringBuffer &out)
  4521. {
  4522. if (!path||!*path)
  4523. return out;
  4524. if (path[1]==':')
  4525. {
  4526. out.append('/').append(*path);
  4527. path+=2;
  4528. }
  4529. for (; *path; path++)
  4530. out.append(isPathSepChar(*path) ? '/' : *path);
  4531. return out;
  4532. }
  4533. StringBuffer &makeAbsolutePath(const char *relpath,StringBuffer &out, bool mustExist)
  4534. {
  4535. if (isPathSepChar(relpath[0])&&(relpath[0]==relpath[1]))
  4536. {
  4537. if (mustExist)
  4538. {
  4539. OwnedIFile iFile = createIFile(relpath);
  4540. if (!iFile->exists())
  4541. throw makeStringExceptionV(-1, "makeAbsolutePath: could not resolve absolute path for %s", relpath);
  4542. }
  4543. return out.append(relpath); // if remote then already should be absolute
  4544. }
  4545. #ifdef _WIN32
  4546. char rPath[MAX_PATH];
  4547. char *filepart;
  4548. if (!relpath || '\0' == *relpath)
  4549. relpath = ".";
  4550. DWORD res = GetFullPathName(relpath, sizeof(rPath), rPath, &filepart);
  4551. if (0 == res)
  4552. throw makeOsExceptionV(GetLastError(), "makeAbsolutePath: could not resolve absolute path for %s", relpath);
  4553. else if (mustExist)
  4554. {
  4555. OwnedIFile iFile = createIFile(rPath);
  4556. if (!iFile->exists())
  4557. throw makeStringExceptionV(-1, "makeAbsolutePath: could not resolve absolute path for %s", rPath);
  4558. }
  4559. out.append(rPath);
  4560. #else
  4561. char rPath[PATH_MAX];
  4562. if (mustExist)
  4563. {
  4564. if (!realpath(relpath, rPath))
  4565. throw makeErrnoExceptionV(errno, "makeAbsolutePath: could not resolve absolute path for %s", relpath);
  4566. out.append(rPath);
  4567. }
  4568. else
  4569. {
  4570. // no error, will attempt to resolve(realpath) as much of relpath as possible and append rest
  4571. if (strlen(relpath))
  4572. {
  4573. const char *end = relpath+strlen(relpath);
  4574. const char *path = relpath;
  4575. const char *tail = end;
  4576. StringBuffer head;
  4577. for (;;)
  4578. {
  4579. if (realpath(path,rPath))
  4580. {
  4581. out.append(rPath);
  4582. if (tail != end)
  4583. out.append(tail);
  4584. return removeTrailingPathSepChar(out);
  4585. }
  4586. // mark next tail
  4587. for (;;)
  4588. {
  4589. --tail;
  4590. if (tail == relpath)
  4591. break;
  4592. else if ('/' == *tail)
  4593. break;
  4594. }
  4595. if (tail == relpath)
  4596. break; // bail out and guess
  4597. head.clear().append(tail-relpath, relpath);
  4598. path = head.str();
  4599. }
  4600. }
  4601. if (isAbsolutePath(relpath))
  4602. out.append(relpath);
  4603. else
  4604. {
  4605. appendCurrentDirectory(out, true);
  4606. if (strlen(relpath))
  4607. addPathSepChar(out).append(relpath);
  4608. }
  4609. }
  4610. #endif
  4611. return removeTrailingPathSepChar(out);
  4612. }
  4613. StringBuffer &makeAbsolutePath(StringBuffer &relpath,bool mustExist)
  4614. {
  4615. StringBuffer out;
  4616. makeAbsolutePath(relpath.str(),out,mustExist);
  4617. relpath.swapWith(out);
  4618. return relpath;
  4619. }
  4620. StringBuffer &makeAbsolutePath(const char *relpath, const char *basedir, StringBuffer &out)
  4621. {
  4622. StringBuffer combined;
  4623. if (basedir && !isAbsolutePath(relpath))
  4624. relpath = combined.append(basedir).append(relpath);
  4625. return makeAbsolutePath(relpath, out);
  4626. }
  4627. const char *splitRelativePath(const char *full,const char *basedir,StringBuffer &reldir)
  4628. {
  4629. if (basedir&&*basedir) {
  4630. size32_t bl = strlen(basedir);
  4631. if (isPathSepChar(basedir[bl-1]))
  4632. bl--;
  4633. if ((memicmp(full,basedir,bl)==0)&&isPathSepChar(full[bl]))
  4634. full += bl+1;
  4635. }
  4636. const char *t = full;
  4637. for (;;) {
  4638. const char *n = findPathSepChar(t);
  4639. if (!n)
  4640. break;
  4641. t = n+1;
  4642. }
  4643. if (t!=full)
  4644. reldir.append(t-full,full);
  4645. return t;
  4646. }
  4647. const char *splitDirMultiTail(const char *multipath,StringBuffer &dir,StringBuffer &tail)
  4648. {
  4649. // the first directory is the significant one
  4650. // others only removed if same
  4651. dir.clear();
  4652. StringArray files;
  4653. RemoteMultiFilename::expand(multipath,files);
  4654. StringBuffer reldir;
  4655. ForEachItemIn(i,files) {
  4656. const char *s = files.item(i);
  4657. if (i==0) {
  4658. if (isAbsolutePath(s)) {
  4659. StringAttr tail(splitDirTail(s,dir));
  4660. if (dir.length())
  4661. files.replace(tail,i);
  4662. }
  4663. }
  4664. else if (dir.length()) {
  4665. s= splitRelativePath(s,dir.str(),reldir.clear());
  4666. if (reldir.length()) {
  4667. reldir.append(s);
  4668. files.replace(reldir.str(),i);
  4669. }
  4670. }
  4671. }
  4672. RemoteMultiFilename::tostr(files,tail);
  4673. return tail.str();
  4674. }
  4675. StringBuffer &mergeDirMultiTail(const char *dir,const char *tail, StringBuffer &multipath)
  4676. {
  4677. StringArray files;
  4678. RemoteMultiFilename::expand(tail,files);
  4679. StringBuffer reldir;
  4680. if (dir && *dir) {
  4681. ForEachItemIn(i,files) {
  4682. const char *s = files.item(i);
  4683. if (!isAbsolutePath(s)) {
  4684. reldir.clear().append(dir);
  4685. addPathSepChar(reldir).append(s);
  4686. files.replace(reldir.str(),i);
  4687. }
  4688. }
  4689. }
  4690. RemoteMultiFilename::tostr(files,multipath);
  4691. return multipath;
  4692. }
  4693. StringBuffer &removeRelativeMultiPath(const char *full,const char *reldir,StringBuffer &res)
  4694. {
  4695. StringArray files;
  4696. RemoteMultiFilename::expand(full,files);
  4697. StringBuffer tmp1;
  4698. StringBuffer tmp2;
  4699. StringBuffer dir;
  4700. ForEachItemIn(i,files) {
  4701. const char *s = files.item(i);
  4702. if (isAbsolutePath(s)) {
  4703. if (!dir.length())
  4704. splitDirTail(s,dir);
  4705. }
  4706. else if (dir.length()) {
  4707. tmp1.clear().append(dir);
  4708. addPathSepChar(tmp1).append(s);
  4709. s = tmp1.str();
  4710. }
  4711. s = splitRelativePath(s,reldir,tmp2.clear());
  4712. tmp2.append(s);
  4713. files.replace(tmp2.str(),i);
  4714. }
  4715. RemoteMultiFilename::tostr(files,res);
  4716. return res;
  4717. }
  4718. // removes basedir if matches, returns relative multipath
  4719. //===================================================================================================
  4720. ExtractedBlobInfo::ExtractedBlobInfo(const char * _filename, offset_t _length, offset_t _offset)
  4721. {
  4722. filename.set(_filename);
  4723. length = _length;
  4724. offset = _offset;
  4725. }
  4726. void ExtractedBlobInfo::serialize(MemoryBuffer & buffer)
  4727. {
  4728. ::serialize(buffer, filename.get());
  4729. buffer.append(length).append(offset);
  4730. }
  4731. void ExtractedBlobInfo::deserialize(MemoryBuffer & buffer)
  4732. {
  4733. ::deserialize(buffer, filename);
  4734. buffer.read(length).read(offset);
  4735. }
  4736. //----------------------------------------------------------------------------
  4737. #define DFTERR_InvalidSplitPrefixFormat 8091
  4738. #define DFTERR_InvalidSplitPrefixFormat_Text "Cannot process file %s using the splitprefix supplied"
  4739. static void * readLength(MemoryBuffer & buffer, IFileIOStream * in, size_t len, const char * filenameText)
  4740. {
  4741. void * ptr = buffer.clear().reserve(len);
  4742. if (in->read(len, ptr) != len)
  4743. throwError1(DFTERR_InvalidSplitPrefixFormat, filenameText);
  4744. return ptr;
  4745. }
  4746. void extractBlobElements(const char * prefix, const RemoteFilename &filename, ExtractedBlobArray & extracted)
  4747. {
  4748. StringBuffer filenameText;
  4749. filename.getPath(filenameText);
  4750. Owned<IFile> file = createIFile(filename);
  4751. Owned<IFileIO> inIO = file->open(IFOread);
  4752. if (!inIO)
  4753. throw MakeStringException(-1, "extractBlobElements: file '%s' not found", filenameText.str());
  4754. Owned<IFileIOStream> in = createIOStream(inIO);
  4755. MemoryBuffer buffer;
  4756. offset_t endOffset = in->size();
  4757. while (in->tell() != endOffset)
  4758. {
  4759. StringAttr blobFilename;
  4760. offset_t blobLength = (offset_t)-1;
  4761. const char * finger = prefix;
  4762. while (finger)
  4763. {
  4764. StringAttr command;
  4765. const char * comma = strchr(finger, ',');
  4766. if (comma)
  4767. {
  4768. command.set(finger, comma-finger);
  4769. finger = comma+1;
  4770. }
  4771. else
  4772. {
  4773. command.set(finger);
  4774. finger = NULL;
  4775. }
  4776. command.toUpperCase();
  4777. if (memcmp(command, "FILENAME", 8) == 0)
  4778. {
  4779. if (command[8] == ':')
  4780. {
  4781. unsigned maxLen = atoi(command+9);
  4782. const char * nameptr = (const char *)readLength(buffer, in, maxLen, filenameText.str());
  4783. blobFilename.set(nameptr, maxLen);
  4784. }
  4785. else
  4786. {
  4787. unsigned * lenptr = (unsigned *)readLength(buffer, in, sizeof(unsigned), filenameText.str());
  4788. #if __BYTE_ORDER != __LITTLE_ENDIAN
  4789. _rev(sizeof(*lenptr), lenptr);
  4790. #endif
  4791. unsigned filenamelen = *lenptr;
  4792. const char * nameptr = (const char *)readLength(buffer, in, filenamelen, filenameText.str());
  4793. blobFilename.set(nameptr, filenamelen);
  4794. }
  4795. }
  4796. else if ((memcmp(command, "FILESIZE", 8) == 0) || (command.length() == 2))
  4797. {
  4798. const char * format = command;
  4799. if (memcmp(format, "FILESIZE", 8) == 0)
  4800. {
  4801. if (format[8] == ':')
  4802. format = format+9;
  4803. else
  4804. format = "L4";
  4805. }
  4806. bool bigEndian;
  4807. char c = format[0];
  4808. if (c == 'B')
  4809. bigEndian = true;
  4810. else if (c == 'L')
  4811. bigEndian = false;
  4812. else
  4813. throwError1(DFTERR_InvalidSplitPrefixFormat, format);
  4814. c = format[1];
  4815. if ((c <= '0') || (c > '8'))
  4816. throwError1(DFTERR_InvalidSplitPrefixFormat, format);
  4817. unsigned length = (c - '0');
  4818. byte * lenptr = (byte *)readLength(buffer, in, length, filenameText.str());
  4819. if (!bigEndian)
  4820. _rev(length, lenptr);
  4821. blobLength = 0;
  4822. for (unsigned i=0; i<length; i++)
  4823. {
  4824. blobLength = (blobLength << 8) | lenptr[i];
  4825. }
  4826. }
  4827. else if (memcmp(command, "SKIP:", 5) == 0)
  4828. {
  4829. unsigned skipLen = atoi(command+5);
  4830. in->seek(in->tell()+skipLen, IFSbegin);
  4831. }
  4832. else if (memcmp(command, "SEQ:", 4) == 0)
  4833. {
  4834. unsigned skipLen = atoi(command+4);
  4835. in->seek(in->tell()+skipLen, IFSbegin);
  4836. }
  4837. else
  4838. throwError1(DFTERR_InvalidSplitPrefixFormat, command.get());
  4839. }
  4840. if ((blobLength == (offset_t)-1) || !blobFilename.get())
  4841. throwError1(DFTERR_InvalidSplitPrefixFormat, filenameText.str());
  4842. offset_t blobOffset = in->tell();
  4843. extracted.append(* new ExtractedBlobInfo(blobFilename, blobLength, blobOffset));
  4844. in->seek(blobOffset + blobLength, IFSbegin);
  4845. }
  4846. }
  4847. bool mountDrive(const char *drv,const RemoteFilename &rfn)
  4848. {
  4849. #ifdef _WIN32
  4850. return false;
  4851. #else
  4852. unmountDrive(drv);
  4853. localCreateDirectory(drv);
  4854. int ret;
  4855. for (unsigned vtry=0;vtry<2;vtry++) {
  4856. StringBuffer cmd;
  4857. cmd.append("mount ");
  4858. rfn.queryIP().getIpText(cmd);
  4859. cmd.append(':');
  4860. rfn.getLocalPath(cmd);
  4861. cmd.append(' ').append(drv).append(" -t nfs ");
  4862. if (vtry==0)
  4863. cmd.append("-o nfsvers=v3 "); // prefer v3
  4864. cmd.append("2> /dev/null");
  4865. ret = system(cmd.str());
  4866. if (ret==0)
  4867. break;
  4868. }
  4869. return (ret==0);
  4870. #endif
  4871. }
  4872. bool unmountDrive(const char *drv)
  4873. {
  4874. #ifdef _WIN32
  4875. return false;
  4876. #else
  4877. StringBuffer cmd;
  4878. cmd.append("umount ").append(drv).append(" 2> /dev/null");
  4879. int ret = system(cmd.str());
  4880. return (ret==0);
  4881. #endif
  4882. }
  4883. IFileIO *createUniqueFile(const char *dir, const char *prefix, const char *ext, StringBuffer &filename)
  4884. {
  4885. CDateTime dt;
  4886. dt.setNow();
  4887. unsigned t = (unsigned)dt.getSimple();
  4888. if (dir)
  4889. {
  4890. filename.append(dir);
  4891. addPathSepChar(filename);
  4892. }
  4893. if (prefix && *prefix)
  4894. filename.append(prefix);
  4895. else
  4896. filename.append("uniq");
  4897. if (!ext || !*ext)
  4898. ext = "tmp";
  4899. filename.appendf("_%" I64F "x.%x.%x.%s", (__int64)GetCurrentThreadId(), (unsigned)GetCurrentProcessId(), t, ext);
  4900. OwnedIFile iFile = createIFile(filename.str());
  4901. unsigned attempts = 5; // max attempts
  4902. for (;;)
  4903. {
  4904. if (!iFile->exists())
  4905. {
  4906. try { return iFile->openShared(IFOcreate, IFSHnone); } // NB: could be null if path not found
  4907. catch (IException *e)
  4908. {
  4909. EXCLOG(e, "createUniqueFile");
  4910. e->Release();
  4911. }
  4912. }
  4913. if (0 == --attempts)
  4914. break;
  4915. t += getRandom();
  4916. filename.clear().appendf("uniq_%" I64F "x.%x.%x.%s", (__int64)GetCurrentThreadId(), (unsigned)GetCurrentProcessId(), t, ext);
  4917. iFile.setown(createIFile(filename.str()));
  4918. }
  4919. return NULL;
  4920. }
  4921. unsigned sortDirectory( CIArrayOf<CDirectoryEntry> &sortedfiles,
  4922. IDirectoryIterator &iter,
  4923. SortDirectoryMode mode,
  4924. bool rev,
  4925. bool includedirs
  4926. )
  4927. {
  4928. sortedfiles.kill();
  4929. StringBuffer name;
  4930. ForEach(iter) {
  4931. if (!iter.isDir()||includedirs)
  4932. sortedfiles.append(*new CDirectoryEntry(iter));
  4933. }
  4934. if (mode!=SD_nosort) {
  4935. struct icmp: implements ICompare
  4936. {
  4937. SortDirectoryMode mode;
  4938. bool rev;
  4939. int docompare(const void *l,const void *r) const
  4940. {
  4941. int ret=0;
  4942. const CDirectoryEntry *dl = (const CDirectoryEntry *)l;
  4943. const CDirectoryEntry *dr = (const CDirectoryEntry *)r;
  4944. switch (mode) {
  4945. case SD_byname:
  4946. ret = strcmp(dl->name,dr->name);
  4947. break;
  4948. case SD_bynameNC:
  4949. ret = stricmp(dl->name,dr->name);
  4950. break;
  4951. case SD_bydate:
  4952. ret = dl->modifiedTime.compare(dr->modifiedTime);
  4953. break;
  4954. case SD_bysize:
  4955. ret = (dl->size>dr->size)?1:((dl->size<dr->size)?-1:0);
  4956. break;
  4957. }
  4958. if (rev)
  4959. ret = -ret;
  4960. return ret;
  4961. }
  4962. } cmp;
  4963. cmp.mode = mode;
  4964. cmp.rev = rev;
  4965. qsortvec((void **)sortedfiles.getArray(), sortedfiles.ordinality(), cmp);
  4966. }
  4967. return sortedfiles.ordinality();
  4968. }
  4969. class CReplicatedFile : implements IReplicatedFile, public CInterface
  4970. {
  4971. RemoteFilenameArray copies;
  4972. public:
  4973. IMPLEMENT_IINTERFACE;
  4974. RemoteFilenameArray &queryCopies()
  4975. {
  4976. return copies;
  4977. }
  4978. IFile *open()
  4979. {
  4980. StringBuffer locations;
  4981. Owned<IException> exc;
  4982. ForEachItemIn(copy,copies) {
  4983. const RemoteFilename &rfn = copies.item(copy);
  4984. try {
  4985. OwnedIFile iFile = createIFile(rfn);
  4986. if (iFile->exists())
  4987. return iFile.getClear();
  4988. if (locations.length())
  4989. locations.append(", ");
  4990. rfn.getRemotePath(locations);
  4991. }
  4992. catch(IException *e) {
  4993. EXCLOG(e,"CReplicatedFile::open");
  4994. if (exc)
  4995. e->Release();
  4996. else
  4997. exc.setown(e);
  4998. }
  4999. }
  5000. if (exc.get())
  5001. throw exc.getClear();
  5002. throw MakeStringException(0, "%s: Failed to open part file at any of the following locations: ", locations.str());
  5003. }
  5004. };
  5005. IReplicatedFile *createReplicatedFile()
  5006. {
  5007. return new CReplicatedFile;
  5008. }
  5009. // ---------------------------------------------------------------------------------
  5010. class CSerialStreamBase : implements ISerialStream, public CInterface
  5011. {
  5012. private:
  5013. size32_t bufsize;
  5014. size32_t bufpos;
  5015. size32_t bufmax;
  5016. offset_t bufbase;
  5017. offset_t endpos; // -1 if not known
  5018. MemoryAttr ma;
  5019. byte *buf;
  5020. bool eoinput;
  5021. IFileSerialStreamCallback *tally;
  5022. inline size32_t doread(offset_t pos, size32_t max_size, void *ptr)
  5023. {
  5024. if (endpos!=(offset_t)-1) {
  5025. if (pos>=endpos)
  5026. return 0;
  5027. if (endpos-pos<max_size)
  5028. max_size = (size32_t)(endpos-pos);
  5029. }
  5030. size32_t size_read = rawread(pos, max_size, ptr);
  5031. if (tally)
  5032. tally->process(pos,size_read,ptr);
  5033. return size_read;
  5034. }
  5035. const void * dopeek(size32_t sz, size32_t &got) __attribute__((noinline))
  5036. {
  5037. for (;;)
  5038. {
  5039. size32_t left = bufmax-bufpos;
  5040. got = left;
  5041. if (left>=sz)
  5042. return buf+bufpos;
  5043. if (eoinput)
  5044. return left?(buf+bufpos):NULL;
  5045. size32_t reqsz = sz+bufsize; // NB not sz-left as want some slack
  5046. if (ma.length()<reqsz) {
  5047. MemoryAttr ma2;
  5048. void *nb = ma2.allocate(reqsz);
  5049. memcpy(nb,buf+bufpos,left);
  5050. ma.setOwn(reqsz,ma2.detach());
  5051. buf = (byte *)nb;
  5052. }
  5053. else
  5054. memmove(buf,buf+bufpos,left);
  5055. bufbase += bufpos;
  5056. size32_t rd = doread(bufbase+left,bufsize,buf+left);
  5057. if (!rd)
  5058. eoinput = true;
  5059. bufmax = rd+left;
  5060. bufpos = 0;
  5061. }
  5062. }
  5063. void getreadnext(size32_t len, void * ptr) __attribute__((noinline))
  5064. {
  5065. bufbase += bufmax;
  5066. bufpos = 0;
  5067. bufmax = 0;
  5068. size32_t rd = 0;
  5069. if (!eoinput) {
  5070. //If reading >= bufsize, read any complete blocks directly into the target
  5071. if (len>=bufsize) {
  5072. size32_t tord = (len/bufsize)*bufsize;
  5073. rd = doread(bufbase,tord,ptr);
  5074. bufbase += rd;
  5075. if (rd!=tord) {
  5076. eoinput = true;
  5077. PrintStackReport();
  5078. ERRLOG("CFileSerialStream::get read past end of stream.1 (%u,%u) %s",rd,tord,eoinput?"eoinput":"");
  5079. throw MakeStringException(-1,"CFileSerialStream::get read past end of stream");
  5080. }
  5081. len -= rd;
  5082. if (!len)
  5083. return;
  5084. ptr = (byte *)ptr+rd;
  5085. }
  5086. const void *p = dopeek(len,rd);
  5087. if (len<=rd) {
  5088. memcpy(ptr,p,len);
  5089. bufpos += len;
  5090. return;
  5091. }
  5092. }
  5093. PrintStackReport();
  5094. ERRLOG("CFileSerialStream::get read past end of stream.2 (%u,%u) %s",len,rd,eoinput?"eoinput":"");
  5095. throw MakeStringException(-1,"CFileSerialStream::get read past end of stream");
  5096. }
  5097. protected:
  5098. virtual size32_t rawread(offset_t pos, size32_t max_size, void *ptr) = 0;
  5099. public:
  5100. IMPLEMENT_IINTERFACE;
  5101. CSerialStreamBase(offset_t _offset, offset_t _len, size32_t _bufsize, IFileSerialStreamCallback *_tally)
  5102. {
  5103. tally = _tally;
  5104. bufsize = _bufsize;
  5105. if (bufsize==(size32_t)-1)
  5106. bufsize = DEFAULT_STREAM_BUFFER_SIZE;
  5107. if (bufsize<4096)
  5108. bufsize = 4096;
  5109. else
  5110. bufsize = ((bufsize+4095)/4096)*4096;
  5111. buf = (byte *)ma.allocate(bufsize+4096); // 4K initial slack
  5112. bufpos = 0;
  5113. bufmax = 0;
  5114. bufbase = _offset;
  5115. if (_len==(offset_t)-1)
  5116. endpos = (offset_t)-1;
  5117. else
  5118. endpos = _offset+_len;
  5119. eoinput = (_len==0);
  5120. }
  5121. void reset(offset_t _offset, offset_t _len)
  5122. {
  5123. bufpos = 0;
  5124. bufmax = 0;
  5125. bufbase = _offset;
  5126. if (_len==(offset_t)-1)
  5127. endpos = (offset_t)-1;
  5128. else
  5129. endpos = _offset+_len;
  5130. eoinput = (_len==0);
  5131. }
  5132. const void *peek(size32_t sz,size32_t &got)
  5133. {
  5134. return dopeek(sz, got);
  5135. }
  5136. void get(size32_t len, void * ptr)
  5137. {
  5138. size32_t cpy = bufmax-bufpos;
  5139. if (cpy>len)
  5140. cpy = len;
  5141. memcpy(ptr,(const byte *)buf+bufpos,cpy);
  5142. len -= cpy;
  5143. if (len==0) {
  5144. bufpos += cpy;
  5145. return;
  5146. }
  5147. return getreadnext(len, (byte *)ptr+cpy);
  5148. }
  5149. bool eos()
  5150. {
  5151. if (bufmax-bufpos)
  5152. return false;
  5153. size32_t rd;
  5154. return dopeek(1,rd)==NULL;
  5155. }
  5156. void skip(size32_t len)
  5157. {
  5158. size32_t left = bufmax-bufpos;
  5159. if (left>=len) {
  5160. bufpos += len;
  5161. return;
  5162. }
  5163. len -= left;
  5164. bufbase += bufmax;
  5165. bufpos = 0;
  5166. bufmax = 0;
  5167. if (!eoinput) {
  5168. while (len>=bufsize) {
  5169. size32_t rd;
  5170. if (tally) {
  5171. rd = doread(bufbase,bufsize,buf);
  5172. if (rd!=bufsize) {
  5173. eoinput = true;
  5174. throw MakeStringException(-1,"CFileSerialStream::skip read past end of stream");
  5175. }
  5176. }
  5177. else {
  5178. rd = (len/bufsize)*bufsize;
  5179. //rd=doskip(bufbase,bufsize,buf); to cope with reading from sockets etc?
  5180. }
  5181. bufbase += rd;
  5182. len -= bufsize;
  5183. }
  5184. if (len==0)
  5185. return;
  5186. size32_t got;
  5187. dopeek(len,got);
  5188. if (len<=got) {
  5189. bufpos += got;
  5190. return;
  5191. }
  5192. }
  5193. throw MakeStringException(-1,"CFileSerialStream::skip read past end of stream");
  5194. }
  5195. offset_t tell()
  5196. {
  5197. return bufbase+bufpos;
  5198. }
  5199. };
  5200. class CFileSerialStream: public CSerialStreamBase
  5201. {
  5202. Linked<IFileIO> fileio;
  5203. virtual size32_t rawread(offset_t pos, size32_t max_size, void *ptr)
  5204. {
  5205. return fileio->read(pos,max_size,ptr);
  5206. }
  5207. public:
  5208. CFileSerialStream(IFileIO *_fileio,offset_t _offset, offset_t _len, size32_t _bufsize, IFileSerialStreamCallback *_tally)
  5209. : CSerialStreamBase(_offset, _len, _bufsize, _tally), fileio(_fileio)
  5210. {
  5211. }
  5212. };
  5213. ISerialStream *createFileSerialStream(IFileIO *fileio,offset_t ofs, offset_t flen, size32_t bufsize,IFileSerialStreamCallback *callback)
  5214. {
  5215. if (!fileio)
  5216. return NULL;
  5217. return new CFileSerialStream(fileio,ofs,flen,bufsize,callback);
  5218. }
  5219. class CIoSerialStream: public CSerialStreamBase
  5220. {
  5221. Linked<IFileIOStream> io;
  5222. offset_t lastpos;
  5223. virtual size32_t rawread(offset_t pos, size32_t max_size, void *ptr)
  5224. {
  5225. if (lastpos!=pos)
  5226. throw MakeStringException(-1,"CIoSerialStream: non-sequential read (%" I64F "d,%" I64F "d)",lastpos,pos);
  5227. size32_t rd = io->read(max_size,ptr);
  5228. lastpos = pos+rd;
  5229. return rd;
  5230. }
  5231. public:
  5232. CIoSerialStream(IFileIOStream * _io,offset_t _offset, size32_t _bufsize, IFileSerialStreamCallback *_tally)
  5233. : CSerialStreamBase(_offset, (offset_t)-1, _bufsize, _tally), io(_io)
  5234. {
  5235. lastpos = _offset;
  5236. }
  5237. };
  5238. ISerialStream *createFileSerialStream(IFileIOStream *io,size32_t bufsize,IFileSerialStreamCallback *callback)
  5239. {
  5240. if (!io)
  5241. return NULL;
  5242. return new CIoSerialStream(io,0,bufsize,callback);
  5243. }
  5244. class CSocketSerialStream: public CSerialStreamBase
  5245. {
  5246. Linked<ISocket> socket;
  5247. unsigned timeout;
  5248. offset_t lastpos;
  5249. virtual size32_t rawread(offset_t pos, size32_t max_size, void *ptr)
  5250. {
  5251. if (lastpos!=pos)
  5252. throw MakeStringException(-1,"CSocketSerialStream: non-sequential read (%" I64F "d,%" I64F "d)",lastpos,pos);
  5253. size32_t size_read;
  5254. socket->readtms(ptr, 0, max_size, size_read, timeout);
  5255. lastpos = pos+size_read;
  5256. return size_read;
  5257. }
  5258. public:
  5259. CSocketSerialStream(ISocket * _socket, unsigned _timeout, offset_t _offset, size32_t _bufsize, IFileSerialStreamCallback *_tally)
  5260. : CSerialStreamBase(_offset, (offset_t)-1, _bufsize, _tally), socket(_socket), timeout(_timeout)
  5261. {
  5262. lastpos = _offset;
  5263. }
  5264. };
  5265. ISerialStream *createSocketSerialStream(ISocket * socket, unsigned timeoutms, size32_t bufsize, IFileSerialStreamCallback *callback)
  5266. {
  5267. if (!socket)
  5268. return NULL;
  5269. return new CSocketSerialStream(socket,timeoutms,0,bufsize,callback);
  5270. }
  5271. class CSimpleReadSerialStream: public CSerialStreamBase
  5272. {
  5273. Linked<ISimpleReadStream> input;
  5274. offset_t lastpos;
  5275. virtual size32_t rawread(offset_t pos, size32_t max_size, void *ptr)
  5276. {
  5277. if (lastpos!=pos)
  5278. throw MakeStringException(-1,"CSimpleReadSerialStream: non-sequential read (%" I64F "d,%" I64F "d)",lastpos,pos);
  5279. size32_t rd = input->read(max_size, ptr);
  5280. lastpos += rd;
  5281. return rd;
  5282. }
  5283. public:
  5284. CSimpleReadSerialStream(ISimpleReadStream * _input, offset_t _offset, size32_t _bufsize, IFileSerialStreamCallback *_tally)
  5285. : CSerialStreamBase(_offset, (offset_t)-1, _bufsize, _tally), input(_input)
  5286. {
  5287. lastpos = _offset;
  5288. }
  5289. void reset(offset_t _ofs, offset_t _len)
  5290. {
  5291. // assume knows what doing
  5292. lastpos = _ofs;
  5293. CSerialStreamBase::reset(_ofs,_len);
  5294. }
  5295. };
  5296. ISerialStream *createSimpleSerialStream(ISimpleReadStream * input, size32_t bufsize, IFileSerialStreamCallback *callback)
  5297. {
  5298. if (!input)
  5299. return NULL;
  5300. return new CSimpleReadSerialStream(input,0,bufsize,callback);
  5301. }
  5302. class CMemoryMappedSerialStream: implements ISerialStream, public CInterface
  5303. {
  5304. Linked<IMemoryMappedFile> mmfile;
  5305. const byte *mmbase;
  5306. memsize_t mmsize;
  5307. memsize_t mmofs;
  5308. bool eoinput;
  5309. IFileSerialStreamCallback *tally;
  5310. public:
  5311. IMPLEMENT_IINTERFACE;
  5312. CMemoryMappedSerialStream(IMemoryMappedFile *_mmfile, offset_t ofs, offset_t flen, IFileSerialStreamCallback *_tally)
  5313. : mmfile(_mmfile)
  5314. {
  5315. tally = _tally;
  5316. offset_t fs = mmfile->fileSize();
  5317. if ((memsize_t)fs!=fs)
  5318. throw MakeStringException(-1,"CMemoryMappedSerialStream file too big to be mapped");
  5319. if ((flen!=(offset_t)-1)&&(fs>flen))
  5320. fs = flen;
  5321. mmsize = (memsize_t)fs;
  5322. mmofs = (memsize_t)((ofs<fs)?ofs:fs);
  5323. mmbase = (const byte *)mmfile->base();
  5324. eoinput = false;
  5325. }
  5326. void reset(offset_t _ofs, offset_t _len)
  5327. {
  5328. offset_t fs = mmfile->fileSize();
  5329. if ((_len!=(offset_t)-1)&&(fs>_len))
  5330. fs = _len;
  5331. mmsize = (memsize_t)fs;
  5332. mmofs = (memsize_t)((_ofs<fs)?_ofs:fs);
  5333. mmsize = (memsize_t)fs;
  5334. }
  5335. CMemoryMappedSerialStream(const void *buf, memsize_t len, IFileSerialStreamCallback *_tally)
  5336. {
  5337. tally = _tally;
  5338. mmsize = len;
  5339. mmofs = 0;
  5340. mmbase = (const byte *)buf;
  5341. eoinput = false;
  5342. }
  5343. const void *peek(size32_t sz,size32_t &got)
  5344. {
  5345. memsize_t left = mmsize-mmofs;
  5346. if (sz>left)
  5347. sz = (size32_t)left;
  5348. else if (left>=UINT_MAX)
  5349. got = UINT_MAX-1;
  5350. else
  5351. got = (size32_t)left;
  5352. return mmbase+mmofs;
  5353. }
  5354. void get(size32_t len, void * ptr)
  5355. {
  5356. memsize_t left = mmsize-mmofs;
  5357. if (len>left) {
  5358. PrintStackReport();
  5359. ERRLOG("CFileSerialStream::get read past end of stream.3 (%u,%u)",(unsigned)len,(unsigned)left);
  5360. throw MakeStringException(-1,"CMemoryMappedSerialStream::get read past end of stream (%u,%u)",(unsigned)len,(unsigned)left);
  5361. }
  5362. if (tally)
  5363. tally->process(mmofs,len,mmbase+mmofs);
  5364. memcpy(ptr,mmbase+mmofs,len);
  5365. mmofs += len;
  5366. }
  5367. bool eos()
  5368. {
  5369. return (mmsize<=mmofs);
  5370. }
  5371. void skip(size32_t len)
  5372. {
  5373. memsize_t left = mmsize-mmofs;
  5374. if (len>left)
  5375. throw MakeStringException(-1,"CMemoryMappedSerialStream::skip read past end of stream (%u,%u)",(unsigned)len,(unsigned)left);
  5376. if (tally)
  5377. tally->process(mmofs,len,mmbase+mmofs);
  5378. mmofs += len;
  5379. }
  5380. offset_t tell()
  5381. {
  5382. return mmofs;
  5383. }
  5384. virtual void reset(offset_t _offset)
  5385. {
  5386. mmofs = (memsize_t)((_offset<mmsize)?_offset:mmsize);
  5387. eoinput = false;
  5388. }
  5389. };
  5390. ISerialStream *createFileSerialStream(IMemoryMappedFile *mmfile, offset_t ofs, offset_t flen, IFileSerialStreamCallback *callback)
  5391. {
  5392. return new CMemoryMappedSerialStream(mmfile,ofs,flen,callback);
  5393. }
  5394. ISerialStream *createMemorySerialStream(const void *buffer, memsize_t len, IFileSerialStreamCallback *callback)
  5395. {
  5396. return new CMemoryMappedSerialStream(buffer,len,callback);
  5397. }
  5398. class CMemoryBufferSerialStream: implements ISerialStream, public CInterface
  5399. {
  5400. MemoryBuffer & buffer;
  5401. IFileSerialStreamCallback *tally;
  5402. public:
  5403. IMPLEMENT_IINTERFACE;
  5404. CMemoryBufferSerialStream(MemoryBuffer & _buffer, IFileSerialStreamCallback * _tally)
  5405. : buffer(_buffer), tally(_tally)
  5406. {
  5407. }
  5408. virtual const void *peek(size32_t sz,size32_t &got)
  5409. {
  5410. got = buffer.remaining();
  5411. return buffer.readDirect(0);
  5412. }
  5413. virtual void get(size32_t len, void * ptr)
  5414. {
  5415. if (len>buffer.remaining()) {
  5416. ERRLOG("CMemoryBufferSerialStream::get read past end of stream.4(%u,%u)",(unsigned)len,(unsigned)buffer.remaining());
  5417. throw MakeStringException(-1,"CMemoryBufferSerialStream::get read past end of stream (%u,%u)",(unsigned)len,(unsigned)buffer.remaining());
  5418. }
  5419. const void * data = buffer.readDirect(len);
  5420. if (tally)
  5421. tally->process(buffer.getPos()-len,len,data);
  5422. memcpy(ptr,data,len);
  5423. }
  5424. virtual bool eos()
  5425. {
  5426. return buffer.remaining() == 0;
  5427. }
  5428. virtual void skip(size32_t len)
  5429. {
  5430. if (len>buffer.remaining())
  5431. throw MakeStringException(-1,"CMemoryBufferSerialStream::skip read past end of stream (%u,%u)",(unsigned)len,(unsigned)buffer.remaining());
  5432. const void * data = buffer.readDirect(len);
  5433. if (tally)
  5434. tally->process(buffer.getPos()-len,len,data);
  5435. }
  5436. virtual offset_t tell()
  5437. {
  5438. return buffer.getPos();
  5439. }
  5440. virtual void reset(offset_t _offset,offset_t _len)
  5441. {
  5442. size32_t ofs = (size32_t)_offset;
  5443. assertex(ofs==_offset);
  5444. assertex((_len==(offset_t)-1)||(_len>=buffer.length())); // don't support len on memory buffer
  5445. buffer.reset(ofs);
  5446. }
  5447. };
  5448. ISerialStream *createMemoryBufferSerialStream(MemoryBuffer & buffer, IFileSerialStreamCallback *callback)
  5449. {
  5450. return new CMemoryBufferSerialStream(buffer,callback);
  5451. }
  5452. // Memory Mapped Files
  5453. #define MEMORYMAP_PAGESIZE (0x1000) // could be different but won't ever be!
  5454. class CMemoryMappedFile: implements IMemoryMappedFile, public CInterface
  5455. {
  5456. byte *ptr; // base
  5457. offset_t ofs;
  5458. offset_t realofs; // rounded down to page size
  5459. offset_t filesize;
  5460. memsize_t size;
  5461. bool writeaccess;
  5462. memsize_t windowsize;
  5463. HANDLE hfile;
  5464. #ifdef _WIN32
  5465. static size32_t pagesize;
  5466. HANDLE hmap;
  5467. #endif
  5468. Linked<IFile> file;
  5469. inline offset_t pageround(offset_t o)
  5470. {
  5471. #ifdef _WIN32
  5472. return o-(o%pagesize);
  5473. #else
  5474. return o-(o%MEMORYMAP_PAGESIZE);
  5475. #endif
  5476. }
  5477. inline void * realptr()
  5478. {
  5479. return ptr?(ptr-(ofs-realofs)):NULL;
  5480. }
  5481. inline memsize_t realsize()
  5482. {
  5483. return (memsize_t)(size+(ofs-realofs));
  5484. }
  5485. public:
  5486. IMPLEMENT_IINTERFACE;
  5487. CMemoryMappedFile(HANDLE _hfile, offset_t _filesize, offset_t _ofs, memsize_t _size, bool _writeaccess)
  5488. {
  5489. hfile = _hfile;
  5490. #ifdef _WIN32
  5491. if (pagesize==0) {
  5492. SYSTEM_INFO sysinfo;
  5493. GetSystemInfo(&sysinfo);
  5494. pagesize = sysinfo.dwAllocationGranularity;
  5495. }
  5496. hmap = NULLFILE;
  5497. #endif
  5498. ptr = NULL;
  5499. filesize = _filesize;
  5500. windowsize = _size; // first open size assumed to be window size
  5501. reinit(_ofs, _size, _writeaccess);
  5502. }
  5503. ~CMemoryMappedFile()
  5504. {
  5505. #ifdef _WIN32
  5506. if (hmap!=NULLFILE)
  5507. CloseHandle(hmap);
  5508. CloseHandle(hfile);
  5509. #else
  5510. close(hfile);
  5511. #endif
  5512. }
  5513. byte *base() { return ptr; }
  5514. offset_t offset() { return ofs; }
  5515. virtual memsize_t length() { return size; }
  5516. virtual offset_t fileSize() { return filesize; }
  5517. virtual int compareWithin(const void *p)
  5518. {
  5519. if (p<ptr)
  5520. return -1;
  5521. if (p>=size+ptr)
  5522. return 1;
  5523. return 0;
  5524. }
  5525. bool writeAccess() { return writeaccess; }
  5526. void flush()
  5527. {
  5528. if (ptr) {
  5529. #ifdef _WIN32
  5530. FlushViewOfFile(realptr(),0);
  5531. #elif defined(__linux__)
  5532. msync(realptr(),realsize(),MS_SYNC);
  5533. #else
  5534. UNIMPLEMENTED;
  5535. #endif
  5536. }
  5537. }
  5538. byte *nextPtr(const void *p,offset_t skip, memsize_t req, memsize_t &got)
  5539. {
  5540. // for scanning in sequence
  5541. if (p==NULL)
  5542. p = ptr;
  5543. else if ((p<ptr)||(p>ptr+size))
  5544. throw MakeStringException(-1,"CMemoryMappedFile::nextPtr - outside map");
  5545. memsize_t d = (byte *)p-ptr;
  5546. offset_t o = ofs+d+skip;
  5547. if (o>=filesize) {
  5548. got = 0;
  5549. return NULL;
  5550. }
  5551. offset_t left = filesize-o;
  5552. if (left<req)
  5553. req = (memsize_t)left;
  5554. if (o+req-ofs>size) {
  5555. reinit(o,windowsize,writeaccess);
  5556. assertex(o==ofs);
  5557. got = windowsize;
  5558. if (left<got)
  5559. got = (memsize_t)left;
  5560. return ptr;
  5561. }
  5562. got = (memsize_t)(size+o-ofs);
  5563. if (left<got)
  5564. got = (memsize_t)left;
  5565. return (byte *)p+skip;
  5566. }
  5567. virtual void reinit(offset_t _ofs, memsize_t _size, bool _writeaccess)
  5568. {
  5569. writeaccess = _writeaccess;
  5570. if (ptr) {
  5571. #ifdef _WIN32
  5572. // no need to unmap view
  5573. if (hmap != INVALID_HANDLE_VALUE) {
  5574. CloseHandle(hmap);
  5575. hmap = INVALID_HANDLE_VALUE;
  5576. }
  5577. #elif defined(__linux__)
  5578. munmap(realptr(),realsize());
  5579. // error checking TBD
  5580. #else
  5581. UNIMPLEMENTED;
  5582. #endif
  5583. ptr = NULL;
  5584. }
  5585. if (_ofs>filesize)
  5586. ofs = filesize;
  5587. else
  5588. ofs = _ofs;
  5589. realofs = pageround(ofs);
  5590. if (_size==(memsize_t)-1) {
  5591. size = (memsize_t)(filesize-_ofs);
  5592. if (size!=(filesize-_ofs))
  5593. throw MakeStringException(-1,"CMemoryMappedFile::reinit file too big");
  5594. }
  5595. else
  5596. size = _size;
  5597. memsize_t mapsz = realsize();
  5598. if (filesize-realofs<mapsz)
  5599. mapsz = (memsize_t)(filesize-realofs);
  5600. #ifdef _WIN32
  5601. LARGE_INTEGER li;
  5602. if (hmap == INVALID_HANDLE_VALUE) {
  5603. hmap = CreateFileMapping(hfile,NULL,writeaccess?PAGE_READWRITE:PAGE_READONLY,0, 0, NULL);
  5604. if (!hmap) {
  5605. DWORD err = GetLastError();
  5606. throw makeOsException(err,"CMemoryMappedFile::reinit");
  5607. }
  5608. }
  5609. li.QuadPart = realofs;
  5610. ptr = (byte *) MapViewOfFile(hmap, writeaccess?(FILE_MAP_READ|FILE_MAP_WRITE):FILE_MAP_READ, li.HighPart, li.LowPart, mapsz);
  5611. if (!ptr) {
  5612. DWORD err = GetLastError();
  5613. throw makeOsException(err,"CMemoryMappedFile::reinit");
  5614. }
  5615. #elif defined (__linux__)
  5616. ptr = (byte *) mmap(NULL, mapsz, writeaccess?(PROT_READ|PROT_WRITE):PROT_READ, MAP_SHARED|MAP_NORESERVE, hfile, realofs);
  5617. // error checking TBD
  5618. #else
  5619. UNIMPLEMENTED;
  5620. #endif
  5621. if (ptr)
  5622. ptr += (ofs-realofs);
  5623. }
  5624. };
  5625. #ifdef _WIN32
  5626. size32_t CMemoryMappedFile::pagesize=0;
  5627. #endif
  5628. // TBD MADV_SEQUENTIAL & MADV_WILLNEED?
  5629. IMemoryMappedFile *CFile::openMemoryMapped(offset_t ofs, memsize_t len, bool write)
  5630. {
  5631. HANDLE hfile = openHandle(write?IFOcreaterw:IFOread, IFSHread, false);
  5632. if (hfile == NULLFILE)
  5633. return NULL;
  5634. return new CMemoryMappedFile(hfile,size(),ofs,len,write);
  5635. }
  5636. bool isSpecialPath(const char *path)
  5637. {
  5638. // used for remote queries
  5639. if (!path)
  5640. return false;
  5641. if (isPathSepChar(path[0])&&(path[0]==path[1])) {
  5642. path += 2;
  5643. for (;;) {
  5644. if (*path=='/')
  5645. return (path[1]=='>');
  5646. if (!*path||(*path=='\\'))
  5647. return false;
  5648. path++;
  5649. }
  5650. }
  5651. return (path&&(*path=='/')&&(path[1]=='>'));
  5652. }
  5653. size32_t SendFile(ISocket *target, IFileIO *fileio,offset_t start,size32_t len)
  5654. {
  5655. assertex(target);
  5656. assertex(fileio);
  5657. offset_t fsz = fileio->size();
  5658. if (start>=fsz)
  5659. return 0;
  5660. if (start+len>fsz)
  5661. len = (size32_t)(fsz-start);
  5662. if (!len)
  5663. return 0;
  5664. CFileIO *cfile = QUERYINTERFACE(fileio,CFileIO);
  5665. if (cfile) {
  5666. HANDLE fh = cfile->queryHandle();
  5667. if ((fh!=(HANDLE)0)&&(fh!=(HANDLE)-1)) {
  5668. unsigned sh = target->OShandle();
  5669. if ((sh!=(unsigned)0)&&(sh!=(unsigned)-1)) {
  5670. #ifdef _WIN32
  5671. // MORE - should there be something here?
  5672. #elif defined(__linux__)
  5673. // TransmitFile not in std DLLs so don't bother with
  5674. off_t ofs = start;
  5675. ssize_t sent = ::sendfile(sh,fh,&ofs,len);
  5676. if (sent!=(ssize_t)-1)
  5677. return (size32_t)sent;
  5678. int err = errno;
  5679. if ((err!=EINVAL)&&(err!=ENOSYS))
  5680. throw makeOsException(err, "sendfile");
  5681. #else
  5682. UNIMPLEMENTED;
  5683. #endif
  5684. }
  5685. }
  5686. }
  5687. #ifdef _DEBUG
  5688. #ifndef _WIN32
  5689. WARNLOG("SendFile falling back");
  5690. #endif
  5691. #endif
  5692. // fallback
  5693. MemoryAttr ma;
  5694. void *buf = ma.allocate(len);
  5695. size32_t rd = fileio->read(start,len,buf);
  5696. target->write(buf,rd);
  5697. return rd;
  5698. }
  5699. void asyncClose(IFileIO *io)
  5700. {
  5701. if (!io)
  5702. return;
  5703. static CriticalSection ADsect;
  5704. CriticalBlock block(ADsect);
  5705. static Owned<IWorkQueueThread> adwp = createWorkQueueThread();
  5706. class cWQI: public CInterface,implements IWorkQueueItem
  5707. {
  5708. Owned<IFileIO> io;
  5709. public:
  5710. IMPLEMENT_IINTERFACE;
  5711. cWQI(IFileIO *_io)
  5712. : io(_io)
  5713. {
  5714. }
  5715. void execute()
  5716. {
  5717. io.clear();
  5718. }
  5719. };
  5720. adwp->post(new cWQI(io));
  5721. };
  5722. int stdIoHandle(const char *path)
  5723. {
  5724. if (strcmp(path,"stdin:")==0)
  5725. return 0;
  5726. if (strcmp(path,"stdout:")==0)
  5727. return 1;
  5728. if (strcmp(path,"stderr:")==0)
  5729. return 2;
  5730. return -1;
  5731. }
  5732. extern jlib_decl bool containsFileWildcard(const char * path)
  5733. {
  5734. if (!path)
  5735. return false;
  5736. return strchr(path, '*') || strchr(path, '?');
  5737. }
  5738. extern jlib_decl bool isDirectory(const char * path)
  5739. {
  5740. Owned<IFile> file = createIFile(path);
  5741. return file->isDirectory() == foundYes;
  5742. }
  5743. // IFileIOCache
  5744. class CLazyFileIOCache;
  5745. class CCachedFileIO: implements IFileIO, public CInterface
  5746. {
  5747. CLazyFileIOCache &owner;
  5748. RemoteFilename filename;
  5749. CriticalSection &sect;
  5750. CRuntimeStatisticCollection fileStats;
  5751. IFOmode mode;
  5752. void writeNotSupported(const char *s)
  5753. {
  5754. StringBuffer tmp;
  5755. filename.getRemotePath(tmp);
  5756. throw MakeStringException(-1, "CCachedFileIO(%s) %s not supported", tmp.str(), s);
  5757. }
  5758. public:
  5759. unsigned accesst;
  5760. Owned<IFileIO> cachedio;
  5761. CCachedFileIO(CLazyFileIOCache &_owner, CriticalSection &_sect, RemoteFilename &_filename, IFOmode _mode)
  5762. : owner(_owner), sect(_sect), fileStats(diskLocalStatistics)
  5763. {
  5764. filename.set(_filename);
  5765. mode = _mode;
  5766. accesst = 0;
  5767. }
  5768. virtual void Link(void) const { CInterface::Link(); } \
  5769. virtual bool Release(void) const;
  5770. unsigned __int64 getStatistic(StatisticKind kind)
  5771. {
  5772. CriticalBlock block(sect);
  5773. unsigned __int64 openValue = cachedio ? cachedio->getStatistic(kind) : 0;
  5774. return openValue + fileStats.getStatisticValue(kind);
  5775. }
  5776. IFileIO *open();
  5777. size32_t read(offset_t pos, size32_t len, void * data)
  5778. {
  5779. CriticalBlock block(sect);
  5780. Owned<IFileIO> io = open();
  5781. return io->read(pos,len,data);
  5782. }
  5783. offset_t size()
  5784. {
  5785. CriticalBlock block(sect);
  5786. Owned<IFileIO> io = open();
  5787. return io->size();
  5788. }
  5789. size32_t write(offset_t pos, size32_t len, const void * data)
  5790. {
  5791. CriticalBlock block(sect);
  5792. Owned<IFileIO> io = open();
  5793. return io->write(pos,len,data);
  5794. }
  5795. virtual void flush()
  5796. {
  5797. CriticalBlock block(sect);
  5798. if (cachedio)
  5799. cachedio->flush();
  5800. }
  5801. virtual void close()
  5802. {
  5803. CriticalBlock block(sect);
  5804. if (cachedio)
  5805. forceClose();
  5806. }
  5807. offset_t appendFile(IFile *file,offset_t pos,offset_t len)
  5808. {
  5809. CriticalBlock block(sect);
  5810. Owned<IFileIO> io = open();
  5811. return io->appendFile(file,pos,len);
  5812. }
  5813. void setSize(offset_t size)
  5814. {
  5815. CriticalBlock block(sect);
  5816. Owned<IFileIO> io = open();
  5817. io->setSize(size);
  5818. }
  5819. void forceClose()
  5820. {
  5821. cachedio->close();
  5822. mergeStats(fileStats, cachedio);
  5823. cachedio.clear();
  5824. }
  5825. };
  5826. class CLazyFileIOCache: implements IFileIOCache, public CInterface
  5827. {
  5828. CriticalSection sect;
  5829. unsigned max;
  5830. IPointerArrayOf<CCachedFileIO> cache;
  5831. public:
  5832. IMPLEMENT_IINTERFACE;
  5833. CLazyFileIOCache(unsigned _max)
  5834. {
  5835. max = _max;
  5836. }
  5837. CCachedFileIO *addFile( RemoteFilename &filename, IFOmode mode )
  5838. {
  5839. // check exists
  5840. unsigned i = cache.ordinality();
  5841. ForEachItemIn(i2,cache) {
  5842. if (!cache.item(i2)) {
  5843. i = i2;
  5844. break;
  5845. }
  5846. }
  5847. Linked<CCachedFileIO> ret = new CCachedFileIO(*this,sect,filename,mode);
  5848. if (i<cache.ordinality())
  5849. cache.replace(ret,i);
  5850. else
  5851. cache.append(ret);
  5852. return ret.getClear();
  5853. }
  5854. void checkCache(unsigned wanted=1)
  5855. {
  5856. // called in sect
  5857. for (;;) {
  5858. CCachedFileIO *oldest = NULL;
  5859. unsigned oldestt = 0;
  5860. unsigned t = msTick();
  5861. unsigned n = wanted;
  5862. ForEachItemIn(i1,cache) {
  5863. CCachedFileIO *cio = cache.item(i1);
  5864. if (cio&&cio->cachedio.get()) {
  5865. if (t-cio->accesst>=oldestt) {
  5866. oldest = cio;
  5867. oldestt = t-cio->accesst;
  5868. }
  5869. n++;
  5870. }
  5871. }
  5872. if (n<max)
  5873. break;
  5874. if (!oldest)
  5875. break;
  5876. oldest->forceClose();
  5877. //If previously had max ios then we now have space.
  5878. if (n == max)
  5879. break;
  5880. }
  5881. }
  5882. const CCachedFileIO *removeFile(const CCachedFileIO *io)
  5883. {
  5884. CriticalBlock block(sect);
  5885. ForEachItemIn(idx, cache)
  5886. {
  5887. if (io == cache.item(idx))
  5888. {
  5889. cache.replace(NULL, idx, true);
  5890. break;
  5891. }
  5892. }
  5893. return io;
  5894. }
  5895. };
  5896. bool CCachedFileIO::Release(void) const
  5897. {
  5898. if (CInterface::Release())
  5899. return true;
  5900. if (!CInterface::IsShared())
  5901. return owner.removeFile(this)->Release();
  5902. return false;
  5903. }
  5904. IFileIO *CCachedFileIO::open()
  5905. {
  5906. // called in sect
  5907. if (!cachedio) {
  5908. owner.checkCache();
  5909. Owned<IFile> inFile = createIFile(filename);
  5910. cachedio.setown(inFile->open(mode));
  5911. if (!cachedio.get()) {
  5912. StringBuffer tmp;
  5913. filename.getRemotePath(tmp);
  5914. throw MakeStringException(-1, "CCachedFileIO::open(%d) '%s' failed", (int)mode, tmp.str());
  5915. }
  5916. }
  5917. accesst = msTick();
  5918. return cachedio.getLink();
  5919. }
  5920. IFileIOCache* createFileIOCache(unsigned max)
  5921. {
  5922. return new CLazyFileIOCache(max);
  5923. }
  5924. extern jlib_decl IFile * createSentinelTarget()
  5925. {
  5926. const char * sentinelFilename = getenv("SENTINEL");
  5927. if (sentinelFilename)
  5928. return createIFile(sentinelFilename);
  5929. else
  5930. return NULL;
  5931. }
  5932. extern jlib_decl void removeSentinelFile(IFile * sentinelFile)
  5933. {
  5934. if (sentinelFile)
  5935. {
  5936. if(sentinelFile->exists() && !sentinelFile->isDirectory())
  5937. {
  5938. DBGLOG("Removing sentinel file %s", sentinelFile->queryFilename());
  5939. try
  5940. {
  5941. sentinelFile->remove();
  5942. }
  5943. catch(IException *E)
  5944. {
  5945. StringBuffer s;
  5946. EXCLOG(E, s.appendf("Failed to remove sentinel file %s", sentinelFile->queryFilename()).str());
  5947. E->Release();
  5948. throw makeOsException(errno, "removeSentinelFile - file cannot be removed.");
  5949. }
  5950. }
  5951. }
  5952. }
  5953. extern jlib_decl void writeSentinelFile(IFile * sentinelFile)
  5954. {
  5955. if ( sentinelFile )
  5956. {
  5957. DBGLOG("Creating sentinel file %s for rerun from script", sentinelFile->queryFilename());
  5958. try
  5959. {
  5960. Owned<IFileIO> sentinel = sentinelFile->open(IFOcreate);
  5961. sentinel->write(0, 5, "rerun");
  5962. }
  5963. catch(IException *E)
  5964. {
  5965. StringBuffer s;
  5966. EXCLOG(E, s.appendf("Failed to create sentinel file %s for rerun from script", sentinelFile->queryFilename()).str());
  5967. E->Release();
  5968. throw makeOsException(errno, "writeSentinelFile - file not created.");
  5969. }
  5970. }
  5971. }
  5972. jlib_decl StringBuffer & appendCurrentDirectory(StringBuffer & target, bool blankIfFails)
  5973. {
  5974. char temp[_MAX_PATH+1];
  5975. if (!getcwd(temp,sizeof(temp)))
  5976. {
  5977. if (blankIfFails)
  5978. return target;
  5979. throw MakeStringException(JLIBERR_InternalError, "getcwd failed (%d)", errno);
  5980. }
  5981. return target.append(temp);
  5982. }
  5983. void removeFileTraceIfFail(const char * filename)
  5984. {
  5985. if (remove(filename) != 0)
  5986. DBGLOG("Could not remove file '%s'", filename);
  5987. }