12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712171317141715171617171718171917201721172217231724172517261727172817291730173117321733173417351736173717381739174017411742174317441745174617471748174917501751175217531754175517561757175817591760176117621763176417651766176717681769177017711772177317741775177617771778177917801781178217831784178517861787178817891790179117921793179417951796179717981799180018011802180318041805180618071808180918101811181218131814181518161817181818191820182118221823182418251826182718281829183018311832183318341835183618371838183918401841184218431844184518461847184818491850185118521853185418551856185718581859186018611862186318641865186618671868186918701871187218731874187518761877187818791880188118821883188418851886188718881889189018911892189318941895189618971898189919001901190219031904190519061907190819091910191119121913191419151916191719181919192019211922192319241925192619271928192919301931193219331934193519361937193819391940194119421943194419451946194719481949195019511952195319541955195619571958195919601961196219631964196519661967196819691970197119721973197419751976197719781979198019811982198319841985198619871988198919901991199219931994199519961997199819992000200120022003200420052006200720082009201020112012201320142015201620172018201920202021202220232024202520262027202820292030203120322033203420352036203720382039204020412042204320442045204620472048204920502051205220532054205520562057205820592060206120622063206420652066206720682069207020712072207320742075207620772078207920802081208220832084208520862087208820892090209120922093209420952096209720982099210021012102210321042105210621072108210921102111211221132114211521162117211821192120212121222123212421252126212721282129213021312132213321342135213621372138213921402141214221432144214521462147214821492150215121522153215421552156215721582159216021612162216321642165216621672168216921702171217221732174217521762177217821792180218121822183218421852186218721882189219021912192219321942195219621972198219922002201220222032204220522062207220822092210221122122213221422152216221722182219222022212222222322242225222622272228222922302231223222332234223522362237223822392240224122422243224422452246224722482249225022512252225322542255225622572258225922602261226222632264226522662267226822692270227122722273227422752276227722782279228022812282228322842285228622872288228922902291229222932294229522962297229822992300230123022303230423052306230723082309231023112312231323142315231623172318231923202321232223232324232523262327232823292330233123322333233423352336233723382339234023412342234323442345234623472348234923502351235223532354235523562357235823592360236123622363236423652366236723682369237023712372237323742375237623772378237923802381238223832384238523862387238823892390239123922393239423952396239723982399240024012402240324042405240624072408240924102411241224132414241524162417241824192420242124222423242424252426242724282429243024312432243324342435243624372438243924402441244224432444244524462447244824492450245124522453245424552456245724582459246024612462246324642465246624672468246924702471247224732474247524762477247824792480248124822483248424852486248724882489249024912492249324942495249624972498249925002501250225032504250525062507250825092510251125122513251425152516251725182519252025212522252325242525252625272528252925302531253225332534253525362537253825392540254125422543254425452546254725482549255025512552255325542555255625572558255925602561256225632564256525662567256825692570257125722573257425752576257725782579258025812582258325842585258625872588258925902591259225932594259525962597259825992600260126022603260426052606260726082609261026112612261326142615261626172618261926202621262226232624262526262627262826292630263126322633263426352636263726382639264026412642264326442645264626472648264926502651265226532654265526562657265826592660266126622663266426652666266726682669267026712672267326742675267626772678267926802681268226832684268526862687268826892690269126922693269426952696269726982699270027012702270327042705270627072708270927102711271227132714271527162717271827192720272127222723272427252726272727282729273027312732273327342735273627372738273927402741274227432744274527462747274827492750275127522753275427552756275727582759276027612762276327642765276627672768276927702771277227732774277527762777277827792780278127822783278427852786278727882789279027912792279327942795279627972798279928002801280228032804280528062807280828092810281128122813281428152816281728182819282028212822282328242825282628272828282928302831283228332834283528362837283828392840284128422843284428452846284728482849285028512852285328542855285628572858285928602861286228632864286528662867286828692870287128722873287428752876287728782879288028812882288328842885288628872888288928902891289228932894289528962897289828992900290129022903290429052906290729082909291029112912291329142915291629172918291929202921292229232924292529262927292829292930293129322933293429352936293729382939294029412942294329442945294629472948294929502951295229532954295529562957295829592960296129622963296429652966296729682969297029712972297329742975297629772978297929802981298229832984298529862987298829892990299129922993299429952996299729982999300030013002300330043005300630073008300930103011301230133014301530163017301830193020302130223023302430253026302730283029303030313032303330343035303630373038303930403041304230433044304530463047304830493050305130523053305430553056305730583059306030613062306330643065306630673068306930703071307230733074307530763077307830793080308130823083308430853086308730883089309030913092309330943095309630973098309931003101310231033104310531063107310831093110311131123113311431153116311731183119312031213122312331243125312631273128312931303131313231333134313531363137313831393140314131423143314431453146314731483149315031513152315331543155315631573158315931603161316231633164316531663167316831693170317131723173317431753176317731783179318031813182318331843185318631873188318931903191319231933194319531963197319831993200320132023203320432053206320732083209321032113212321332143215321632173218321932203221322232233224322532263227322832293230323132323233323432353236323732383239324032413242324332443245324632473248324932503251325232533254325532563257325832593260326132623263326432653266326732683269327032713272327332743275327632773278327932803281328232833284328532863287328832893290329132923293329432953296329732983299330033013302330333043305330633073308330933103311331233133314331533163317331833193320332133223323332433253326332733283329333033313332333333343335333633373338333933403341334233433344334533463347334833493350335133523353335433553356335733583359336033613362336333643365336633673368336933703371337233733374337533763377337833793380338133823383338433853386338733883389339033913392339333943395339633973398339934003401340234033404340534063407340834093410341134123413341434153416341734183419342034213422342334243425342634273428342934303431343234333434343534363437343834393440344134423443344434453446344734483449345034513452345334543455345634573458345934603461346234633464346534663467346834693470347134723473347434753476347734783479348034813482348334843485348634873488348934903491349234933494349534963497349834993500350135023503350435053506350735083509351035113512351335143515351635173518351935203521352235233524352535263527352835293530353135323533353435353536353735383539354035413542354335443545354635473548354935503551355235533554355535563557355835593560356135623563356435653566356735683569357035713572357335743575357635773578357935803581358235833584358535863587358835893590359135923593359435953596359735983599360036013602360336043605360636073608360936103611361236133614361536163617361836193620362136223623362436253626362736283629363036313632363336343635363636373638363936403641364236433644364536463647364836493650365136523653365436553656365736583659366036613662366336643665366636673668366936703671367236733674367536763677367836793680368136823683368436853686368736883689369036913692369336943695369636973698369937003701370237033704370537063707370837093710371137123713371437153716371737183719372037213722372337243725372637273728372937303731373237333734373537363737373837393740374137423743374437453746374737483749375037513752375337543755375637573758375937603761376237633764376537663767376837693770377137723773377437753776377737783779378037813782378337843785378637873788378937903791379237933794379537963797379837993800380138023803380438053806380738083809381038113812381338143815381638173818381938203821382238233824382538263827382838293830383138323833383438353836383738383839384038413842384338443845384638473848384938503851385238533854385538563857385838593860386138623863386438653866386738683869387038713872387338743875387638773878387938803881388238833884388538863887388838893890389138923893389438953896389738983899390039013902390339043905390639073908390939103911391239133914391539163917391839193920392139223923392439253926392739283929393039313932393339343935393639373938393939403941394239433944394539463947394839493950395139523953395439553956395739583959396039613962396339643965396639673968396939703971397239733974397539763977397839793980398139823983398439853986398739883989399039913992399339943995399639973998399940004001400240034004400540064007400840094010401140124013401440154016401740184019402040214022402340244025402640274028402940304031403240334034403540364037403840394040404140424043404440454046404740484049405040514052405340544055405640574058405940604061406240634064406540664067406840694070407140724073407440754076407740784079408040814082408340844085408640874088408940904091409240934094409540964097409840994100410141024103410441054106410741084109411041114112411341144115411641174118411941204121412241234124412541264127412841294130413141324133413441354136413741384139414041414142414341444145414641474148414941504151415241534154415541564157415841594160416141624163416441654166416741684169417041714172417341744175417641774178417941804181418241834184418541864187418841894190419141924193419441954196419741984199420042014202420342044205420642074208420942104211421242134214421542164217421842194220422142224223422442254226422742284229423042314232423342344235423642374238423942404241424242434244424542464247424842494250425142524253425442554256425742584259426042614262426342644265426642674268426942704271427242734274427542764277427842794280428142824283428442854286428742884289429042914292429342944295429642974298429943004301430243034304430543064307430843094310431143124313431443154316431743184319432043214322432343244325432643274328432943304331433243334334433543364337433843394340434143424343434443454346434743484349435043514352435343544355435643574358435943604361436243634364436543664367436843694370437143724373437443754376437743784379438043814382438343844385438643874388438943904391439243934394439543964397439843994400440144024403440444054406440744084409441044114412441344144415441644174418441944204421442244234424442544264427442844294430443144324433443444354436443744384439444044414442444344444445444644474448444944504451445244534454445544564457445844594460446144624463446444654466446744684469447044714472447344744475447644774478447944804481448244834484448544864487448844894490449144924493449444954496449744984499450045014502450345044505450645074508450945104511451245134514451545164517451845194520452145224523452445254526452745284529453045314532453345344535453645374538453945404541454245434544454545464547454845494550455145524553455445554556455745584559456045614562456345644565456645674568456945704571457245734574457545764577457845794580458145824583458445854586458745884589459045914592459345944595459645974598459946004601460246034604460546064607460846094610461146124613461446154616461746184619462046214622462346244625462646274628462946304631463246334634463546364637463846394640464146424643464446454646464746484649465046514652465346544655465646574658465946604661466246634664466546664667466846694670467146724673467446754676467746784679468046814682468346844685468646874688468946904691469246934694469546964697469846994700470147024703470447054706470747084709471047114712471347144715471647174718471947204721472247234724472547264727472847294730473147324733473447354736473747384739474047414742474347444745474647474748474947504751475247534754475547564757475847594760476147624763476447654766476747684769477047714772477347744775477647774778477947804781478247834784478547864787478847894790479147924793479447954796479747984799480048014802480348044805480648074808480948104811481248134814481548164817481848194820482148224823482448254826482748284829483048314832483348344835483648374838483948404841484248434844484548464847484848494850485148524853485448554856485748584859486048614862486348644865486648674868486948704871487248734874487548764877487848794880488148824883488448854886488748884889489048914892489348944895489648974898489949004901490249034904490549064907490849094910491149124913491449154916491749184919492049214922492349244925492649274928492949304931493249334934493549364937493849394940494149424943494449454946494749484949495049514952495349544955495649574958495949604961496249634964496549664967496849694970497149724973497449754976497749784979498049814982498349844985498649874988498949904991499249934994499549964997499849995000500150025003500450055006500750085009501050115012501350145015501650175018501950205021502250235024502550265027502850295030503150325033503450355036503750385039504050415042504350445045504650475048504950505051505250535054505550565057505850595060506150625063506450655066506750685069507050715072507350745075507650775078507950805081508250835084508550865087508850895090509150925093509450955096509750985099510051015102510351045105510651075108510951105111511251135114511551165117511851195120512151225123512451255126512751285129513051315132513351345135513651375138513951405141514251435144514551465147514851495150515151525153515451555156515751585159516051615162516351645165516651675168516951705171517251735174517551765177517851795180518151825183518451855186518751885189519051915192519351945195519651975198519952005201520252035204520552065207520852095210521152125213521452155216521752185219522052215222522352245225522652275228522952305231523252335234523552365237523852395240524152425243524452455246524752485249525052515252525352545255525652575258525952605261526252635264526552665267526852695270527152725273527452755276527752785279528052815282528352845285528652875288528952905291529252935294529552965297529852995300530153025303530453055306530753085309531053115312531353145315531653175318531953205321532253235324532553265327532853295330533153325333533453355336533753385339534053415342534353445345534653475348534953505351535253535354535553565357535853595360536153625363536453655366536753685369537053715372537353745375537653775378537953805381538253835384538553865387538853895390539153925393539453955396539753985399540054015402540354045405540654075408540954105411541254135414541554165417541854195420542154225423542454255426542754285429543054315432543354345435543654375438543954405441544254435444544554465447544854495450545154525453545454555456545754585459546054615462546354645465546654675468546954705471547254735474547554765477547854795480548154825483548454855486548754885489549054915492549354945495549654975498549955005501550255035504550555065507550855095510551155125513551455155516551755185519552055215522552355245525552655275528552955305531553255335534553555365537553855395540554155425543554455455546554755485549555055515552555355545555555655575558555955605561556255635564556555665567556855695570557155725573557455755576557755785579558055815582558355845585558655875588558955905591559255935594559555965597559855995600560156025603560456055606560756085609561056115612561356145615561656175618561956205621562256235624562556265627562856295630563156325633563456355636563756385639564056415642564356445645564656475648564956505651565256535654565556565657565856595660566156625663566456655666566756685669567056715672567356745675567656775678567956805681568256835684568556865687568856895690569156925693569456955696569756985699570057015702570357045705570657075708570957105711571257135714571557165717571857195720572157225723572457255726572757285729573057315732573357345735573657375738573957405741574257435744574557465747574857495750575157525753575457555756575757585759576057615762576357645765576657675768576957705771577257735774577557765777577857795780578157825783578457855786578757885789579057915792579357945795579657975798579958005801580258035804580558065807580858095810581158125813581458155816581758185819582058215822582358245825582658275828582958305831583258335834583558365837583858395840584158425843584458455846584758485849585058515852585358545855585658575858585958605861586258635864586558665867586858695870587158725873587458755876587758785879588058815882588358845885588658875888588958905891589258935894589558965897589858995900590159025903590459055906590759085909591059115912591359145915591659175918591959205921592259235924592559265927592859295930593159325933593459355936593759385939594059415942594359445945594659475948594959505951595259535954595559565957595859595960596159625963596459655966596759685969597059715972597359745975597659775978597959805981598259835984598559865987598859895990599159925993599459955996599759985999600060016002600360046005600660076008600960106011601260136014601560166017601860196020602160226023602460256026602760286029603060316032603360346035603660376038603960406041604260436044604560466047604860496050605160526053605460556056605760586059606060616062606360646065606660676068606960706071607260736074607560766077607860796080608160826083608460856086608760886089609060916092609360946095609660976098609961006101610261036104610561066107610861096110611161126113611461156116611761186119612061216122612361246125612661276128612961306131613261336134613561366137613861396140614161426143614461456146614761486149615061516152615361546155615661576158615961606161616261636164616561666167616861696170617161726173617461756176617761786179618061816182618361846185618661876188618961906191619261936194619561966197619861996200620162026203620462056206620762086209621062116212621362146215621662176218621962206221622262236224622562266227622862296230623162326233623462356236623762386239624062416242624362446245624662476248624962506251625262536254625562566257625862596260626162626263626462656266626762686269627062716272627362746275627662776278627962806281628262836284628562866287628862896290629162926293629462956296629762986299630063016302630363046305630663076308630963106311631263136314631563166317631863196320632163226323632463256326632763286329633063316332633363346335633663376338633963406341634263436344634563466347634863496350635163526353635463556356635763586359636063616362636363646365636663676368636963706371637263736374637563766377637863796380638163826383638463856386638763886389639063916392639363946395639663976398639964006401640264036404640564066407640864096410641164126413641464156416641764186419642064216422642364246425642664276428642964306431643264336434643564366437643864396440644164426443644464456446644764486449645064516452645364546455645664576458645964606461646264636464646564666467646864696470647164726473647464756476647764786479648064816482648364846485648664876488648964906491649264936494649564966497649864996500650165026503650465056506650765086509651065116512651365146515651665176518651965206521652265236524652565266527652865296530653165326533653465356536653765386539654065416542654365446545654665476548654965506551655265536554655565566557655865596560656165626563656465656566656765686569657065716572657365746575657665776578657965806581658265836584658565866587658865896590659165926593659465956596659765986599660066016602660366046605660666076608660966106611661266136614661566166617661866196620662166226623662466256626662766286629663066316632663366346635663666376638663966406641664266436644664566466647664866496650665166526653665466556656665766586659666066616662666366646665666666676668666966706671667266736674667566766677667866796680668166826683668466856686668766886689669066916692669366946695669666976698669967006701670267036704670567066707670867096710671167126713671467156716671767186719672067216722672367246725672667276728672967306731673267336734673567366737673867396740674167426743674467456746674767486749675067516752675367546755675667576758675967606761676267636764676567666767676867696770677167726773677467756776677767786779678067816782678367846785678667876788678967906791679267936794679567966797679867996800680168026803680468056806680768086809681068116812681368146815681668176818681968206821682268236824682568266827682868296830683168326833683468356836683768386839684068416842684368446845684668476848684968506851685268536854685568566857685868596860686168626863686468656866686768686869687068716872687368746875687668776878687968806881688268836884688568866887688868896890689168926893689468956896689768986899690069016902690369046905690669076908690969106911691269136914691569166917691869196920692169226923692469256926692769286929693069316932693369346935693669376938693969406941694269436944694569466947694869496950695169526953695469556956695769586959696069616962696369646965696669676968696969706971697269736974697569766977697869796980698169826983698469856986698769886989699069916992699369946995699669976998699970007001700270037004700570067007700870097010701170127013701470157016701770187019702070217022702370247025702670277028702970307031703270337034703570367037703870397040704170427043704470457046704770487049705070517052705370547055705670577058705970607061706270637064706570667067706870697070707170727073707470757076707770787079708070817082708370847085708670877088708970907091709270937094709570967097709870997100710171027103710471057106710771087109711071117112711371147115711671177118711971207121712271237124712571267127712871297130713171327133713471357136713771387139714071417142714371447145714671477148714971507151715271537154715571567157715871597160716171627163716471657166716771687169717071717172717371747175717671777178717971807181718271837184718571867187718871897190719171927193719471957196719771987199720072017202720372047205720672077208720972107211721272137214721572167217721872197220722172227223722472257226722772287229723072317232723372347235723672377238723972407241724272437244724572467247724872497250725172527253725472557256725772587259726072617262726372647265726672677268726972707271727272737274727572767277727872797280728172827283728472857286728772887289729072917292729372947295729672977298729973007301730273037304730573067307730873097310731173127313731473157316731773187319732073217322732373247325732673277328732973307331733273337334733573367337733873397340734173427343734473457346734773487349735073517352735373547355735673577358735973607361736273637364736573667367736873697370737173727373737473757376737773787379738073817382738373847385738673877388738973907391739273937394739573967397739873997400740174027403740474057406740774087409741074117412741374147415741674177418741974207421742274237424742574267427742874297430743174327433743474357436743774387439744074417442744374447445744674477448744974507451745274537454745574567457745874597460746174627463746474657466746774687469747074717472747374747475747674777478747974807481748274837484748574867487748874897490749174927493749474957496749774987499750075017502750375047505750675077508750975107511751275137514751575167517751875197520752175227523752475257526752775287529753075317532753375347535753675377538753975407541754275437544754575467547754875497550755175527553755475557556755775587559756075617562756375647565756675677568756975707571757275737574757575767577 |
- diff -Nuar ostree-2018.8.orig/bsdiff/autogen.sh ostree-2018.8/bsdiff/autogen.sh
- --- ostree-2018.8.orig/bsdiff/autogen.sh 1970-01-01 02:00:00.000000000 +0200
- +++ ostree-2018.8/bsdiff/autogen.sh 2018-05-26 00:37:31.985018846 +0300
- @@ -0,0 +1,6 @@
- +#!/bin/sh
- +
- +touch AUTHORS NEWS README ChangeLog
- +cp LICENSE COPYING
- +
- +autoreconf -fis
- diff -Nuar ostree-2018.8.orig/bsdiff/bsdiff.c ostree-2018.8/bsdiff/bsdiff.c
- --- ostree-2018.8.orig/bsdiff/bsdiff.c 1970-01-01 02:00:00.000000000 +0200
- +++ ostree-2018.8/bsdiff/bsdiff.c 2018-05-26 00:37:31.985018846 +0300
- @@ -0,0 +1,445 @@
- +/*-
- + * Copyright 2003-2005 Colin Percival
- + * Copyright 2012 Matthew Endsley
- + * All rights reserved
- + *
- + * Redistribution and use in source and binary forms, with or without
- + * modification, are permitted providing that the following conditions
- + * are met:
- + * 1. Redistributions of source code must retain the above copyright
- + * notice, this list of conditions and the following disclaimer.
- + * 2. Redistributions in binary form must reproduce the above copyright
- + * notice, this list of conditions and the following disclaimer in the
- + * documentation and/or other materials provided with the distribution.
- + *
- + * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
- + * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
- + * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
- + * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY
- + * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
- + * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
- + * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
- + * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
- + * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING
- + * IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
- + * POSSIBILITY OF SUCH DAMAGE.
- + */
- +
- +#include "bsdiff.h"
- +
- +#include <limits.h>
- +#include <string.h>
- +
- +#define MIN(x,y) (((x)<(y)) ? (x) : (y))
- +
- +static void split(int64_t *I,int64_t *V,int64_t start,int64_t len,int64_t h)
- +{
- + int64_t i,j,k,x,tmp,jj,kk;
- +
- + if(len<16) {
- + for(k=start;k<start+len;k+=j) {
- + j=1;x=V[I[k]+h];
- + for(i=1;k+i<start+len;i++) {
- + if(V[I[k+i]+h]<x) {
- + x=V[I[k+i]+h];
- + j=0;
- + };
- + if(V[I[k+i]+h]==x) {
- + tmp=I[k+j];I[k+j]=I[k+i];I[k+i]=tmp;
- + j++;
- + };
- + };
- + for(i=0;i<j;i++) V[I[k+i]]=k+j-1;
- + if(j==1) I[k]=-1;
- + };
- + return;
- + };
- +
- + x=V[I[start+len/2]+h];
- + jj=0;kk=0;
- + for(i=start;i<start+len;i++) {
- + if(V[I[i]+h]<x) jj++;
- + if(V[I[i]+h]==x) kk++;
- + };
- + jj+=start;kk+=jj;
- +
- + i=start;j=0;k=0;
- + while(i<jj) {
- + if(V[I[i]+h]<x) {
- + i++;
- + } else if(V[I[i]+h]==x) {
- + tmp=I[i];I[i]=I[jj+j];I[jj+j]=tmp;
- + j++;
- + } else {
- + tmp=I[i];I[i]=I[kk+k];I[kk+k]=tmp;
- + k++;
- + };
- + };
- +
- + while(jj+j<kk) {
- + if(V[I[jj+j]+h]==x) {
- + j++;
- + } else {
- + tmp=I[jj+j];I[jj+j]=I[kk+k];I[kk+k]=tmp;
- + k++;
- + };
- + };
- +
- + if(jj>start) split(I,V,start,jj-start,h);
- +
- + for(i=0;i<kk-jj;i++) V[I[jj+i]]=kk-1;
- + if(jj==kk-1) I[jj]=-1;
- +
- + if(start+len>kk) split(I,V,kk,start+len-kk,h);
- +}
- +
- +static void qsufsort(int64_t *I,int64_t *V,const uint8_t *old,int64_t oldsize)
- +{
- + int64_t buckets[256];
- + int64_t i,h,len;
- +
- + for(i=0;i<256;i++) buckets[i]=0;
- + for(i=0;i<oldsize;i++) buckets[old[i]]++;
- + for(i=1;i<256;i++) buckets[i]+=buckets[i-1];
- + for(i=255;i>0;i--) buckets[i]=buckets[i-1];
- + buckets[0]=0;
- +
- + for(i=0;i<oldsize;i++) I[++buckets[old[i]]]=i;
- + I[0]=oldsize;
- + for(i=0;i<oldsize;i++) V[i]=buckets[old[i]];
- + V[oldsize]=0;
- + for(i=1;i<256;i++) if(buckets[i]==buckets[i-1]+1) I[buckets[i]]=-1;
- + I[0]=-1;
- +
- + for(h=1;I[0]!=-(oldsize+1);h+=h) {
- + len=0;
- + for(i=0;i<oldsize+1;) {
- + if(I[i]<0) {
- + len-=I[i];
- + i-=I[i];
- + } else {
- + if(len) I[i-len]=-len;
- + len=V[I[i]]+1-i;
- + split(I,V,i,len,h);
- + i+=len;
- + len=0;
- + };
- + };
- + if(len) I[i-len]=-len;
- + };
- +
- + for(i=0;i<oldsize+1;i++) I[V[i]]=i;
- +}
- +
- +static int64_t matchlen(const uint8_t *old,int64_t oldsize,const uint8_t *new,int64_t newsize)
- +{
- + int64_t i;
- +
- + for(i=0;(i<oldsize)&&(i<newsize);i++)
- + if(old[i]!=new[i]) break;
- +
- + return i;
- +}
- +
- +static int64_t search(const int64_t *I,const uint8_t *old,int64_t oldsize,
- + const uint8_t *new,int64_t newsize,int64_t st,int64_t en,int64_t *pos)
- +{
- + int64_t x,y;
- +
- + if(en-st<2) {
- + x=matchlen(old+I[st],oldsize-I[st],new,newsize);
- + y=matchlen(old+I[en],oldsize-I[en],new,newsize);
- +
- + if(x>y) {
- + *pos=I[st];
- + return x;
- + } else {
- + *pos=I[en];
- + return y;
- + }
- + };
- +
- + x=st+(en-st)/2;
- + if(memcmp(old+I[x],new,MIN(oldsize-I[x],newsize))<0) {
- + return search(I,old,oldsize,new,newsize,x,en,pos);
- + } else {
- + return search(I,old,oldsize,new,newsize,st,x,pos);
- + };
- +}
- +
- +static void offtout(int64_t x,uint8_t *buf)
- +{
- + int64_t y;
- +
- + if(x<0) y=-x; else y=x;
- +
- + buf[0]=y%256;y-=buf[0];
- + y=y/256;buf[1]=y%256;y-=buf[1];
- + y=y/256;buf[2]=y%256;y-=buf[2];
- + y=y/256;buf[3]=y%256;y-=buf[3];
- + y=y/256;buf[4]=y%256;y-=buf[4];
- + y=y/256;buf[5]=y%256;y-=buf[5];
- + y=y/256;buf[6]=y%256;y-=buf[6];
- + y=y/256;buf[7]=y%256;
- +
- + if(x<0) buf[7]|=0x80;
- +}
- +
- +static int64_t writedata(struct bsdiff_stream* stream, const void* buffer, int64_t length)
- +{
- + int64_t result = 0;
- +
- + while (length > 0)
- + {
- + const int smallsize = (int)MIN(length, INT_MAX);
- + const int writeresult = stream->write(stream, buffer, smallsize);
- + if (writeresult == -1)
- + {
- + return -1;
- + }
- +
- + result += writeresult;
- + length -= smallsize;
- + buffer = (uint8_t*)buffer + smallsize;
- + }
- +
- + return result;
- +}
- +
- +struct bsdiff_request
- +{
- + const uint8_t* old;
- + int64_t oldsize;
- + const uint8_t* new;
- + int64_t newsize;
- + struct bsdiff_stream* stream;
- + int64_t *I;
- + uint8_t *buffer;
- +};
- +
- +static int bsdiff_internal(const struct bsdiff_request req)
- +{
- + int64_t *I,*V;
- + int64_t scan,pos,len;
- + int64_t lastscan,lastpos,lastoffset;
- + int64_t oldscore,scsc;
- + int64_t s,Sf,lenf,Sb,lenb;
- + int64_t overlap,Ss,lens;
- + int64_t i;
- + uint8_t *buffer;
- + uint8_t buf[8 * 3];
- +
- + if((V=req.stream->malloc((req.oldsize+1)*sizeof(int64_t)))==NULL) return -1;
- + I = req.I;
- +
- + qsufsort(I,V,req.old,req.oldsize);
- + req.stream->free(V);
- +
- + buffer = req.buffer;
- +
- + /* Compute the differences, writing ctrl as we go */
- + scan=0;len=0;pos=0;
- + lastscan=0;lastpos=0;lastoffset=0;
- + while(scan<req.newsize) {
- + oldscore=0;
- +
- + for(scsc=scan+=len;scan<req.newsize;scan++) {
- + len=search(I,req.old,req.oldsize,req.new+scan,req.newsize-scan,
- + 0,req.oldsize,&pos);
- +
- + for(;scsc<scan+len;scsc++)
- + if((scsc+lastoffset<req.oldsize) &&
- + (req.old[scsc+lastoffset] == req.new[scsc]))
- + oldscore++;
- +
- + if(((len==oldscore) && (len!=0)) ||
- + (len>oldscore+8)) break;
- +
- + if((scan+lastoffset<req.oldsize) &&
- + (req.old[scan+lastoffset] == req.new[scan]))
- + oldscore--;
- + };
- +
- + if((len!=oldscore) || (scan==req.newsize)) {
- + s=0;Sf=0;lenf=0;
- + for(i=0;(lastscan+i<scan)&&(lastpos+i<req.oldsize);) {
- + if(req.old[lastpos+i]==req.new[lastscan+i]) s++;
- + i++;
- + if(s*2-i>Sf*2-lenf) { Sf=s; lenf=i; };
- + };
- +
- + lenb=0;
- + if(scan<req.newsize) {
- + s=0;Sb=0;
- + for(i=1;(scan>=lastscan+i)&&(pos>=i);i++) {
- + if(req.old[pos-i]==req.new[scan-i]) s++;
- + if(s*2-i>Sb*2-lenb) { Sb=s; lenb=i; };
- + };
- + };
- +
- + if(lastscan+lenf>scan-lenb) {
- + overlap=(lastscan+lenf)-(scan-lenb);
- + s=0;Ss=0;lens=0;
- + for(i=0;i<overlap;i++) {
- + if(req.new[lastscan+lenf-overlap+i]==
- + req.old[lastpos+lenf-overlap+i]) s++;
- + if(req.new[scan-lenb+i]==
- + req.old[pos-lenb+i]) s--;
- + if(s>Ss) { Ss=s; lens=i+1; };
- + };
- +
- + lenf+=lens-overlap;
- + lenb-=lens;
- + };
- +
- + offtout(lenf,buf);
- + offtout((scan-lenb)-(lastscan+lenf),buf+8);
- + offtout((pos-lenb)-(lastpos+lenf),buf+16);
- +
- + /* Write control data */
- + if (writedata(req.stream, buf, sizeof(buf)))
- + return -1;
- +
- + /* Write diff data */
- + for(i=0;i<lenf;i++)
- + buffer[i]=req.new[lastscan+i]-req.old[lastpos+i];
- + if (writedata(req.stream, buffer, lenf))
- + return -1;
- +
- + /* Write extra data */
- + for(i=0;i<(scan-lenb)-(lastscan+lenf);i++)
- + buffer[i]=req.new[lastscan+lenf+i];
- + if (writedata(req.stream, buffer, (scan-lenb)-(lastscan+lenf)))
- + return -1;
- +
- + lastscan=scan-lenb;
- + lastpos=pos-lenb;
- + lastoffset=pos-scan;
- + };
- + };
- +
- + return 0;
- +}
- +
- +int bsdiff(const uint8_t* old, int64_t oldsize, const uint8_t* new, int64_t newsize, struct bsdiff_stream* stream)
- +{
- + int result;
- + struct bsdiff_request req;
- +
- + if((req.I=stream->malloc((oldsize+1)*sizeof(int64_t)))==NULL)
- + return -1;
- +
- + if((req.buffer=stream->malloc(newsize+1))==NULL)
- + {
- + stream->free(req.I);
- + return -1;
- + }
- +
- + req.old = old;
- + req.oldsize = oldsize;
- + req.new = new;
- + req.newsize = newsize;
- + req.stream = stream;
- +
- + result = bsdiff_internal(req);
- +
- + stream->free(req.buffer);
- + stream->free(req.I);
- +
- + return result;
- +}
- +
- +#if defined(BSDIFF_EXECUTABLE)
- +
- +#include <sys/types.h>
- +
- +#include <bzlib.h>
- +#include <err.h>
- +#include <fcntl.h>
- +#include <stdio.h>
- +#include <stdlib.h>
- +#include <unistd.h>
- +
- +static int bz2_write(struct bsdiff_stream* stream, const void* buffer, int size)
- +{
- + int bz2err;
- + BZFILE* bz2;
- +
- + bz2 = (BZFILE*)stream->opaque;
- + BZ2_bzWrite(&bz2err, bz2, (void*)buffer, size);
- + if (bz2err != BZ_STREAM_END && bz2err != BZ_OK)
- + return -1;
- +
- + return 0;
- +}
- +
- +int main(int argc,char *argv[])
- +{
- + int fd;
- + int bz2err;
- + uint8_t *old,*new;
- + off_t oldsize,newsize;
- + uint8_t buf[8];
- + FILE * pf;
- + struct bsdiff_stream stream;
- + BZFILE* bz2;
- +
- + memset(&bz2, 0, sizeof(bz2));
- + stream.malloc = malloc;
- + stream.free = free;
- + stream.write = bz2_write;
- +
- + if(argc!=4) errx(1,"usage: %s oldfile newfile patchfile\n",argv[0]);
- +
- + /* Allocate oldsize+1 bytes instead of oldsize bytes to ensure
- + that we never try to malloc(0) and get a NULL pointer */
- + if(((fd=open(argv[1],O_RDONLY,0))<0) ||
- + ((oldsize=lseek(fd,0,SEEK_END))==-1) ||
- + ((old=malloc(oldsize+1))==NULL) ||
- + (lseek(fd,0,SEEK_SET)!=0) ||
- + (read(fd,old,oldsize)!=oldsize) ||
- + (close(fd)==-1)) err(1,"%s",argv[1]);
- +
- +
- + /* Allocate newsize+1 bytes instead of newsize bytes to ensure
- + that we never try to malloc(0) and get a NULL pointer */
- + if(((fd=open(argv[2],O_RDONLY,0))<0) ||
- + ((newsize=lseek(fd,0,SEEK_END))==-1) ||
- + ((new=malloc(newsize+1))==NULL) ||
- + (lseek(fd,0,SEEK_SET)!=0) ||
- + (read(fd,new,newsize)!=newsize) ||
- + (close(fd)==-1)) err(1,"%s",argv[2]);
- +
- + /* Create the patch file */
- + if ((pf = fopen(argv[3], "w")) == NULL)
- + err(1, "%s", argv[3]);
- +
- + /* Write header (signature+newsize)*/
- + offtout(newsize, buf);
- + if (fwrite("ENDSLEY/BSDIFF43", 16, 1, pf) != 1 ||
- + fwrite(buf, sizeof(buf), 1, pf) != 1)
- + err(1, "Failed to write header");
- +
- +
- + if (NULL == (bz2 = BZ2_bzWriteOpen(&bz2err, pf, 9, 0, 0)))
- + errx(1, "BZ2_bzWriteOpen, bz2err=%d", bz2err);
- +
- + stream.opaque = bz2;
- + if (bsdiff(old, oldsize, new, newsize, &stream))
- + err(1, "bsdiff");
- +
- + BZ2_bzWriteClose(&bz2err, bz2, 0, NULL, NULL);
- + if (bz2err != BZ_OK)
- + err(1, "BZ2_bzWriteClose, bz2err=%d", bz2err);
- +
- + if (fclose(pf))
- + err(1, "fclose");
- +
- + /* Free the memory we used */
- + free(old);
- + free(new);
- +
- + return 0;
- +}
- +
- +#endif
- diff -Nuar ostree-2018.8.orig/bsdiff/bsdiff.h ostree-2018.8/bsdiff/bsdiff.h
- --- ostree-2018.8.orig/bsdiff/bsdiff.h 1970-01-01 02:00:00.000000000 +0200
- +++ ostree-2018.8/bsdiff/bsdiff.h 2018-05-26 00:37:31.985018846 +0300
- @@ -0,0 +1,45 @@
- +/*-
- + * Copyright 2003-2005 Colin Percival
- + * Copyright 2012 Matthew Endsley
- + * All rights reserved
- + *
- + * Redistribution and use in source and binary forms, with or without
- + * modification, are permitted providing that the following conditions
- + * are met:
- + * 1. Redistributions of source code must retain the above copyright
- + * notice, this list of conditions and the following disclaimer.
- + * 2. Redistributions in binary form must reproduce the above copyright
- + * notice, this list of conditions and the following disclaimer in the
- + * documentation and/or other materials provided with the distribution.
- + *
- + * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
- + * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
- + * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
- + * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY
- + * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
- + * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
- + * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
- + * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
- + * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING
- + * IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
- + * POSSIBILITY OF SUCH DAMAGE.
- + */
- +
- +#ifndef BSDIFF_H
- +# define BSDIFF_H
- +
- +# include <stddef.h>
- +# include <stdint.h>
- +
- +struct bsdiff_stream
- +{
- + void* opaque;
- +
- + void* (*malloc)(size_t size);
- + void (*free)(void* ptr);
- + int (*write)(struct bsdiff_stream* stream, const void* buffer, int size);
- +};
- +
- +int bsdiff(const uint8_t* old, int64_t oldsize, const uint8_t* new, int64_t newsize, struct bsdiff_stream* stream);
- +
- +#endif
- diff -Nuar ostree-2018.8.orig/bsdiff/bspatch.c ostree-2018.8/bsdiff/bspatch.c
- --- ostree-2018.8.orig/bsdiff/bspatch.c 1970-01-01 02:00:00.000000000 +0200
- +++ ostree-2018.8/bsdiff/bspatch.c 2018-05-26 00:37:31.985018846 +0300
- @@ -0,0 +1,187 @@
- +/*-
- + * Copyright 2003-2005 Colin Percival
- + * Copyright 2012 Matthew Endsley
- + * All rights reserved
- + *
- + * Redistribution and use in source and binary forms, with or without
- + * modification, are permitted providing that the following conditions
- + * are met:
- + * 1. Redistributions of source code must retain the above copyright
- + * notice, this list of conditions and the following disclaimer.
- + * 2. Redistributions in binary form must reproduce the above copyright
- + * notice, this list of conditions and the following disclaimer in the
- + * documentation and/or other materials provided with the distribution.
- + *
- + * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
- + * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
- + * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
- + * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY
- + * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
- + * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
- + * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
- + * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
- + * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING
- + * IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
- + * POSSIBILITY OF SUCH DAMAGE.
- + */
- +
- +#include "bspatch.h"
- +
- +static int64_t offtin(uint8_t *buf)
- +{
- + int64_t y;
- +
- + y=buf[7]&0x7F;
- + y=y*256;y+=buf[6];
- + y=y*256;y+=buf[5];
- + y=y*256;y+=buf[4];
- + y=y*256;y+=buf[3];
- + y=y*256;y+=buf[2];
- + y=y*256;y+=buf[1];
- + y=y*256;y+=buf[0];
- +
- + if(buf[7]&0x80) y=-y;
- +
- + return y;
- +}
- +
- +int bspatch(const uint8_t* old, int64_t oldsize, uint8_t* new, int64_t newsize, struct bspatch_stream* stream)
- +{
- + uint8_t buf[8];
- + int64_t oldpos,newpos;
- + int64_t ctrl[3];
- + int64_t i;
- +
- + oldpos=0;newpos=0;
- + while(newpos<newsize) {
- + /* Read control data */
- + for(i=0;i<=2;i++) {
- + if (stream->read(stream, buf, 8))
- + return -1;
- + ctrl[i]=offtin(buf);
- + };
- +
- + /* Sanity-check */
- + if(newpos+ctrl[0]>newsize)
- + return -1;
- +
- + /* Read diff string */
- + if (stream->read(stream, new + newpos, ctrl[0]))
- + return -1;
- +
- + /* Add old data to diff string */
- + for(i=0;i<ctrl[0];i++)
- + if((oldpos+i>=0) && (oldpos+i<oldsize))
- + new[newpos+i]+=old[oldpos+i];
- +
- + /* Adjust pointers */
- + newpos+=ctrl[0];
- + oldpos+=ctrl[0];
- +
- + /* Sanity-check */
- + if(newpos+ctrl[1]>newsize)
- + return -1;
- +
- + /* Read extra string */
- + if (stream->read(stream, new + newpos, ctrl[1]))
- + return -1;
- +
- + /* Adjust pointers */
- + newpos+=ctrl[1];
- + oldpos+=ctrl[2];
- + };
- +
- + return 0;
- +}
- +
- +#if defined(BSPATCH_EXECUTABLE)
- +
- +#include <bzlib.h>
- +#include <stdlib.h>
- +#include <stdint.h>
- +#include <stdio.h>
- +#include <string.h>
- +#include <err.h>
- +#include <unistd.h>
- +#include <fcntl.h>
- +
- +static int bz2_read(const struct bspatch_stream* stream, void* buffer, int length)
- +{
- + int n;
- + int bz2err;
- + BZFILE* bz2;
- +
- + bz2 = (BZFILE*)stream->opaque;
- + n = BZ2_bzRead(&bz2err, bz2, buffer, length);
- + if (n != length)
- + return -1;
- +
- + return 0;
- +}
- +
- +int main(int argc,char * argv[])
- +{
- + FILE * f;
- + int fd;
- + int bz2err;
- + uint8_t header[24];
- + uint8_t *old, *new;
- + int64_t oldsize, newsize;
- + BZFILE* bz2;
- + struct bspatch_stream stream;
- +
- + if(argc!=4) errx(1,"usage: %s oldfile newfile patchfile\n",argv[0]);
- +
- + /* Open patch file */
- + if ((f = fopen(argv[3], "r")) == NULL)
- + err(1, "fopen(%s)", argv[3]);
- +
- + /* Read header */
- + if (fread(header, 1, 24, f) != 24) {
- + if (feof(f))
- + errx(1, "Corrupt patch\n");
- + err(1, "fread(%s)", argv[3]);
- + }
- +
- + /* Check for appropriate magic */
- + if (memcmp(header, "ENDSLEY/BSDIFF43", 16) != 0)
- + errx(1, "Corrupt patch\n");
- +
- + /* Read lengths from header */
- + newsize=offtin(header+16);
- + if(newsize<0)
- + errx(1,"Corrupt patch\n");
- +
- + /* Close patch file and re-open it via libbzip2 at the right places */
- + if(((fd=open(argv[1],O_RDONLY,0))<0) ||
- + ((oldsize=lseek(fd,0,SEEK_END))==-1) ||
- + ((old=malloc(oldsize+1))==NULL) ||
- + (lseek(fd,0,SEEK_SET)!=0) ||
- + (read(fd,old,oldsize)!=oldsize) ||
- + (close(fd)==-1)) err(1,"%s",argv[1]);
- + if((new=malloc(newsize+1))==NULL) err(1,NULL);
- +
- + if (NULL == (bz2 = BZ2_bzReadOpen(&bz2err, f, 0, 0, NULL, 0)))
- + errx(1, "BZ2_bzReadOpen, bz2err=%d", bz2err);
- +
- + stream.read = bz2_read;
- + stream.opaque = bz2;
- + if (bspatch(old, oldsize, new, newsize, &stream))
- + errx(1, "bspatch");
- +
- + /* Clean up the bzip2 reads */
- + BZ2_bzReadClose(&bz2err, bz2);
- + fclose(f);
- +
- + /* Write the new file */
- + if(((fd=open(argv[2],O_CREAT|O_TRUNC|O_WRONLY,0666))<0) ||
- + (write(fd,new,newsize)!=newsize) || (close(fd)==-1))
- + err(1,"%s",argv[2]);
- +
- + free(new);
- + free(old);
- +
- + return 0;
- +}
- +
- +#endif
- diff -Nuar ostree-2018.8.orig/bsdiff/bspatch.h ostree-2018.8/bsdiff/bspatch.h
- --- ostree-2018.8.orig/bsdiff/bspatch.h 1970-01-01 02:00:00.000000000 +0200
- +++ ostree-2018.8/bsdiff/bspatch.h 2018-05-26 00:37:31.985018846 +0300
- @@ -0,0 +1,42 @@
- +/*-
- + * Copyright 2003-2005 Colin Percival
- + * Copyright 2012 Matthew Endsley
- + * All rights reserved
- + *
- + * Redistribution and use in source and binary forms, with or without
- + * modification, are permitted providing that the following conditions
- + * are met:
- + * 1. Redistributions of source code must retain the above copyright
- + * notice, this list of conditions and the following disclaimer.
- + * 2. Redistributions in binary form must reproduce the above copyright
- + * notice, this list of conditions and the following disclaimer in the
- + * documentation and/or other materials provided with the distribution.
- + *
- + * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
- + * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
- + * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
- + * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY
- + * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
- + * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
- + * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
- + * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
- + * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING
- + * IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
- + * POSSIBILITY OF SUCH DAMAGE.
- + */
- +
- +#ifndef BSPATCH_H
- +# define BSPATCH_H
- +
- +# include <stdint.h>
- +
- +struct bspatch_stream
- +{
- + void* opaque;
- + int (*read)(const struct bspatch_stream* stream, void* buffer, int length);
- +};
- +
- +int bspatch(const uint8_t* old, int64_t oldsize, uint8_t* new, int64_t newsize, struct bspatch_stream* stream);
- +
- +#endif
- +
- diff -Nuar ostree-2018.8.orig/bsdiff/configure.ac ostree-2018.8/bsdiff/configure.ac
- --- ostree-2018.8.orig/bsdiff/configure.ac 1970-01-01 02:00:00.000000000 +0200
- +++ ostree-2018.8/bsdiff/configure.ac 2018-05-26 00:37:31.985018846 +0300
- @@ -0,0 +1,30 @@
- +# -*- Autoconf -*-
- +# Process this file with autoconf to produce a configure script.
- +
- +AC_PREREQ([2.69])
- +AC_INIT([bsdiff], [0.1])
- +AC_CONFIG_SRCDIR([bsdiff.c])
- +AC_CONFIG_HEADERS([config.h])
- +AM_INIT_AUTOMAKE([1.9])
- +
- +# Checks for programs.
- +AC_PROG_CC
- +
- +# Checks for libraries.
- +# FIXME: Replace `main' with a function in `-lbz2':
- +AC_CHECK_LIB([bz2], [BZ2_bzReadOpen])
- +
- +AC_CHECK_HEADERS([fcntl.h limits.h stddef.h stdint.h stdlib.h string.h unistd.h])
- +
- +# Checks for typedefs, structures, and compiler characteristics.
- +AC_TYPE_INT64_T
- +AC_TYPE_OFF_T
- +AC_TYPE_SIZE_T
- +AC_TYPE_UINT8_T
- +
- +# Checks for library functions.
- +AC_FUNC_MALLOC
- +AC_CHECK_FUNCS([memset])
- +
- +AC_CONFIG_FILES([Makefile])
- +AC_OUTPUT
- diff -Nuar ostree-2018.8.orig/bsdiff/.git ostree-2018.8/bsdiff/.git
- --- ostree-2018.8.orig/bsdiff/.git 1970-01-01 02:00:00.000000000 +0200
- +++ ostree-2018.8/bsdiff/.git 2018-05-26 00:37:14.345018427 +0300
- @@ -0,0 +1 @@
- +gitdir: ../.git/modules/bsdiff
- diff -Nuar ostree-2018.8.orig/bsdiff/.gitignore ostree-2018.8/bsdiff/.gitignore
- --- ostree-2018.8.orig/bsdiff/.gitignore 1970-01-01 02:00:00.000000000 +0200
- +++ ostree-2018.8/bsdiff/.gitignore 2018-05-26 00:37:31.984018846 +0300
- @@ -0,0 +1,11 @@
- +.deps/*
- +.libs/*
- +*.lo
- +*.o
- +.dirstamp
- +Makefile-bsdiff.am.inc
- +AUTHORS
- +NEWS
- +README
- +ChangeLog
- +COPYING
- diff -Nuar ostree-2018.8.orig/bsdiff/LICENSE ostree-2018.8/bsdiff/LICENSE
- --- ostree-2018.8.orig/bsdiff/LICENSE 1970-01-01 02:00:00.000000000 +0200
- +++ ostree-2018.8/bsdiff/LICENSE 2018-05-26 00:37:31.984018846 +0300
- @@ -0,0 +1,24 @@
- + Copyright 2003-2005 Colin Percival
- + Copyright 2012 Matthew Endsley
- + All rights reserved
- +
- + Redistribution and use in source and binary forms, with or without
- + modification, are permitted providing that the following conditions
- + are met:
- + 1. Redistributions of source code must retain the above copyright
- + notice, this list of conditions and the following disclaimer.
- + 2. Redistributions in binary form must reproduce the above copyright
- + notice, this list of conditions and the following disclaimer in the
- + documentation and/or other materials provided with the distribution.
- +
- + THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
- + IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
- + WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
- + ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY
- + DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
- + DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
- + OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
- + HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
- + STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING
- + IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
- + POSSIBILITY OF SUCH DAMAGE.
- diff -Nuar ostree-2018.8.orig/bsdiff/Makefile.am ostree-2018.8/bsdiff/Makefile.am
- --- ostree-2018.8.orig/bsdiff/Makefile.am 1970-01-01 02:00:00.000000000 +0200
- +++ ostree-2018.8/bsdiff/Makefile.am 2018-05-26 00:37:31.984018846 +0300
- @@ -0,0 +1,11 @@
- +bin_PROGRAMS = bsdiff bspatch
- +
- +bsdiff_SOURCES = bsdiff.c
- +
- +bspatch_SOURCES = bspatch.c
- +
- +bsdiff_CFLAGS = -DBSDIFF_EXECUTABLE
- +bspatch_CFLAGS = -DBSPATCH_EXECUTABLE
- +
- +EXTRA_DIST = bsdiff.h bspatch.h
- +
- diff -Nuar ostree-2018.8.orig/bsdiff/Makefile-bsdiff.am ostree-2018.8/bsdiff/Makefile-bsdiff.am
- --- ostree-2018.8.orig/bsdiff/Makefile-bsdiff.am 1970-01-01 02:00:00.000000000 +0200
- +++ ostree-2018.8/bsdiff/Makefile-bsdiff.am 2018-05-26 00:37:31.984018846 +0300
- @@ -0,0 +1,32 @@
- +# Copyright (C) 2015 Giuseppe Scrivano <gscrivan@redhat.com>
- +#
- +# Redistribution and use in source and binary forms, with or without
- +# modification, are permitted providing that the following conditions
- +# are met:
- +# 1. Redistributions of source code must retain the above copyright
- +# notice, this list of conditions and the following disclaimer.
- +# 2. Redistributions in binary form must reproduce the above copyright
- +# notice, this list of conditions and the following disclaimer in the
- +# documentation and/or other materials provided with the distribution.
- +#
- +# THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
- +# IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
- +# WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
- +# ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY
- +# DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
- +# DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
- +# OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
- +# HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
- +# STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING
- +# IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
- +# POSSIBILITY OF SUCH DAMAGE.
- +#
- +
- +EXTRA_DIST += $(libbsdiff_srcpath)/bsdiff.h $(libbsdiff_srcpath)/bspatch.h $(libbsdiff_srcpath)/LICENSE $(libbsdiff_srcpath)/README.md
- +
- +libbsdiff_la_SOURCES = \
- + $(libbsdiff_srcpath)/bsdiff.c \
- + $(libbsdiff_srcpath)/bspatch.c \
- + $(NULL)
- +
- +libbsdiff_la_CFLAGS = $(AM_CFLAGS)
- diff -Nuar ostree-2018.8.orig/bsdiff/README.md ostree-2018.8/bsdiff/README.md
- --- ostree-2018.8.orig/bsdiff/README.md 1970-01-01 02:00:00.000000000 +0200
- +++ ostree-2018.8/bsdiff/README.md 2018-05-26 00:37:31.985018846 +0300
- @@ -0,0 +1,129 @@
- +bsdiff/bspatch
- +==============
- +bsdiff and bspatch are libraries for building and applying patches to binary
- +files.
- +
- +The original algorithm and implementation was developed by Colin Percival. The
- +algorithm is detailed in his (unpublished) paper, [Naïve Differences of Executable Code](http://www.daemonology.net/papers/bsdiff.pdf). For more information, visit his
- +website at <http://www.daemonology.net/bsdiff/>.
- +
- +I maintain this project seperately from Colin's work, with the goal of making
- +the core functionality easily embedable in existing projects.
- +
- +Contact
- +-------
- +[@MatthewEndsley](https://twitter.com/#!/MatthewEndsley)
- +<https://github.com/mendsley/bsdiff>
- +
- +License
- +-------
- +Copyright 2003-2005 Colin Percival
- +Copyright 2012 Matthew Endsley
- +
- +This project is governed by the BSD 2-clause license. For details see the file
- +titled LICENSE in the project root folder.
- +
- +Overview
- +--------
- +There are two separate libraries in the project, bsdiff and bspatch. Each are
- +self contained in bsdiff.c and bspatch.c The easiest way to integrate is to
- +simply copy the c file to your source folder and build it.
- +
- +The overarching goal was to modify the original bsdiff/bspatch code from Colin
- +and eliminate external dependencies and provide a simple interface to the core
- +functionality.
- +
- +I've exposed relevant functions via the `_stream` classes. The only external
- +dependency not exposed is `memcmp` in `bsdiff`.
- +
- +This library generates patches that are not compatible with the original bsdiff
- +tool. The impompatibilities were motivated by the patching needs for the game
- +AirMech <https://www.carbongames.com> and the following requirements:
- +
- +* Eliminate/minimize any seek operations when applying patches
- +* Eliminate any required disk I/O and support embedded streams
- +* Ability to easily embed the routines as a library instead of an external binary
- +* Compile+run on all platforms we use to build the game (Windows, Linux, NaCl, OSX)
- +
- +Compiling
- +---------
- +The libraries should compile warning free in any moderately recent version of
- +gcc. The project uses `<stdint.h>` which is technically a C99 file and not
- +available in Microsoft Visual Studio. The easiest solution here is to use the
- +msinttypes version of stdint.h from <https://code.google.com/p/msinttypes/>.
- +The direct link for the lazy people is:
- +<https://msinttypes.googlecode.com/svn/trunk/stdint.h>.
- +
- +If your compiler does not provide an implementation of `<stdint.h>` you can
- +remove the header from the bsdiff/bspatch files and provide your own typedefs
- +for the following symbols: `uint8_t`, `uint64_t` and `int64_t`.
- +
- +Examples
- +--------
- +Each project has an optional main function that serves as an example for using
- +the library. Simply defined `BSDIFF_EXECUTABLE` or `BSPATCH_EXECUTABLE` to
- +enable building the standalone tools.
- +
- +Reference
- +---------
- +### bsdiff
- +
- + struct bsdiff_stream
- + {
- + void* opaque;
- + void* (*malloc)(size_t size);
- + void (*free)(void* ptr);
- + int (*write)(struct bsdiff_stream* stream,
- + const void* buffer, int size);
- + };
- +
- + int bsdiff(const uint8_t* old, int64_t oldsize, const uint8_t* new,
- + int64_t newsize, struct bsdiff_stream* stream);
- +
- +
- +In order to use `bsdiff`, you need to define functions for allocating memory and
- +writing binary data. This behavior is controlled by the `stream` parameted
- +passed to to `bsdiff(...)`.
- +
- +The `opaque` field is never read or modified from within the `bsdiff` function.
- +The caller can use this field to store custom state data needed for the callback
- +functions.
- +
- +The `malloc` and `free` members should point to functions that behave like the
- +standard `malloc` and `free` C functions.
- +
- +The `write` function is called by bsdiff to write a block of binary data to the
- +stream. The return value for `write` should be `0` on success and non-zero if
- +the callback failed to write all data. In the default example, bzip2 is used to
- +compress output data.
- +
- +`bsdiff` returns `0` on success and `-1` on failure.
- +
- +### bspatch
- +
- + struct bspatch_stream
- + {
- + void* opaque;
- + int (*read)(const struct bspatch_stream* stream,
- + void* buffer, int length);
- + };
- +
- + int bspatch(const uint8_t* old, int64_t oldsize, uint8_t* new,
- + int64_t newsize, struct bspatch_stream* stream);
- +
- +The `bspatch` function transforms the data for a file using data generated from
- +`bsdiff`. The caller takes care of loading the old file and allocating space for
- +new file data. The `stream` parameter controls the process for reading binary
- +patch data.
- +
- +The `opaque` field is never read or modified from within the bspatch function.
- +The caller can use this field to store custom state data needed for the read
- +function.
- +
- +The `read` function is called by `bspatch` to read a block of binary data from
- +the stream. The return value for `read` should be `0` on success and non-zero
- +if the callback failed to read the requested amount of data. In the default
- +example, bzip2 is used to decompress input data.
- +
- +`bspatch` returns `0` on success and `-1` on failure. On success, `new` contains
- +the data for the patched file.
- diff -Nuar ostree-2018.8.orig/libglnx/COPYING ostree-2018.8/libglnx/COPYING
- --- ostree-2018.8.orig/libglnx/COPYING 1970-01-01 02:00:00.000000000 +0200
- +++ ostree-2018.8/libglnx/COPYING 2018-05-26 00:37:32.006018846 +0300
- @@ -0,0 +1,502 @@
- + GNU LESSER GENERAL PUBLIC LICENSE
- + Version 2.1, February 1999
- +
- + Copyright (C) 1991, 1999 Free Software Foundation, Inc.
- + 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
- + Everyone is permitted to copy and distribute verbatim copies
- + of this license document, but changing it is not allowed.
- +
- +[This is the first released version of the Lesser GPL. It also counts
- + as the successor of the GNU Library Public License, version 2, hence
- + the version number 2.1.]
- +
- + Preamble
- +
- + The licenses for most software are designed to take away your
- +freedom to share and change it. By contrast, the GNU General Public
- +Licenses are intended to guarantee your freedom to share and change
- +free software--to make sure the software is free for all its users.
- +
- + This license, the Lesser General Public License, applies to some
- +specially designated software packages--typically libraries--of the
- +Free Software Foundation and other authors who decide to use it. You
- +can use it too, but we suggest you first think carefully about whether
- +this license or the ordinary General Public License is the better
- +strategy to use in any particular case, based on the explanations below.
- +
- + When we speak of free software, we are referring to freedom of use,
- +not price. Our General Public Licenses are designed to make sure that
- +you have the freedom to distribute copies of free software (and charge
- +for this service if you wish); that you receive source code or can get
- +it if you want it; that you can change the software and use pieces of
- +it in new free programs; and that you are informed that you can do
- +these things.
- +
- + To protect your rights, we need to make restrictions that forbid
- +distributors to deny you these rights or to ask you to surrender these
- +rights. These restrictions translate to certain responsibilities for
- +you if you distribute copies of the library or if you modify it.
- +
- + For example, if you distribute copies of the library, whether gratis
- +or for a fee, you must give the recipients all the rights that we gave
- +you. You must make sure that they, too, receive or can get the source
- +code. If you link other code with the library, you must provide
- +complete object files to the recipients, so that they can relink them
- +with the library after making changes to the library and recompiling
- +it. And you must show them these terms so they know their rights.
- +
- + We protect your rights with a two-step method: (1) we copyright the
- +library, and (2) we offer you this license, which gives you legal
- +permission to copy, distribute and/or modify the library.
- +
- + To protect each distributor, we want to make it very clear that
- +there is no warranty for the free library. Also, if the library is
- +modified by someone else and passed on, the recipients should know
- +that what they have is not the original version, so that the original
- +author's reputation will not be affected by problems that might be
- +introduced by others.
- +
- + Finally, software patents pose a constant threat to the existence of
- +any free program. We wish to make sure that a company cannot
- +effectively restrict the users of a free program by obtaining a
- +restrictive license from a patent holder. Therefore, we insist that
- +any patent license obtained for a version of the library must be
- +consistent with the full freedom of use specified in this license.
- +
- + Most GNU software, including some libraries, is covered by the
- +ordinary GNU General Public License. This license, the GNU Lesser
- +General Public License, applies to certain designated libraries, and
- +is quite different from the ordinary General Public License. We use
- +this license for certain libraries in order to permit linking those
- +libraries into non-free programs.
- +
- + When a program is linked with a library, whether statically or using
- +a shared library, the combination of the two is legally speaking a
- +combined work, a derivative of the original library. The ordinary
- +General Public License therefore permits such linking only if the
- +entire combination fits its criteria of freedom. The Lesser General
- +Public License permits more lax criteria for linking other code with
- +the library.
- +
- + We call this license the "Lesser" General Public License because it
- +does Less to protect the user's freedom than the ordinary General
- +Public License. It also provides other free software developers Less
- +of an advantage over competing non-free programs. These disadvantages
- +are the reason we use the ordinary General Public License for many
- +libraries. However, the Lesser license provides advantages in certain
- +special circumstances.
- +
- + For example, on rare occasions, there may be a special need to
- +encourage the widest possible use of a certain library, so that it becomes
- +a de-facto standard. To achieve this, non-free programs must be
- +allowed to use the library. A more frequent case is that a free
- +library does the same job as widely used non-free libraries. In this
- +case, there is little to gain by limiting the free library to free
- +software only, so we use the Lesser General Public License.
- +
- + In other cases, permission to use a particular library in non-free
- +programs enables a greater number of people to use a large body of
- +free software. For example, permission to use the GNU C Library in
- +non-free programs enables many more people to use the whole GNU
- +operating system, as well as its variant, the GNU/Linux operating
- +system.
- +
- + Although the Lesser General Public License is Less protective of the
- +users' freedom, it does ensure that the user of a program that is
- +linked with the Library has the freedom and the wherewithal to run
- +that program using a modified version of the Library.
- +
- + The precise terms and conditions for copying, distribution and
- +modification follow. Pay close attention to the difference between a
- +"work based on the library" and a "work that uses the library". The
- +former contains code derived from the library, whereas the latter must
- +be combined with the library in order to run.
- +
- + GNU LESSER GENERAL PUBLIC LICENSE
- + TERMS AND CONDITIONS FOR COPYING, DISTRIBUTION AND MODIFICATION
- +
- + 0. This License Agreement applies to any software library or other
- +program which contains a notice placed by the copyright holder or
- +other authorized party saying it may be distributed under the terms of
- +this Lesser General Public License (also called "this License").
- +Each licensee is addressed as "you".
- +
- + A "library" means a collection of software functions and/or data
- +prepared so as to be conveniently linked with application programs
- +(which use some of those functions and data) to form executables.
- +
- + The "Library", below, refers to any such software library or work
- +which has been distributed under these terms. A "work based on the
- +Library" means either the Library or any derivative work under
- +copyright law: that is to say, a work containing the Library or a
- +portion of it, either verbatim or with modifications and/or translated
- +straightforwardly into another language. (Hereinafter, translation is
- +included without limitation in the term "modification".)
- +
- + "Source code" for a work means the preferred form of the work for
- +making modifications to it. For a library, complete source code means
- +all the source code for all modules it contains, plus any associated
- +interface definition files, plus the scripts used to control compilation
- +and installation of the library.
- +
- + Activities other than copying, distribution and modification are not
- +covered by this License; they are outside its scope. The act of
- +running a program using the Library is not restricted, and output from
- +such a program is covered only if its contents constitute a work based
- +on the Library (independent of the use of the Library in a tool for
- +writing it). Whether that is true depends on what the Library does
- +and what the program that uses the Library does.
- +
- + 1. You may copy and distribute verbatim copies of the Library's
- +complete source code as you receive it, in any medium, provided that
- +you conspicuously and appropriately publish on each copy an
- +appropriate copyright notice and disclaimer of warranty; keep intact
- +all the notices that refer to this License and to the absence of any
- +warranty; and distribute a copy of this License along with the
- +Library.
- +
- + You may charge a fee for the physical act of transferring a copy,
- +and you may at your option offer warranty protection in exchange for a
- +fee.
- +
- + 2. You may modify your copy or copies of the Library or any portion
- +of it, thus forming a work based on the Library, and copy and
- +distribute such modifications or work under the terms of Section 1
- +above, provided that you also meet all of these conditions:
- +
- + a) The modified work must itself be a software library.
- +
- + b) You must cause the files modified to carry prominent notices
- + stating that you changed the files and the date of any change.
- +
- + c) You must cause the whole of the work to be licensed at no
- + charge to all third parties under the terms of this License.
- +
- + d) If a facility in the modified Library refers to a function or a
- + table of data to be supplied by an application program that uses
- + the facility, other than as an argument passed when the facility
- + is invoked, then you must make a good faith effort to ensure that,
- + in the event an application does not supply such function or
- + table, the facility still operates, and performs whatever part of
- + its purpose remains meaningful.
- +
- + (For example, a function in a library to compute square roots has
- + a purpose that is entirely well-defined independent of the
- + application. Therefore, Subsection 2d requires that any
- + application-supplied function or table used by this function must
- + be optional: if the application does not supply it, the square
- + root function must still compute square roots.)
- +
- +These requirements apply to the modified work as a whole. If
- +identifiable sections of that work are not derived from the Library,
- +and can be reasonably considered independent and separate works in
- +themselves, then this License, and its terms, do not apply to those
- +sections when you distribute them as separate works. But when you
- +distribute the same sections as part of a whole which is a work based
- +on the Library, the distribution of the whole must be on the terms of
- +this License, whose permissions for other licensees extend to the
- +entire whole, and thus to each and every part regardless of who wrote
- +it.
- +
- +Thus, it is not the intent of this section to claim rights or contest
- +your rights to work written entirely by you; rather, the intent is to
- +exercise the right to control the distribution of derivative or
- +collective works based on the Library.
- +
- +In addition, mere aggregation of another work not based on the Library
- +with the Library (or with a work based on the Library) on a volume of
- +a storage or distribution medium does not bring the other work under
- +the scope of this License.
- +
- + 3. You may opt to apply the terms of the ordinary GNU General Public
- +License instead of this License to a given copy of the Library. To do
- +this, you must alter all the notices that refer to this License, so
- +that they refer to the ordinary GNU General Public License, version 2,
- +instead of to this License. (If a newer version than version 2 of the
- +ordinary GNU General Public License has appeared, then you can specify
- +that version instead if you wish.) Do not make any other change in
- +these notices.
- +
- + Once this change is made in a given copy, it is irreversible for
- +that copy, so the ordinary GNU General Public License applies to all
- +subsequent copies and derivative works made from that copy.
- +
- + This option is useful when you wish to copy part of the code of
- +the Library into a program that is not a library.
- +
- + 4. You may copy and distribute the Library (or a portion or
- +derivative of it, under Section 2) in object code or executable form
- +under the terms of Sections 1 and 2 above provided that you accompany
- +it with the complete corresponding machine-readable source code, which
- +must be distributed under the terms of Sections 1 and 2 above on a
- +medium customarily used for software interchange.
- +
- + If distribution of object code is made by offering access to copy
- +from a designated place, then offering equivalent access to copy the
- +source code from the same place satisfies the requirement to
- +distribute the source code, even though third parties are not
- +compelled to copy the source along with the object code.
- +
- + 5. A program that contains no derivative of any portion of the
- +Library, but is designed to work with the Library by being compiled or
- +linked with it, is called a "work that uses the Library". Such a
- +work, in isolation, is not a derivative work of the Library, and
- +therefore falls outside the scope of this License.
- +
- + However, linking a "work that uses the Library" with the Library
- +creates an executable that is a derivative of the Library (because it
- +contains portions of the Library), rather than a "work that uses the
- +library". The executable is therefore covered by this License.
- +Section 6 states terms for distribution of such executables.
- +
- + When a "work that uses the Library" uses material from a header file
- +that is part of the Library, the object code for the work may be a
- +derivative work of the Library even though the source code is not.
- +Whether this is true is especially significant if the work can be
- +linked without the Library, or if the work is itself a library. The
- +threshold for this to be true is not precisely defined by law.
- +
- + If such an object file uses only numerical parameters, data
- +structure layouts and accessors, and small macros and small inline
- +functions (ten lines or less in length), then the use of the object
- +file is unrestricted, regardless of whether it is legally a derivative
- +work. (Executables containing this object code plus portions of the
- +Library will still fall under Section 6.)
- +
- + Otherwise, if the work is a derivative of the Library, you may
- +distribute the object code for the work under the terms of Section 6.
- +Any executables containing that work also fall under Section 6,
- +whether or not they are linked directly with the Library itself.
- +
- + 6. As an exception to the Sections above, you may also combine or
- +link a "work that uses the Library" with the Library to produce a
- +work containing portions of the Library, and distribute that work
- +under terms of your choice, provided that the terms permit
- +modification of the work for the customer's own use and reverse
- +engineering for debugging such modifications.
- +
- + You must give prominent notice with each copy of the work that the
- +Library is used in it and that the Library and its use are covered by
- +this License. You must supply a copy of this License. If the work
- +during execution displays copyright notices, you must include the
- +copyright notice for the Library among them, as well as a reference
- +directing the user to the copy of this License. Also, you must do one
- +of these things:
- +
- + a) Accompany the work with the complete corresponding
- + machine-readable source code for the Library including whatever
- + changes were used in the work (which must be distributed under
- + Sections 1 and 2 above); and, if the work is an executable linked
- + with the Library, with the complete machine-readable "work that
- + uses the Library", as object code and/or source code, so that the
- + user can modify the Library and then relink to produce a modified
- + executable containing the modified Library. (It is understood
- + that the user who changes the contents of definitions files in the
- + Library will not necessarily be able to recompile the application
- + to use the modified definitions.)
- +
- + b) Use a suitable shared library mechanism for linking with the
- + Library. A suitable mechanism is one that (1) uses at run time a
- + copy of the library already present on the user's computer system,
- + rather than copying library functions into the executable, and (2)
- + will operate properly with a modified version of the library, if
- + the user installs one, as long as the modified version is
- + interface-compatible with the version that the work was made with.
- +
- + c) Accompany the work with a written offer, valid for at
- + least three years, to give the same user the materials
- + specified in Subsection 6a, above, for a charge no more
- + than the cost of performing this distribution.
- +
- + d) If distribution of the work is made by offering access to copy
- + from a designated place, offer equivalent access to copy the above
- + specified materials from the same place.
- +
- + e) Verify that the user has already received a copy of these
- + materials or that you have already sent this user a copy.
- +
- + For an executable, the required form of the "work that uses the
- +Library" must include any data and utility programs needed for
- +reproducing the executable from it. However, as a special exception,
- +the materials to be distributed need not include anything that is
- +normally distributed (in either source or binary form) with the major
- +components (compiler, kernel, and so on) of the operating system on
- +which the executable runs, unless that component itself accompanies
- +the executable.
- +
- + It may happen that this requirement contradicts the license
- +restrictions of other proprietary libraries that do not normally
- +accompany the operating system. Such a contradiction means you cannot
- +use both them and the Library together in an executable that you
- +distribute.
- +
- + 7. You may place library facilities that are a work based on the
- +Library side-by-side in a single library together with other library
- +facilities not covered by this License, and distribute such a combined
- +library, provided that the separate distribution of the work based on
- +the Library and of the other library facilities is otherwise
- +permitted, and provided that you do these two things:
- +
- + a) Accompany the combined library with a copy of the same work
- + based on the Library, uncombined with any other library
- + facilities. This must be distributed under the terms of the
- + Sections above.
- +
- + b) Give prominent notice with the combined library of the fact
- + that part of it is a work based on the Library, and explaining
- + where to find the accompanying uncombined form of the same work.
- +
- + 8. You may not copy, modify, sublicense, link with, or distribute
- +the Library except as expressly provided under this License. Any
- +attempt otherwise to copy, modify, sublicense, link with, or
- +distribute the Library is void, and will automatically terminate your
- +rights under this License. However, parties who have received copies,
- +or rights, from you under this License will not have their licenses
- +terminated so long as such parties remain in full compliance.
- +
- + 9. You are not required to accept this License, since you have not
- +signed it. However, nothing else grants you permission to modify or
- +distribute the Library or its derivative works. These actions are
- +prohibited by law if you do not accept this License. Therefore, by
- +modifying or distributing the Library (or any work based on the
- +Library), you indicate your acceptance of this License to do so, and
- +all its terms and conditions for copying, distributing or modifying
- +the Library or works based on it.
- +
- + 10. Each time you redistribute the Library (or any work based on the
- +Library), the recipient automatically receives a license from the
- +original licensor to copy, distribute, link with or modify the Library
- +subject to these terms and conditions. You may not impose any further
- +restrictions on the recipients' exercise of the rights granted herein.
- +You are not responsible for enforcing compliance by third parties with
- +this License.
- +
- + 11. If, as a consequence of a court judgment or allegation of patent
- +infringement or for any other reason (not limited to patent issues),
- +conditions are imposed on you (whether by court order, agreement or
- +otherwise) that contradict the conditions of this License, they do not
- +excuse you from the conditions of this License. If you cannot
- +distribute so as to satisfy simultaneously your obligations under this
- +License and any other pertinent obligations, then as a consequence you
- +may not distribute the Library at all. For example, if a patent
- +license would not permit royalty-free redistribution of the Library by
- +all those who receive copies directly or indirectly through you, then
- +the only way you could satisfy both it and this License would be to
- +refrain entirely from distribution of the Library.
- +
- +If any portion of this section is held invalid or unenforceable under any
- +particular circumstance, the balance of the section is intended to apply,
- +and the section as a whole is intended to apply in other circumstances.
- +
- +It is not the purpose of this section to induce you to infringe any
- +patents or other property right claims or to contest validity of any
- +such claims; this section has the sole purpose of protecting the
- +integrity of the free software distribution system which is
- +implemented by public license practices. Many people have made
- +generous contributions to the wide range of software distributed
- +through that system in reliance on consistent application of that
- +system; it is up to the author/donor to decide if he or she is willing
- +to distribute software through any other system and a licensee cannot
- +impose that choice.
- +
- +This section is intended to make thoroughly clear what is believed to
- +be a consequence of the rest of this License.
- +
- + 12. If the distribution and/or use of the Library is restricted in
- +certain countries either by patents or by copyrighted interfaces, the
- +original copyright holder who places the Library under this License may add
- +an explicit geographical distribution limitation excluding those countries,
- +so that distribution is permitted only in or among countries not thus
- +excluded. In such case, this License incorporates the limitation as if
- +written in the body of this License.
- +
- + 13. The Free Software Foundation may publish revised and/or new
- +versions of the Lesser General Public License from time to time.
- +Such new versions will be similar in spirit to the present version,
- +but may differ in detail to address new problems or concerns.
- +
- +Each version is given a distinguishing version number. If the Library
- +specifies a version number of this License which applies to it and
- +"any later version", you have the option of following the terms and
- +conditions either of that version or of any later version published by
- +the Free Software Foundation. If the Library does not specify a
- +license version number, you may choose any version ever published by
- +the Free Software Foundation.
- +
- + 14. If you wish to incorporate parts of the Library into other free
- +programs whose distribution conditions are incompatible with these,
- +write to the author to ask for permission. For software which is
- +copyrighted by the Free Software Foundation, write to the Free
- +Software Foundation; we sometimes make exceptions for this. Our
- +decision will be guided by the two goals of preserving the free status
- +of all derivatives of our free software and of promoting the sharing
- +and reuse of software generally.
- +
- + NO WARRANTY
- +
- + 15. BECAUSE THE LIBRARY IS LICENSED FREE OF CHARGE, THERE IS NO
- +WARRANTY FOR THE LIBRARY, TO THE EXTENT PERMITTED BY APPLICABLE LAW.
- +EXCEPT WHEN OTHERWISE STATED IN WRITING THE COPYRIGHT HOLDERS AND/OR
- +OTHER PARTIES PROVIDE THE LIBRARY "AS IS" WITHOUT WARRANTY OF ANY
- +KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING, BUT NOT LIMITED TO, THE
- +IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
- +PURPOSE. THE ENTIRE RISK AS TO THE QUALITY AND PERFORMANCE OF THE
- +LIBRARY IS WITH YOU. SHOULD THE LIBRARY PROVE DEFECTIVE, YOU ASSUME
- +THE COST OF ALL NECESSARY SERVICING, REPAIR OR CORRECTION.
- +
- + 16. IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN
- +WRITING WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MAY MODIFY
- +AND/OR REDISTRIBUTE THE LIBRARY AS PERMITTED ABOVE, BE LIABLE TO YOU
- +FOR DAMAGES, INCLUDING ANY GENERAL, SPECIAL, INCIDENTAL OR
- +CONSEQUENTIAL DAMAGES ARISING OUT OF THE USE OR INABILITY TO USE THE
- +LIBRARY (INCLUDING BUT NOT LIMITED TO LOSS OF DATA OR DATA BEING
- +RENDERED INACCURATE OR LOSSES SUSTAINED BY YOU OR THIRD PARTIES OR A
- +FAILURE OF THE LIBRARY TO OPERATE WITH ANY OTHER SOFTWARE), EVEN IF
- +SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE POSSIBILITY OF SUCH
- +DAMAGES.
- +
- + END OF TERMS AND CONDITIONS
- +
- + How to Apply These Terms to Your New Libraries
- +
- + If you develop a new library, and you want it to be of the greatest
- +possible use to the public, we recommend making it free software that
- +everyone can redistribute and change. You can do so by permitting
- +redistribution under these terms (or, alternatively, under the terms of the
- +ordinary General Public License).
- +
- + To apply these terms, attach the following notices to the library. It is
- +safest to attach them to the start of each source file to most effectively
- +convey the exclusion of warranty; and each file should have at least the
- +"copyright" line and a pointer to where the full notice is found.
- +
- + <one line to give the library's name and a brief idea of what it does.>
- + Copyright (C) <year> <name of author>
- +
- + This library is free software; you can redistribute it and/or
- + modify it under the terms of the GNU Lesser General Public
- + License as published by the Free Software Foundation; either
- + version 2.1 of the License, or (at your option) any later version.
- +
- + This library is distributed in the hope that it will be useful,
- + but WITHOUT ANY WARRANTY; without even the implied warranty of
- + MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
- + Lesser General Public License for more details.
- +
- + You should have received a copy of the GNU Lesser General Public
- + License along with this library; if not, write to the Free Software
- + Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
- +
- +Also add information on how to contact you by electronic and paper mail.
- +
- +You should also get your employer (if you work as a programmer) or your
- +school, if any, to sign a "copyright disclaimer" for the library, if
- +necessary. Here is a sample; alter the names:
- +
- + Yoyodyne, Inc., hereby disclaims all copyright interest in the
- + library `Frob' (a library for tweaking knobs) written by James Random Hacker.
- +
- + <signature of Ty Coon>, 1 April 1990
- + Ty Coon, President of Vice
- +
- +That's all there is to it!
- diff -Nuar ostree-2018.8.orig/libglnx/.git ostree-2018.8/libglnx/.git
- --- ostree-2018.8.orig/libglnx/.git 1970-01-01 02:00:00.000000000 +0200
- +++ ostree-2018.8/libglnx/.git 2018-05-26 00:37:31.964018845 +0300
- @@ -0,0 +1 @@
- +gitdir: ../.git/modules/libglnx
- diff -Nuar ostree-2018.8.orig/libglnx/.gitignore ostree-2018.8/libglnx/.gitignore
- --- ostree-2018.8.orig/libglnx/.gitignore 1970-01-01 02:00:00.000000000 +0200
- +++ ostree-2018.8/libglnx/.gitignore 2018-05-26 00:37:32.005018846 +0300
- @@ -0,0 +1,16 @@
- +# A path ostree writes to work around automake bug with
- +# subdir-objects
- +Makefile-libglnx.am.inc
- +
- +# Some standard bits
- +.deps
- +.libs
- +.dirstamp
- +*.typelib
- +*.la
- +*.lo
- +*.o
- +*.pyc
- +*.stamp
- +*~
- +
- diff -Nuar ostree-2018.8.orig/libglnx/glnx-backport-autocleanups.h ostree-2018.8/libglnx/glnx-backport-autocleanups.h
- --- ostree-2018.8.orig/libglnx/glnx-backport-autocleanups.h 1970-01-01 02:00:00.000000000 +0200
- +++ ostree-2018.8/libglnx/glnx-backport-autocleanups.h 2018-05-26 00:37:32.006018846 +0300
- @@ -0,0 +1,124 @@
- +/*
- + * Copyright © 2015 Canonical Limited
- + *
- + * This library is free software; you can redistribute it and/or
- + * modify it under the terms of the GNU Lesser General Public
- + * License as published by the Free Software Foundation; either
- + * version 2 of the licence, or (at your option) any later version.
- + *
- + * This library is distributed in the hope that it will be useful,
- + * but WITHOUT ANY WARRANTY; without even the implied warranty of
- + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
- + * Lesser General Public License for more details.
- + *
- + * You should have received a copy of the GNU Lesser General Public
- + * License along with this library; if not, see <http://www.gnu.org/licenses/>.
- + *
- + * Author: Ryan Lortie <desrt@desrt.ca>
- + */
- +
- +#pragma once
- +
- +#include <glnx-backport-autoptr.h>
- +
- +#if !GLIB_CHECK_VERSION(2, 43, 4)
- +
- +static inline void
- +g_autoptr_cleanup_generic_gfree (void *p)
- +{
- + void **pp = (void**)p;
- + if (*pp)
- + g_free (*pp);
- +}
- +
- +G_DEFINE_AUTOPTR_CLEANUP_FUNC(GAsyncQueue, g_async_queue_unref)
- +G_DEFINE_AUTOPTR_CLEANUP_FUNC(GBookmarkFile, g_bookmark_file_free)
- +G_DEFINE_AUTOPTR_CLEANUP_FUNC(GBytes, g_bytes_unref)
- +G_DEFINE_AUTOPTR_CLEANUP_FUNC(GChecksum, g_checksum_free)
- +G_DEFINE_AUTOPTR_CLEANUP_FUNC(GDateTime, g_date_time_unref)
- +G_DEFINE_AUTOPTR_CLEANUP_FUNC(GDir, g_dir_close)
- +G_DEFINE_AUTOPTR_CLEANUP_FUNC(GError, g_error_free)
- +G_DEFINE_AUTOPTR_CLEANUP_FUNC(GHashTable, g_hash_table_unref)
- +G_DEFINE_AUTOPTR_CLEANUP_FUNC(GHmac, g_hmac_unref)
- +G_DEFINE_AUTOPTR_CLEANUP_FUNC(GIOChannel, g_io_channel_unref)
- +G_DEFINE_AUTOPTR_CLEANUP_FUNC(GKeyFile, g_key_file_unref)
- +G_DEFINE_AUTOPTR_CLEANUP_FUNC(GList, g_list_free)
- +G_DEFINE_AUTOPTR_CLEANUP_FUNC(GArray, g_array_unref)
- +G_DEFINE_AUTOPTR_CLEANUP_FUNC(GPtrArray, g_ptr_array_unref)
- +G_DEFINE_AUTOPTR_CLEANUP_FUNC(GMainContext, g_main_context_unref)
- +G_DEFINE_AUTOPTR_CLEANUP_FUNC(GMainLoop, g_main_loop_unref)
- +G_DEFINE_AUTOPTR_CLEANUP_FUNC(GSource, g_source_unref)
- +G_DEFINE_AUTOPTR_CLEANUP_FUNC(GMappedFile, g_mapped_file_unref)
- +G_DEFINE_AUTOPTR_CLEANUP_FUNC(GMarkupParseContext, g_markup_parse_context_unref)
- +G_DEFINE_AUTOPTR_CLEANUP_FUNC(gchar, g_free)
- +G_DEFINE_AUTOPTR_CLEANUP_FUNC(GNode, g_node_destroy)
- +G_DEFINE_AUTOPTR_CLEANUP_FUNC(GOptionContext, g_option_context_free)
- +G_DEFINE_AUTOPTR_CLEANUP_FUNC(GOptionGroup, g_option_group_free)
- +G_DEFINE_AUTOPTR_CLEANUP_FUNC(GPatternSpec, g_pattern_spec_free)
- +G_DEFINE_AUTOPTR_CLEANUP_FUNC(GQueue, g_queue_free)
- +G_DEFINE_AUTO_CLEANUP_CLEAR_FUNC(GQueue, g_queue_clear)
- +G_DEFINE_AUTOPTR_CLEANUP_FUNC(GRand, g_rand_free)
- +G_DEFINE_AUTOPTR_CLEANUP_FUNC(GRegex, g_regex_unref)
- +G_DEFINE_AUTOPTR_CLEANUP_FUNC(GMatchInfo, g_match_info_unref)
- +G_DEFINE_AUTOPTR_CLEANUP_FUNC(GScanner, g_scanner_destroy)
- +G_DEFINE_AUTOPTR_CLEANUP_FUNC(GSequence, g_sequence_free)
- +G_DEFINE_AUTOPTR_CLEANUP_FUNC(GSList, g_slist_free)
- +G_DEFINE_AUTOPTR_CLEANUP_FUNC(GStringChunk, g_string_chunk_free)
- +G_DEFINE_AUTO_CLEANUP_FREE_FUNC(GStrv, g_strfreev, NULL)
- +G_DEFINE_AUTOPTR_CLEANUP_FUNC(GThread, g_thread_unref)
- +G_DEFINE_AUTO_CLEANUP_CLEAR_FUNC(GMutex, g_mutex_clear)
- +G_DEFINE_AUTO_CLEANUP_CLEAR_FUNC(GCond, g_cond_clear)
- +G_DEFINE_AUTOPTR_CLEANUP_FUNC(GTimer, g_timer_destroy)
- +G_DEFINE_AUTOPTR_CLEANUP_FUNC(GTimeZone, g_time_zone_unref)
- +G_DEFINE_AUTOPTR_CLEANUP_FUNC(GTree, g_tree_unref)
- +G_DEFINE_AUTOPTR_CLEANUP_FUNC(GVariant, g_variant_unref)
- +G_DEFINE_AUTOPTR_CLEANUP_FUNC(GVariantBuilder, g_variant_builder_unref)
- +G_DEFINE_AUTO_CLEANUP_CLEAR_FUNC(GVariantBuilder, g_variant_builder_clear)
- +G_DEFINE_AUTOPTR_CLEANUP_FUNC(GVariantIter, g_variant_iter_free)
- +G_DEFINE_AUTOPTR_CLEANUP_FUNC(GVariantDict, g_variant_dict_unref)
- +G_DEFINE_AUTO_CLEANUP_CLEAR_FUNC(GVariantDict, g_variant_dict_clear)
- +G_DEFINE_AUTOPTR_CLEANUP_FUNC(GVariantType, g_variant_type_free)
- +G_DEFINE_AUTOPTR_CLEANUP_FUNC(GSubprocess, g_object_unref)
- +G_DEFINE_AUTOPTR_CLEANUP_FUNC(GSubprocessLauncher, g_object_unref)
- +
- +/* Add GObject-based types as needed. */
- +G_DEFINE_AUTOPTR_CLEANUP_FUNC(GCancellable, g_object_unref)
- +G_DEFINE_AUTOPTR_CLEANUP_FUNC(GConverter, g_object_unref)
- +G_DEFINE_AUTOPTR_CLEANUP_FUNC(GConverterOutputStream, g_object_unref)
- +G_DEFINE_AUTOPTR_CLEANUP_FUNC(GDataInputStream, g_object_unref)
- +G_DEFINE_AUTOPTR_CLEANUP_FUNC(GFile, g_object_unref)
- +G_DEFINE_AUTOPTR_CLEANUP_FUNC(GFileEnumerator, g_object_unref)
- +G_DEFINE_AUTOPTR_CLEANUP_FUNC(GFileIOStream, g_object_unref)
- +G_DEFINE_AUTOPTR_CLEANUP_FUNC(GFileInfo, g_object_unref)
- +G_DEFINE_AUTOPTR_CLEANUP_FUNC(GFileInputStream, g_object_unref)
- +G_DEFINE_AUTOPTR_CLEANUP_FUNC(GFileMonitor, g_object_unref)
- +G_DEFINE_AUTOPTR_CLEANUP_FUNC(GFileOutputStream, g_object_unref)
- +G_DEFINE_AUTOPTR_CLEANUP_FUNC(GInputStream, g_object_unref)
- +G_DEFINE_AUTOPTR_CLEANUP_FUNC(GMemoryInputStream, g_object_unref)
- +G_DEFINE_AUTOPTR_CLEANUP_FUNC(GMemoryOutputStream, g_object_unref)
- +G_DEFINE_AUTOPTR_CLEANUP_FUNC(GOutputStream, g_object_unref)
- +G_DEFINE_AUTOPTR_CLEANUP_FUNC(GSocket, g_object_unref)
- +G_DEFINE_AUTOPTR_CLEANUP_FUNC(GSocketAddress, g_object_unref)
- +G_DEFINE_AUTOPTR_CLEANUP_FUNC(GTask, g_object_unref)
- +G_DEFINE_AUTOPTR_CLEANUP_FUNC(GTlsCertificate, g_object_unref)
- +G_DEFINE_AUTOPTR_CLEANUP_FUNC(GTlsDatabase, g_object_unref)
- +G_DEFINE_AUTOPTR_CLEANUP_FUNC(GTlsInteraction, g_object_unref)
- +G_DEFINE_AUTOPTR_CLEANUP_FUNC(GDBusConnection, g_object_unref)
- +G_DEFINE_AUTOPTR_CLEANUP_FUNC(GDBusMessage, g_object_unref)
- +G_DEFINE_AUTOPTR_CLEANUP_FUNC(GZlibCompressor, g_object_unref)
- +G_DEFINE_AUTOPTR_CLEANUP_FUNC(GZlibDecompressor, g_object_unref)
- +
- +#endif
- +
- +#if !GLIB_CHECK_VERSION(2, 45, 8)
- +
- +static inline void
- +g_autoptr_cleanup_gstring_free (GString *string)
- +{
- + if (string)
- + g_string_free (string, TRUE);
- +}
- +
- +G_DEFINE_AUTOPTR_CLEANUP_FUNC(GString, g_autoptr_cleanup_gstring_free)
- +
- +#endif
- diff -Nuar ostree-2018.8.orig/libglnx/glnx-backport-autoptr.h ostree-2018.8/libglnx/glnx-backport-autoptr.h
- --- ostree-2018.8.orig/libglnx/glnx-backport-autoptr.h 1970-01-01 02:00:00.000000000 +0200
- +++ ostree-2018.8/libglnx/glnx-backport-autoptr.h 2018-05-26 00:37:32.006018846 +0300
- @@ -0,0 +1,133 @@
- +/* -*- mode: C; c-file-style: "gnu"; indent-tabs-mode: nil; -*-
- + *
- + * Copyright (C) 2015 Colin Walters <walters@verbum.org>
- + *
- + * GLIB - Library of useful routines for C programming
- + * Copyright (C) 1995-1997 Peter Mattis, Spencer Kimball and Josh MacDonald
- + *
- + * This library is free software; you can redistribute it and/or
- + * modify it under the terms of the GNU Lesser General Public
- + * License as published by the Free Software Foundation; either
- + * version 2 of the License, or (at your option) any later version.
- + *
- + * This library is distributed in the hope that it will be useful,
- + * but WITHOUT ANY WARRANTY; without even the implied warranty of
- + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
- + * Lesser General Public License for more details.
- + *
- + * You should have received a copy of the GNU Lesser General Public
- + * License along with this library; if not, write to the
- + * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
- + * Boston, MA 02111-1307, USA.
- + */
- +
- +#pragma once
- +
- +#include <gio/gio.h>
- +
- +G_BEGIN_DECLS
- +
- +#if !GLIB_CHECK_VERSION(2, 43, 4)
- +
- +#define _GLIB_AUTOPTR_FUNC_NAME(TypeName) glib_autoptr_cleanup_##TypeName
- +#define _GLIB_AUTOPTR_TYPENAME(TypeName) TypeName##_autoptr
- +#define _GLIB_AUTO_FUNC_NAME(TypeName) glib_auto_cleanup_##TypeName
- +#define _GLIB_CLEANUP(func) __attribute__((cleanup(func)))
- +#define _GLIB_DEFINE_AUTOPTR_CHAINUP(ModuleObjName, ParentName) \
- + typedef ModuleObjName *_GLIB_AUTOPTR_TYPENAME(ModuleObjName); \
- + static inline void _GLIB_AUTOPTR_FUNC_NAME(ModuleObjName) (ModuleObjName **_ptr) { \
- + _GLIB_AUTOPTR_FUNC_NAME(ParentName) ((ParentName **) _ptr); } \
- +
- +
- +/* these macros are API */
- +#define G_DEFINE_AUTOPTR_CLEANUP_FUNC(TypeName, func) \
- + typedef TypeName *_GLIB_AUTOPTR_TYPENAME(TypeName); \
- + G_GNUC_BEGIN_IGNORE_DEPRECATIONS \
- + static inline void _GLIB_AUTOPTR_FUNC_NAME(TypeName) (TypeName **_ptr) { if (*_ptr) (func) (*_ptr); } \
- + G_GNUC_END_IGNORE_DEPRECATIONS
- +#define G_DEFINE_AUTO_CLEANUP_CLEAR_FUNC(TypeName, func) \
- + G_GNUC_BEGIN_IGNORE_DEPRECATIONS \
- + static inline void _GLIB_AUTO_FUNC_NAME(TypeName) (TypeName *_ptr) { (func) (_ptr); } \
- + G_GNUC_END_IGNORE_DEPRECATIONS
- +#define G_DEFINE_AUTO_CLEANUP_FREE_FUNC(TypeName, func, none) \
- + G_GNUC_BEGIN_IGNORE_DEPRECATIONS \
- + static inline void _GLIB_AUTO_FUNC_NAME(TypeName) (TypeName *_ptr) { if (*_ptr != none) (func) (*_ptr); } \
- + G_GNUC_END_IGNORE_DEPRECATIONS
- +#define g_autoptr(TypeName) _GLIB_CLEANUP(_GLIB_AUTOPTR_FUNC_NAME(TypeName)) _GLIB_AUTOPTR_TYPENAME(TypeName)
- +#define g_auto(TypeName) _GLIB_CLEANUP(_GLIB_AUTO_FUNC_NAME(TypeName)) TypeName
- +#define g_autofree _GLIB_CLEANUP(g_autoptr_cleanup_generic_gfree)
- +
- +/**
- + * g_steal_pointer:
- + * @pp: a pointer to a pointer
- + *
- + * Sets @pp to %NULL, returning the value that was there before.
- + *
- + * Conceptually, this transfers the ownership of the pointer from the
- + * referenced variable to the "caller" of the macro (ie: "steals" the
- + * reference).
- + *
- + * The return value will be properly typed, according to the type of
- + * @pp.
- + *
- + * This can be very useful when combined with g_autoptr() to prevent the
- + * return value of a function from being automatically freed. Consider
- + * the following example (which only works on GCC and clang):
- + *
- + * |[
- + * GObject *
- + * create_object (void)
- + * {
- + * g_autoptr(GObject) obj = g_object_new (G_TYPE_OBJECT, NULL);
- + *
- + * if (early_error_case)
- + * return NULL;
- + *
- + * return g_steal_pointer (&obj);
- + * }
- + * ]|
- + *
- + * It can also be used in similar ways for 'out' parameters and is
- + * particularly useful for dealing with optional out parameters:
- + *
- + * |[
- + * gboolean
- + * get_object (GObject **obj_out)
- + * {
- + * g_autoptr(GObject) obj = g_object_new (G_TYPE_OBJECT, NULL);
- + *
- + * if (early_error_case)
- + * return FALSE;
- + *
- + * if (obj_out)
- + * *obj_out = g_steal_pointer (&obj);
- + *
- + * return TRUE;
- + * }
- + * ]|
- + *
- + * In the above example, the object will be automatically freed in the
- + * early error case and also in the case that %NULL was given for
- + * @obj_out.
- + *
- + * Since: 2.44
- + */
- +static inline gpointer
- +(g_steal_pointer) (gpointer pp)
- +{
- + gpointer *ptr = (gpointer *) pp;
- + gpointer ref;
- +
- + ref = *ptr;
- + *ptr = NULL;
- +
- + return ref;
- +}
- +
- +/* type safety */
- +#define g_steal_pointer(pp) \
- + (0 ? (*(pp)) : (g_steal_pointer) (pp))
- +
- +#endif /* !GLIB_CHECK_VERSION(2, 43, 3) */
- +
- +G_END_DECLS
- diff -Nuar ostree-2018.8.orig/libglnx/glnx-backports.c ostree-2018.8/libglnx/glnx-backports.c
- --- ostree-2018.8.orig/libglnx/glnx-backports.c 1970-01-01 02:00:00.000000000 +0200
- +++ ostree-2018.8/libglnx/glnx-backports.c 2018-05-26 00:37:32.006018846 +0300
- @@ -0,0 +1,61 @@
- +/* -*- mode: C; c-file-style: "gnu"; indent-tabs-mode: nil; -*-
- + *
- + * Copyright (C) 2015 Colin Walters <walters@verbum.org>
- + *
- + * This program is free software: you can redistribute it and/or modify
- + * it under the terms of the GNU Lesser General Public License as published
- + * by the Free Software Foundation; either version 2 of the licence or (at
- + * your option) any later version.
- + *
- + * This library is distributed in the hope that it will be useful,
- + * but WITHOUT ANY WARRANTY; without even the implied warranty of
- + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
- + * Lesser General Public License for more details.
- + *
- + * You should have received a copy of the GNU Lesser General
- + * Public License along with this library; if not, write to the
- + * Free Software Foundation, Inc., 59 Temple Place, Suite 330,
- + * Boston, MA 02111-1307, USA.
- + */
- +
- +#include "config.h"
- +
- +#include "glnx-backports.h"
- +
- +#if !GLIB_CHECK_VERSION(2, 44, 0)
- +gboolean
- +glnx_strv_contains (const gchar * const *strv,
- + const gchar *str)
- +{
- + g_return_val_if_fail (strv != NULL, FALSE);
- + g_return_val_if_fail (str != NULL, FALSE);
- +
- + for (; *strv != NULL; strv++)
- + {
- + if (g_str_equal (str, *strv))
- + return TRUE;
- + }
- +
- + return FALSE;
- +}
- +
- +gboolean
- +glnx_set_object (GObject **object_ptr,
- + GObject *new_object)
- +{
- + GObject *old_object = *object_ptr;
- +
- + if (old_object == new_object)
- + return FALSE;
- +
- + if (new_object != NULL)
- + g_object_ref (new_object);
- +
- + *object_ptr = new_object;
- +
- + if (old_object != NULL)
- + g_object_unref (old_object);
- +
- + return TRUE;
- +}
- +#endif
- diff -Nuar ostree-2018.8.orig/libglnx/glnx-backports.h ostree-2018.8/libglnx/glnx-backports.h
- --- ostree-2018.8.orig/libglnx/glnx-backports.h 1970-01-01 02:00:00.000000000 +0200
- +++ ostree-2018.8/libglnx/glnx-backports.h 2018-05-26 00:37:32.006018846 +0300
- @@ -0,0 +1,46 @@
- +/* -*- mode: C; c-file-style: "gnu"; indent-tabs-mode: nil; -*-
- + *
- + * Copyright (C) 2015 Colin Walters <walters@verbum.org>
- + *
- + * GLIB - Library of useful routines for C programming
- + * Copyright (C) 1995-1997 Peter Mattis, Spencer Kimball and Josh MacDonald
- + *
- + * This library is free software; you can redistribute it and/or
- + * modify it under the terms of the GNU Lesser General Public
- + * License as published by the Free Software Foundation; either
- + * version 2 of the License, or (at your option) any later version.
- + *
- + * This library is distributed in the hope that it will be useful,
- + * but WITHOUT ANY WARRANTY; without even the implied warranty of
- + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
- + * Lesser General Public License for more details.
- + *
- + * You should have received a copy of the GNU Lesser General Public
- + * License along with this library; if not, write to the
- + * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
- + * Boston, MA 02111-1307, USA.
- + */
- +
- +#pragma once
- +
- +#include <gio/gio.h>
- +
- +G_BEGIN_DECLS
- +
- +#if !GLIB_CHECK_VERSION(2, 44, 0)
- +
- +#define g_strv_contains glnx_strv_contains
- +gboolean glnx_strv_contains (const gchar * const *strv,
- + const gchar *str);
- +
- +#define g_set_object(object_ptr, new_object) \
- + (/* Check types match. */ \
- + 0 ? *(object_ptr) = (new_object), FALSE : \
- + glnx_set_object ((GObject **) (object_ptr), (GObject *) (new_object)) \
- + )
- +gboolean glnx_set_object (GObject **object_ptr,
- + GObject *new_object);
- +
- +#endif /* !GLIB_CHECK_VERSION(2, 44, 0) */
- +
- +G_END_DECLS
- diff -Nuar ostree-2018.8.orig/libglnx/glnx-console.c ostree-2018.8/libglnx/glnx-console.c
- --- ostree-2018.8.orig/libglnx/glnx-console.c 1970-01-01 02:00:00.000000000 +0200
- +++ ostree-2018.8/libglnx/glnx-console.c 2018-05-26 00:37:32.006018846 +0300
- @@ -0,0 +1,359 @@
- +/* -*- mode: C; c-file-style: "gnu"; indent-tabs-mode: nil; -*-
- + *
- + * Copyright (C) 2013,2014,2015 Colin Walters <walters@verbum.org>
- + *
- + * This program is free software: you can redistribute it and/or modify
- + * it under the terms of the GNU Lesser General Public License as published
- + * by the Free Software Foundation; either version 2 of the licence or (at
- + * your option) any later version.
- + *
- + * This library is distributed in the hope that it will be useful,
- + * but WITHOUT ANY WARRANTY; without even the implied warranty of
- + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
- + * Lesser General Public License for more details.
- + *
- + * You should have received a copy of the GNU Lesser General
- + * Public License along with this library; if not, write to the
- + * Free Software Foundation, Inc., 59 Temple Place, Suite 330,
- + * Boston, MA 02111-1307, USA.
- + */
- +
- +#include "config.h"
- +
- +#include "glnx-console.h"
- +
- +#include <unistd.h>
- +#include <string.h>
- +#include <fcntl.h>
- +#include <stdio.h>
- +#include <errno.h>
- +#include <sys/ioctl.h>
- +
- +/* For people with widescreen monitors and maximized terminals, it looks pretty
- + * bad to have an enormous progress bar. For much the same reason as web pages
- + * tend to have a maximum width;
- + * https://ux.stackexchange.com/questions/48982/suggest-good-max-width-for-fluid-width-design
- + */
- +#define MAX_PROGRESSBAR_COLUMNS 20
- +
- +/* Max updates output per second. On a tty there's no point to rendering
- + * extremely fast; and for a non-tty we're probably in a Jenkins job
- + * or whatever and having percentages spam multiple lines there is annoying.
- + */
- +#define MAX_TTY_UPDATE_HZ (5)
- +#define MAX_NONTTY_UPDATE_HZ (1)
- +
- +static gboolean locked;
- +static guint64 last_update_ms; /* monotonic time in millis we last updated */
- +
- +gboolean
- +glnx_stdout_is_tty (void)
- +{
- + static gsize initialized = 0;
- + static gboolean stdout_is_tty_v;
- +
- + if (g_once_init_enter (&initialized))
- + {
- + stdout_is_tty_v = isatty (1);
- + g_once_init_leave (&initialized, 1);
- + }
- +
- + return stdout_is_tty_v;
- +}
- +
- +static volatile guint cached_columns = 0;
- +static volatile guint cached_lines = 0;
- +
- +static int
- +fd_columns (int fd)
- +{
- + struct winsize ws = {};
- +
- + if (ioctl (fd, TIOCGWINSZ, &ws) < 0)
- + return -errno;
- +
- + if (ws.ws_col <= 0)
- + return -EIO;
- +
- + return ws.ws_col;
- +}
- +
- +/**
- + * glnx_console_columns:
- + *
- + * Returns: The number of columns for terminal output
- + */
- +guint
- +glnx_console_columns (void)
- +{
- + if (G_UNLIKELY (cached_columns == 0))
- + {
- + int c;
- +
- + c = fd_columns (STDOUT_FILENO);
- +
- + if (c <= 0)
- + c = 80;
- +
- + if (c > 256)
- + c = 256;
- +
- + cached_columns = c;
- + }
- +
- + return cached_columns;
- +}
- +
- +static int
- +fd_lines (int fd)
- +{
- + struct winsize ws = {};
- +
- + if (ioctl (fd, TIOCGWINSZ, &ws) < 0)
- + return -errno;
- +
- + if (ws.ws_row <= 0)
- + return -EIO;
- +
- + return ws.ws_row;
- +}
- +
- +/**
- + * glnx_console_lines:
- + *
- + * Returns: The number of lines for terminal output
- + */
- +guint
- +glnx_console_lines (void)
- +{
- + if (G_UNLIKELY (cached_lines == 0))
- + {
- + int l;
- +
- + l = fd_lines (STDOUT_FILENO);
- +
- + if (l <= 0)
- + l = 24;
- +
- + cached_lines = l;
- + }
- +
- + return cached_lines;
- +}
- +
- +static void
- +on_sigwinch (int signum)
- +{
- + cached_columns = 0;
- + cached_lines = 0;
- +}
- +
- +void
- +glnx_console_lock (GLnxConsoleRef *console)
- +{
- + static gsize sigwinch_initialized = 0;
- +
- + g_return_if_fail (!locked);
- + g_return_if_fail (!console->locked);
- +
- + console->is_tty = glnx_stdout_is_tty ();
- +
- + locked = console->locked = TRUE;
- +
- + if (console->is_tty)
- + {
- + if (g_once_init_enter (&sigwinch_initialized))
- + {
- + signal (SIGWINCH, on_sigwinch);
- + g_once_init_leave (&sigwinch_initialized, 1);
- + }
- +
- + { static const char initbuf[] = { '\n', 0x1B, 0x37 };
- + (void) fwrite (initbuf, 1, sizeof (initbuf), stdout);
- + }
- + }
- +}
- +
- +static void
- +printpad (const char *padbuf,
- + guint padbuf_len,
- + guint n)
- +{
- + const guint d = n / padbuf_len;
- + const guint r = n % padbuf_len;
- + guint i;
- +
- + for (i = 0; i < d; i++)
- + fwrite (padbuf, 1, padbuf_len, stdout);
- + fwrite (padbuf, 1, r, stdout);
- +}
- +
- +static void
- +text_percent_internal (const char *text,
- + int percentage)
- +{
- + /* Check whether we're trying to render too fast; unless percentage is 100, in
- + * which case we assume this is the last call, so we always render it.
- + */
- + const guint64 current_ms = g_get_monotonic_time () / 1000;
- + if (percentage != 100)
- + {
- + const guint64 diff_ms = current_ms - last_update_ms;
- + if (glnx_stdout_is_tty ())
- + {
- + if (diff_ms < (1000/MAX_TTY_UPDATE_HZ))
- + return;
- + }
- + else
- + {
- + if (diff_ms < (1000/MAX_NONTTY_UPDATE_HZ))
- + return;
- + }
- + }
- + last_update_ms = current_ms;
- +
- + static const char equals[] = "====================";
- + const guint n_equals = sizeof (equals) - 1;
- + static const char spaces[] = " ";
- + const guint n_spaces = sizeof (spaces) - 1;
- + const guint ncolumns = glnx_console_columns ();
- + const guint bar_min = 10;
- +
- + if (text && !*text)
- + text = NULL;
- +
- + const guint input_textlen = text ? strlen (text) : 0;
- +
- + if (!glnx_stdout_is_tty ())
- + {
- + if (text)
- + fprintf (stdout, "%s", text);
- + if (percentage != -1)
- + {
- + if (text)
- + fputc (' ', stdout);
- + fprintf (stdout, "%u%%", percentage);
- + }
- + fputc ('\n', stdout);
- + fflush (stdout);
- + return;
- + }
- +
- + if (ncolumns < bar_min)
- + return; /* TODO: spinner */
- +
- + /* Restore cursor */
- + { const char beginbuf[2] = { 0x1B, 0x38 };
- + (void) fwrite (beginbuf, 1, sizeof (beginbuf), stdout);
- + }
- +
- + if (percentage == -1)
- + {
- + if (text != NULL)
- + fwrite (text, 1, input_textlen, stdout);
- +
- + /* Overwrite remaining space, if any */
- + if (ncolumns > input_textlen)
- + printpad (spaces, n_spaces, ncolumns - input_textlen);
- + }
- + else
- + {
- + const guint textlen = MIN (input_textlen, ncolumns - bar_min);
- + const guint barlen = MIN (MAX_PROGRESSBAR_COLUMNS, ncolumns - (textlen + 1));
- +
- + if (textlen > 0)
- + {
- + fwrite (text, 1, textlen, stdout);
- + fputc (' ', stdout);
- + }
- +
- + {
- + const guint nbraces = 2;
- + const guint textpercent_len = 5;
- + const guint bar_internal_len = barlen - nbraces - textpercent_len;
- + const guint eqlen = bar_internal_len * (percentage / 100.0);
- + const guint spacelen = bar_internal_len - eqlen;
- +
- + fputc ('[', stdout);
- + printpad (equals, n_equals, eqlen);
- + printpad (spaces, n_spaces, spacelen);
- + fputc (']', stdout);
- + fprintf (stdout, " %3d%%", percentage);
- + }
- + }
- +
- + fflush (stdout);
- +}
- +
- +/**
- + * glnx_console_progress_text_percent:
- + * @text: Show this text before the progress bar
- + * @percentage: An integer in the range of 0 to 100
- + *
- + * On a tty, print to the console @text followed by an ASCII art
- + * progress bar whose percentage is @percentage. If stdout is not a
- + * tty, a more basic line by line change will be printed.
- + *
- + * You must have called glnx_console_lock() before invoking this
- + * function.
- + *
- + */
- +void
- +glnx_console_progress_text_percent (const char *text,
- + guint percentage)
- +{
- + g_return_if_fail (percentage <= 100);
- +
- + text_percent_internal (text, percentage);
- +}
- +
- +/**
- + * glnx_console_progress_n_items:
- + * @text: Show this text before the progress bar
- + * @current: An integer for how many items have been processed
- + * @total: An integer for how many items there are total
- + *
- + * On a tty, print to the console @text followed by [@current/@total],
- + * then an ASCII art progress bar, like glnx_console_progress_text_percent().
- + *
- + * You must have called glnx_console_lock() before invoking this
- + * function.
- + */
- +void
- +glnx_console_progress_n_items (const char *text,
- + guint current,
- + guint total)
- +{
- + g_return_if_fail (current <= total);
- + g_return_if_fail (total > 0);
- +
- + g_autofree char *newtext = g_strdup_printf ("%s (%u/%u)", text, current, total);
- + /* Special case current == total to ensure we end at 100% */
- + int percentage = (current == total) ? 100 : (((double)current) / total * 100);
- + glnx_console_progress_text_percent (newtext, percentage);
- +}
- +
- +void
- +glnx_console_text (const char *text)
- +{
- + text_percent_internal (text, -1);
- +}
- +
- +/**
- + * glnx_console_unlock:
- + *
- + * Print a newline, and reset all cached console progress state.
- + *
- + * This function does nothing if stdout is not a tty.
- + */
- +void
- +glnx_console_unlock (GLnxConsoleRef *console)
- +{
- + g_return_if_fail (locked);
- + g_return_if_fail (console->locked);
- +
- + if (console->is_tty)
- + fputc ('\n', stdout);
- +
- + locked = console->locked = FALSE;
- +}
- diff -Nuar ostree-2018.8.orig/libglnx/glnx-console.h ostree-2018.8/libglnx/glnx-console.h
- --- ostree-2018.8.orig/libglnx/glnx-console.h 1970-01-01 02:00:00.000000000 +0200
- +++ ostree-2018.8/libglnx/glnx-console.h 2018-05-26 00:37:32.006018846 +0300
- @@ -0,0 +1,61 @@
- +/* -*- mode: C; c-file-style: "gnu"; indent-tabs-mode: nil; -*-
- + *
- + * Copyright (C) 2013,2014,2015 Colin Walters <walters@verbum.org>
- + *
- + * This program is free software: you can redistribute it and/or modify
- + * it under the terms of the GNU Lesser General Public License as published
- + * by the Free Software Foundation; either version 2 of the licence or (at
- + * your option) any later version.
- + *
- + * This library is distributed in the hope that it will be useful,
- + * but WITHOUT ANY WARRANTY; without even the implied warranty of
- + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
- + * Lesser General Public License for more details.
- + *
- + * You should have received a copy of the GNU Lesser General
- + * Public License along with this library; if not, write to the
- + * Free Software Foundation, Inc., 59 Temple Place, Suite 330,
- + * Boston, MA 02111-1307, USA.
- + */
- +
- +#pragma once
- +
- +#include <glnx-backport-autocleanups.h>
- +
- +G_BEGIN_DECLS
- +
- +struct GLnxConsoleRef {
- + gboolean locked;
- + gboolean is_tty;
- +};
- +
- +typedef struct GLnxConsoleRef GLnxConsoleRef;
- +
- +gboolean glnx_stdout_is_tty (void);
- +
- +void glnx_console_lock (GLnxConsoleRef *ref);
- +
- +void glnx_console_text (const char *text);
- +
- +void glnx_console_progress_text_percent (const char *text,
- + guint percentage);
- +
- +void glnx_console_progress_n_items (const char *text,
- + guint current,
- + guint total);
- +
- +void glnx_console_unlock (GLnxConsoleRef *ref);
- +
- +guint glnx_console_lines (void);
- +
- +guint glnx_console_columns (void);
- +
- +static inline void
- +glnx_console_ref_cleanup (GLnxConsoleRef *p)
- +{
- + if (p->locked)
- + glnx_console_unlock (p);
- +}
- +G_DEFINE_AUTO_CLEANUP_CLEAR_FUNC(GLnxConsoleRef, glnx_console_ref_cleanup)
- +
- +G_END_DECLS
- diff -Nuar ostree-2018.8.orig/libglnx/glnx-dirfd.c ostree-2018.8/libglnx/glnx-dirfd.c
- --- ostree-2018.8.orig/libglnx/glnx-dirfd.c 1970-01-01 02:00:00.000000000 +0200
- +++ ostree-2018.8/libglnx/glnx-dirfd.c 2018-05-26 00:37:32.006018846 +0300
- @@ -0,0 +1,425 @@
- +/* -*- mode: C; c-file-style: "gnu"; indent-tabs-mode: nil; -*-
- + *
- + * Copyright (C) 2014,2015 Colin Walters <walters@verbum.org>.
- + *
- + * This library is free software; you can redistribute it and/or
- + * modify it under the terms of the GNU Lesser General Public
- + * License as published by the Free Software Foundation; either
- + * version 2 of the License, or (at your option) any later version.
- + *
- + * This library is distributed in the hope that it will be useful,
- + * but WITHOUT ANY WARRANTY; without even the implied warranty of
- + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
- + * Lesser General Public License for more details.
- + *
- + * You should have received a copy of the GNU Lesser General Public
- + * License along with this library; if not, write to the
- + * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
- + * Boston, MA 02111-1307, USA.
- + */
- +
- +#include "config.h"
- +
- +#include <string.h>
- +
- +#include <glnx-dirfd.h>
- +#include <glnx-fdio.h>
- +#include <glnx-errors.h>
- +#include <glnx-local-alloc.h>
- +#include <glnx-shutil.h>
- +
- +/**
- + * glnx_opendirat_with_errno:
- + * @dfd: File descriptor for origin directory
- + * @name: Pathname, relative to @dfd
- + * @follow: Whether or not to follow symbolic links
- + *
- + * Use openat() to open a directory, using a standard set of flags.
- + * This function sets errno.
- + */
- +int
- +glnx_opendirat_with_errno (int dfd,
- + const char *path,
- + gboolean follow)
- +{
- + int flags = O_RDONLY | O_NONBLOCK | O_DIRECTORY | O_CLOEXEC | O_NOCTTY;
- + if (!follow)
- + flags |= O_NOFOLLOW;
- +
- + dfd = glnx_dirfd_canonicalize (dfd);
- +
- + return openat (dfd, path, flags);
- +}
- +
- +/**
- + * glnx_opendirat:
- + * @dfd: File descriptor for origin directory
- + * @path: Pathname, relative to @dfd
- + * @follow: Whether or not to follow symbolic links
- + * @error: Error
- + *
- + * Use openat() to open a directory, using a standard set of flags.
- + */
- +gboolean
- +glnx_opendirat (int dfd,
- + const char *path,
- + gboolean follow,
- + int *out_fd,
- + GError **error)
- +{
- + int ret = glnx_opendirat_with_errno (dfd, path, follow);
- + if (ret == -1)
- + return glnx_throw_errno_prefix (error, "opendir(%s)", path);
- + *out_fd = ret;
- + return TRUE;
- +}
- +
- +struct GLnxRealDirfdIterator
- +{
- + gboolean initialized;
- + int fd;
- + DIR *d;
- +};
- +typedef struct GLnxRealDirfdIterator GLnxRealDirfdIterator;
- +
- +/**
- + * glnx_dirfd_iterator_init_at:
- + * @dfd: File descriptor, may be AT_FDCWD or -1
- + * @path: Path, may be relative to @dfd
- + * @follow: If %TRUE and the last component of @path is a symlink, follow it
- + * @out_dfd_iter: (out caller-allocates): A directory iterator, will be initialized
- + * @error: Error
- + *
- + * Initialize @out_dfd_iter from @dfd and @path.
- + */
- +gboolean
- +glnx_dirfd_iterator_init_at (int dfd,
- + const char *path,
- + gboolean follow,
- + GLnxDirFdIterator *out_dfd_iter,
- + GError **error)
- +{
- + glnx_autofd int fd = -1;
- + if (!glnx_opendirat (dfd, path, follow, &fd, error))
- + return FALSE;
- +
- + if (!glnx_dirfd_iterator_init_take_fd (&fd, out_dfd_iter, error))
- + return FALSE;
- +
- + return TRUE;
- +}
- +
- +/**
- + * glnx_dirfd_iterator_init_take_fd:
- + * @dfd: File descriptor - ownership is taken, and the value is set to -1
- + * @dfd_iter: A directory iterator
- + * @error: Error
- + *
- + * Steal ownership of @dfd, using it to initialize @dfd_iter for
- + * iteration.
- + */
- +gboolean
- +glnx_dirfd_iterator_init_take_fd (int *dfd,
- + GLnxDirFdIterator *dfd_iter,
- + GError **error)
- +{
- + GLnxRealDirfdIterator *real_dfd_iter = (GLnxRealDirfdIterator*) dfd_iter;
- + DIR *d = fdopendir (*dfd);
- + if (!d)
- + return glnx_throw_errno_prefix (error, "fdopendir");
- +
- + real_dfd_iter->fd = glnx_steal_fd (dfd);
- + real_dfd_iter->d = d;
- + real_dfd_iter->initialized = TRUE;
- +
- + return TRUE;
- +}
- +
- +/**
- + * glnx_dirfd_iterator_next_dent:
- + * @dfd_iter: A directory iterator
- + * @out_dent: (out) (transfer none): Pointer to dirent; do not free
- + * @cancellable: Cancellable
- + * @error: Error
- + *
- + * Read the next value from @dfd_iter, causing @out_dent to be
- + * updated. If end of stream is reached, @out_dent will be set
- + * to %NULL, and %TRUE will be returned.
- + */
- +gboolean
- +glnx_dirfd_iterator_next_dent (GLnxDirFdIterator *dfd_iter,
- + struct dirent **out_dent,
- + GCancellable *cancellable,
- + GError **error)
- +{
- + GLnxRealDirfdIterator *real_dfd_iter = (GLnxRealDirfdIterator*) dfd_iter;
- +
- + g_return_val_if_fail (out_dent, FALSE);
- + g_return_val_if_fail (dfd_iter->initialized, FALSE);
- +
- + if (g_cancellable_set_error_if_cancelled (cancellable, error))
- + return FALSE;
- +
- + do
- + {
- + errno = 0;
- + *out_dent = readdir (real_dfd_iter->d);
- + if (*out_dent == NULL && errno != 0)
- + return glnx_throw_errno_prefix (error, "readdir");
- + } while (*out_dent &&
- + (strcmp ((*out_dent)->d_name, ".") == 0 ||
- + strcmp ((*out_dent)->d_name, "..") == 0));
- +
- + return TRUE;
- +}
- +
- +/**
- + * glnx_dirfd_iterator_next_dent_ensure_dtype:
- + * @dfd_iter: A directory iterator
- + * @out_dent: (out) (transfer none): Pointer to dirent; do not free
- + * @cancellable: Cancellable
- + * @error: Error
- + *
- + * A variant of @glnx_dirfd_iterator_next_dent, which will ensure the
- + * `dent->d_type` member is filled in by calling `fstatat`
- + * automatically if the underlying filesystem type sets `DT_UNKNOWN`.
- + */
- +gboolean
- +glnx_dirfd_iterator_next_dent_ensure_dtype (GLnxDirFdIterator *dfd_iter,
- + struct dirent **out_dent,
- + GCancellable *cancellable,
- + GError **error)
- +{
- + g_return_val_if_fail (out_dent, FALSE);
- +
- + if (!glnx_dirfd_iterator_next_dent (dfd_iter, out_dent, cancellable, error))
- + return FALSE;
- +
- + struct dirent *ret_dent = *out_dent;
- + if (ret_dent)
- + {
- +
- + if (ret_dent->d_type == DT_UNKNOWN)
- + {
- + struct stat stbuf;
- + if (!glnx_fstatat (dfd_iter->fd, ret_dent->d_name, &stbuf, AT_SYMLINK_NOFOLLOW, error))
- + return FALSE;
- + ret_dent->d_type = IFTODT (stbuf.st_mode);
- + }
- + }
- +
- + return TRUE;
- +}
- +
- +/**
- + * glnx_dirfd_iterator_clear:
- + * @dfd_iter: Iterator, will be de-initialized
- + *
- + * Unset @dfd_iter, freeing any resources. If @dfd_iter is not
- + * initialized, do nothing.
- + */
- +void
- +glnx_dirfd_iterator_clear (GLnxDirFdIterator *dfd_iter)
- +{
- + GLnxRealDirfdIterator *real_dfd_iter = (GLnxRealDirfdIterator*) dfd_iter;
- + /* fd is owned by dfd_iter */
- + if (!real_dfd_iter->initialized)
- + return;
- + (void) closedir (real_dfd_iter->d);
- + real_dfd_iter->initialized = FALSE;
- +}
- +
- +/**
- + * glnx_fdrel_abspath:
- + * @dfd: Directory fd
- + * @path: Path
- + *
- + * Turn a fd-relative pair into something that can be used for legacy
- + * APIs expecting absolute paths.
- + *
- + * This is Linux specific, and only valid inside this process (unless
- + * you set up the child process to have the exact same fd number, but
- + * don't try that).
- + */
- +char *
- +glnx_fdrel_abspath (int dfd,
- + const char *path)
- +{
- + dfd = glnx_dirfd_canonicalize (dfd);
- + if (dfd == AT_FDCWD)
- + return g_strdup (path);
- + return g_strdup_printf ("/proc/self/fd/%d/%s", dfd, path);
- +}
- +
- +/**
- + * glnx_gen_temp_name:
- + * @tmpl: (type filename): template directory name, the last 6 characters will be replaced
- + *
- + * Replace the last 6 characters of @tmpl with random ASCII. You must
- + * use this in combination with a mechanism to ensure race-free file
- + * creation such as `O_EXCL`.
- + */
- +void
- +glnx_gen_temp_name (gchar *tmpl)
- +{
- + g_return_if_fail (tmpl != NULL);
- + const size_t len = strlen (tmpl);
- + g_return_if_fail (len >= 6);
- +
- + static const char letters[] =
- + "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789";
- + static const int NLETTERS = sizeof (letters) - 1;
- +
- + char *XXXXXX = tmpl + (len - 6);
- + for (int i = 0; i < 6; i++)
- + XXXXXX[i] = letters[g_random_int_range(0, NLETTERS)];
- +}
- +
- +/**
- + * glnx_mkdtempat:
- + * @dfd: Directory fd
- + * @tmpl: (type filename): Initial template directory name, last 6 characters will be replaced
- + * @mode: permissions with which to create the temporary directory
- + * @out_tmpdir: (out caller-allocates): Initialized tempdir structure
- + * @error: Error
- + *
- + * Somewhat similar to g_mkdtemp_full(), but fd-relative, and returns a
- + * structure that uses autocleanups. Note that the supplied @dfd lifetime
- + * must match or exceed that of @out_tmpdir in order to remove the directory.
- + */
- +gboolean
- +glnx_mkdtempat (int dfd, const char *tmpl, int mode,
- + GLnxTmpDir *out_tmpdir, GError **error)
- +{
- + g_return_val_if_fail (tmpl != NULL, FALSE);
- + g_return_val_if_fail (out_tmpdir != NULL, FALSE);
- + g_return_val_if_fail (!out_tmpdir->initialized, FALSE);
- +
- + dfd = glnx_dirfd_canonicalize (dfd);
- +
- + g_autofree char *path = g_strdup (tmpl);
- + for (int count = 0; count < 100; count++)
- + {
- + glnx_gen_temp_name (path);
- +
- + /* Ideally we could use openat(O_DIRECTORY | O_CREAT | O_EXCL) here
- + * to create and open the directory atomically, but that’s not supported by
- + * current kernel versions: http://www.openwall.com/lists/oss-security/2014/11/26/14
- + * (Tested on kernel 4.10.10-200.fc25.x86_64). For the moment, accept a
- + * TOCTTOU race here. */
- + if (mkdirat (dfd, path, mode) == -1)
- + {
- + if (errno == EEXIST)
- + continue;
- +
- + /* Any other error will apply also to other names we might
- + * try, and there are 2^32 or so of them, so give up now.
- + */
- + return glnx_throw_errno_prefix (error, "mkdirat");
- + }
- +
- + /* And open it */
- + glnx_autofd int ret_dfd = -1;
- + if (!glnx_opendirat (dfd, path, FALSE, &ret_dfd, error))
- + {
- + /* If we fail to open, let's try to clean up */
- + (void)unlinkat (dfd, path, AT_REMOVEDIR);
- + return FALSE;
- + }
- +
- + /* Return the initialized directory struct */
- + out_tmpdir->initialized = TRUE;
- + out_tmpdir->src_dfd = dfd; /* referenced; see above docs */
- + out_tmpdir->fd = glnx_steal_fd (&ret_dfd);
- + out_tmpdir->path = g_steal_pointer (&path);
- + return TRUE;
- + }
- +
- + /* Failure */
- + g_set_error (error, G_IO_ERROR, G_IO_ERROR_EXISTS,
- + "glnx_mkdtempat ran out of combinations to try");
- + return FALSE;
- +}
- +
- +/**
- + * glnx_mkdtemp:
- + * @tmpl: (type filename): Source template directory name, last 6 characters will be replaced
- + * @mode: permissions to create the temporary directory with
- + * @out_tmpdir: (out caller-allocates): Return location for tmpdir data
- + * @error: Return location for a #GError, or %NULL
- + *
- + * Similar to glnx_mkdtempat(), but will use g_get_tmp_dir() as the parent
- + * directory to @tmpl.
- + *
- + * Returns: %TRUE on success, %FALSE otherwise
- + * Since: UNRELEASED
- + */
- +gboolean
- +glnx_mkdtemp (const gchar *tmpl,
- + int mode,
- + GLnxTmpDir *out_tmpdir,
- + GError **error)
- +{
- + g_autofree char *path = g_build_filename (g_get_tmp_dir (), tmpl, NULL);
- + return glnx_mkdtempat (AT_FDCWD, path, mode,
- + out_tmpdir, error);
- +}
- +
- +static gboolean
- +_glnx_tmpdir_free (GLnxTmpDir *tmpd,
- + gboolean delete_dir,
- + GCancellable *cancellable,
- + GError **error)
- +{
- + /* Support being passed NULL so we work nicely in a GPtrArray */
- + if (!(tmpd && tmpd->initialized))
- + return TRUE;
- + g_assert_cmpint (tmpd->fd, !=, -1);
- + glnx_close_fd (&tmpd->fd);
- + g_assert (tmpd->path);
- + g_assert_cmpint (tmpd->src_dfd, !=, -1);
- + g_autofree char *path = tmpd->path; /* Take ownership */
- + tmpd->initialized = FALSE;
- + if (delete_dir)
- + {
- + if (!glnx_shutil_rm_rf_at (tmpd->src_dfd, path, cancellable, error))
- + return FALSE;
- + }
- + return TRUE;
- +}
- +
- +/**
- + * glnx_tmpdir_delete:
- + * @tmpf: Temporary dir
- + * @cancellable: Cancellable
- + * @error: Error
- + *
- + * Deallocate a tmpdir, closing the fd and recursively deleting the path. This
- + * is normally called indirectly via glnx_tmpdir_cleanup() by the autocleanup
- + * attribute, but you can also invoke this directly.
- + *
- + * If an error occurs while deleting the filesystem path, @tmpf will still have
- + * been deallocated and should not be reused.
- + *
- + * See also `glnx_tmpdir_unset` to avoid deleting the path.
- + */
- +gboolean
- +glnx_tmpdir_delete (GLnxTmpDir *tmpf, GCancellable *cancellable, GError **error)
- +{
- + return _glnx_tmpdir_free (tmpf, TRUE, cancellable, error);
- +}
- +
- +/**
- + * glnx_tmpdir_unset:
- + * @tmpf: Temporary dir
- + * @cancellable: Cancellable
- + * @error: Error
- + *
- + * Deallocate a tmpdir, but do not delete the filesystem path. See also
- + * `glnx_tmpdir_delete()`.
- + */
- +void
- +glnx_tmpdir_unset (GLnxTmpDir *tmpf)
- +{
- + (void) _glnx_tmpdir_free (tmpf, FALSE, NULL, NULL);
- +}
- diff -Nuar ostree-2018.8.orig/libglnx/glnx-dirfd.h ostree-2018.8/libglnx/glnx-dirfd.h
- --- ostree-2018.8.orig/libglnx/glnx-dirfd.h 1970-01-01 02:00:00.000000000 +0200
- +++ ostree-2018.8/libglnx/glnx-dirfd.h 2018-05-26 00:37:32.006018846 +0300
- @@ -0,0 +1,137 @@
- +/* -*- mode: C; c-file-style: "gnu"; indent-tabs-mode: nil; -*-
- + *
- + * Copyright (C) 2014,2015 Colin Walters <walters@verbum.org>.
- + *
- + * This library is free software; you can redistribute it and/or
- + * modify it under the terms of the GNU Lesser General Public
- + * License as published by the Free Software Foundation; either
- + * version 2 of the License, or (at your option) any later version.
- + *
- + * This library is distributed in the hope that it will be useful,
- + * but WITHOUT ANY WARRANTY; without even the implied warranty of
- + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
- + * Lesser General Public License for more details.
- + *
- + * You should have received a copy of the GNU Lesser General Public
- + * License along with this library; if not, write to the
- + * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
- + * Boston, MA 02111-1307, USA.
- + */
- +
- +#pragma once
- +
- +#include <glnx-backport-autocleanups.h>
- +#include <glnx-macros.h>
- +#include <glnx-errors.h>
- +#include <limits.h>
- +#include <dirent.h>
- +#include <sys/stat.h>
- +#include <fcntl.h>
- +
- +G_BEGIN_DECLS
- +
- +/**
- + * glnx_dirfd_canonicalize:
- + * @fd: A directory file descriptor
- + *
- + * It's often convenient in programs to use `-1` for "unassigned fd",
- + * and also because gobject-introspection doesn't support `AT_FDCWD`,
- + * libglnx honors `-1` to mean `AT_FDCWD`. This small inline function
- + * canonicalizes `-1 -> AT_FDCWD`.
- + */
- +static inline int
- +glnx_dirfd_canonicalize (int fd)
- +{
- + if (fd == -1)
- + return AT_FDCWD;
- + return fd;
- +}
- +
- +struct GLnxDirFdIterator {
- + gboolean initialized;
- + int fd;
- + gpointer padding_data[4];
- +};
- +
- +typedef struct GLnxDirFdIterator GLnxDirFdIterator;
- +gboolean glnx_dirfd_iterator_init_at (int dfd, const char *path,
- + gboolean follow,
- + GLnxDirFdIterator *dfd_iter, GError **error);
- +gboolean glnx_dirfd_iterator_init_take_fd (int *dfd, GLnxDirFdIterator *dfd_iter, GError **error);
- +gboolean glnx_dirfd_iterator_next_dent (GLnxDirFdIterator *dfd_iter,
- + struct dirent **out_dent,
- + GCancellable *cancellable,
- + GError **error);
- +gboolean glnx_dirfd_iterator_next_dent_ensure_dtype (GLnxDirFdIterator *dfd_iter,
- + struct dirent **out_dent,
- + GCancellable *cancellable,
- + GError **error);
- +void glnx_dirfd_iterator_clear (GLnxDirFdIterator *dfd_iter);
- +
- +G_DEFINE_AUTO_CLEANUP_CLEAR_FUNC(GLnxDirFdIterator, glnx_dirfd_iterator_clear)
- +
- +int glnx_opendirat_with_errno (int dfd,
- + const char *path,
- + gboolean follow);
- +
- +gboolean glnx_opendirat (int dfd,
- + const char *path,
- + gboolean follow,
- + int *out_fd,
- + GError **error);
- +
- +char *glnx_fdrel_abspath (int dfd,
- + const char *path);
- +
- +void glnx_gen_temp_name (gchar *tmpl);
- +
- +/**
- + * glnx_ensure_dir:
- + * @dfd: directory fd
- + * @path: Directory path
- + * @mode: Mode
- + * @error: Return location for a #GError, or %NULL
- + *
- + * Wrapper around mkdirat() which adds #GError support, ensures that
- + * it retries on %EINTR, and also ignores `EEXIST`.
- + *
- + * See also `glnx_shutil_mkdir_p_at()` for recursive handling.
- + *
- + * Returns: %TRUE on success, %FALSE otherwise
- + */
- +static inline gboolean
- +glnx_ensure_dir (int dfd,
- + const char *path,
- + mode_t mode,
- + GError **error)
- +{
- + if (TEMP_FAILURE_RETRY (mkdirat (dfd, path, mode)) != 0)
- + {
- + if (G_UNLIKELY (errno != EEXIST))
- + return glnx_throw_errno_prefix (error, "mkdirat(%s)", path);
- + }
- + return TRUE;
- +}
- +
- +typedef struct {
- + gboolean initialized;
- + int src_dfd;
- + int fd;
- + char *path;
- +} GLnxTmpDir;
- +gboolean glnx_tmpdir_delete (GLnxTmpDir *tmpf, GCancellable *cancellable, GError **error);
- +void glnx_tmpdir_unset (GLnxTmpDir *tmpf);
- +static inline void
- +glnx_tmpdir_cleanup (GLnxTmpDir *tmpf)
- +{
- + (void)glnx_tmpdir_delete (tmpf, NULL, NULL);
- +}
- +G_DEFINE_AUTO_CLEANUP_CLEAR_FUNC(GLnxTmpDir, glnx_tmpdir_cleanup)
- +
- +gboolean glnx_mkdtempat (int dfd, const char *tmpl, int mode,
- + GLnxTmpDir *out_tmpdir, GError **error);
- +
- +gboolean glnx_mkdtemp (const char *tmpl, int mode,
- + GLnxTmpDir *out_tmpdir, GError **error);
- +
- +G_END_DECLS
- diff -Nuar ostree-2018.8.orig/libglnx/glnx-errors.c ostree-2018.8/libglnx/glnx-errors.c
- --- ostree-2018.8.orig/libglnx/glnx-errors.c 1970-01-01 02:00:00.000000000 +0200
- +++ ostree-2018.8/libglnx/glnx-errors.c 2018-05-26 00:37:32.006018846 +0300
- @@ -0,0 +1,131 @@
- +/* -*- mode: C; c-file-style: "gnu"; indent-tabs-mode: nil; -*-
- + *
- + * Copyright (C) 2014,2015 Colin Walters <walters@verbum.org>.
- + *
- + * This library is free software; you can redistribute it and/or
- + * modify it under the terms of the GNU Lesser General Public
- + * License as published by the Free Software Foundation; either
- + * version 2 of the License, or (at your option) any later version.
- + *
- + * This library is distributed in the hope that it will be useful,
- + * but WITHOUT ANY WARRANTY; without even the implied warranty of
- + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
- + * Lesser General Public License for more details.
- + *
- + * You should have received a copy of the GNU Lesser General Public
- + * License along with this library; if not, write to the
- + * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
- + * Boston, MA 02111-1307, USA.
- + */
- +
- +#include "config.h"
- +
- +#include <glnx-backport-autocleanups.h>
- +#include <glnx-errors.h>
- +
- +/* Set @error with G_IO_ERROR/G_IO_ERROR_FAILED.
- + *
- + * This function returns %FALSE so it can be used conveniently in a single
- + * statement:
- + *
- + * ```
- + * if (strcmp (foo, "somevalue") != 0)
- + * return glnx_throw (error, "key must be somevalue, not '%s'", foo);
- + * ```
- + */
- +gboolean
- +glnx_throw (GError **error,
- + const char *fmt,
- + ...)
- +{
- + if (error == NULL)
- + return FALSE;
- +
- + va_list args;
- + va_start (args, fmt);
- + GError *new = g_error_new_valist (G_IO_ERROR, G_IO_ERROR_FAILED, fmt, args);
- + va_end (args);
- + g_propagate_error (error, g_steal_pointer (&new));
- + return FALSE;
- +}
- +
- +void
- +glnx_real_set_prefix_error_va (GError *error,
- + const char *format,
- + va_list args)
- +{
- + if (error == NULL)
- + return;
- +
- + g_autofree char *old_msg = g_steal_pointer (&error->message);
- + g_autoptr(GString) buf = g_string_new ("");
- + g_string_append_vprintf (buf, format, args);
- + g_string_append (buf, ": ");
- + g_string_append (buf, old_msg);
- + error->message = g_string_free (g_steal_pointer (&buf), FALSE);
- +}
- +
- +/* Prepend to @error's message by `$prefix: ` where `$prefix` is computed via
- + * printf @fmt. Returns %FALSE so it can be used conveniently in a single
- + * statement:
- + *
- + * ```
- + * if (!function_that_fails (s, error))
- + * return glnx_throw_prefix (error, "while handling '%s'", s);
- + * ```
- + * */
- +gboolean
- +glnx_prefix_error (GError **error,
- + const char *fmt,
- + ...)
- +{
- + if (error == NULL)
- + return FALSE;
- +
- + va_list args;
- + va_start (args, fmt);
- + glnx_real_set_prefix_error_va (*error, fmt, args);
- + va_end (args);
- + return FALSE;
- +}
- +
- +void
- +glnx_real_set_prefix_error_from_errno_va (GError **error,
- + gint errsv,
- + const char *format,
- + va_list args)
- +{
- + if (!error)
- + return;
- +
- + g_set_error_literal (error,
- + G_IO_ERROR,
- + g_io_error_from_errno (errsv),
- + g_strerror (errsv));
- + glnx_real_set_prefix_error_va (*error, format, args);
- +}
- +
- +/* Set @error using the value of `$prefix: g_strerror (errno)` where `$prefix`
- + * is computed via printf @fmt.
- + *
- + * This function returns %FALSE so it can be used conveniently in a single
- + * statement:
- + *
- + * ```
- + * return glnx_throw_errno_prefix (error, "unlinking %s", pathname);
- + * ```
- + */
- +gboolean
- +glnx_throw_errno_prefix (GError **error,
- + const char *fmt,
- + ...)
- +{
- + int errsv = errno;
- + va_list args;
- + va_start (args, fmt);
- + glnx_real_set_prefix_error_from_errno_va (error, errsv, fmt, args);
- + va_end (args);
- + /* See comment in glnx_throw_errno() about preserving errno */
- + errno = errsv;
- + return FALSE;
- +}
- diff -Nuar ostree-2018.8.orig/libglnx/glnx-errors.h ostree-2018.8/libglnx/glnx-errors.h
- --- ostree-2018.8.orig/libglnx/glnx-errors.h 1970-01-01 02:00:00.000000000 +0200
- +++ ostree-2018.8/libglnx/glnx-errors.h 2018-05-26 00:37:32.006018846 +0300
- @@ -0,0 +1,134 @@
- +/* -*- mode: C; c-file-style: "gnu"; indent-tabs-mode: nil; -*-
- + *
- + * Copyright (C) 2014,2015 Colin Walters <walters@verbum.org>.
- + *
- + * This library is free software; you can redistribute it and/or
- + * modify it under the terms of the GNU Lesser General Public
- + * License as published by the Free Software Foundation; either
- + * version 2 of the License, or (at your option) any later version.
- + *
- + * This library is distributed in the hope that it will be useful,
- + * but WITHOUT ANY WARRANTY; without even the implied warranty of
- + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
- + * Lesser General Public License for more details.
- + *
- + * You should have received a copy of the GNU Lesser General Public
- + * License along with this library; if not, write to the
- + * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
- + * Boston, MA 02111-1307, USA.
- + */
- +
- +#pragma once
- +
- +#include <glnx-backport-autocleanups.h>
- +#include <errno.h>
- +
- +G_BEGIN_DECLS
- +
- +gboolean glnx_throw (GError **error, const char *fmt, ...) G_GNUC_PRINTF (2,3);
- +
- +/* Like `glnx_throw ()`, but returns %NULL. */
- +#define glnx_null_throw(error, args...) \
- + ({glnx_throw (error, args); NULL;})
- +
- +/* Implementation detail of glnx_throw_prefix() */
- +void glnx_real_set_prefix_error_va (GError *error,
- + const char *format,
- + va_list args) G_GNUC_PRINTF (2,0);
- +
- +gboolean glnx_prefix_error (GError **error, const char *fmt, ...) G_GNUC_PRINTF (2,3);
- +
- +/* Like `glnx_prefix_error ()`, but returns %NULL. */
- +#define glnx_prefix_error_null(error, args...) \
- + ({glnx_prefix_error (error, args); NULL;})
- +
- +/**
- + * GLNX_AUTO_PREFIX_ERROR:
- + *
- + * An autocleanup-based macro to automatically call `g_prefix_error()` (also with a colon+space `: `)
- + * when it goes out of scope. This is useful when one wants error strings built up by the callee
- + * function, not all callers.
- + *
- + * ```
- + * gboolean start_http_request (..., GError **error)
- + * {
- + * GLNX_AUTO_PREFIX_ERROR ("HTTP request", error)
- + *
- + * if (!libhttp_request_start (..., error))
- + * return FALSE;
- + * ...
- + * return TRUE;
- + * ```
- + */
- +typedef struct {
- + const char *prefix;
- + GError **error;
- +} GLnxAutoErrorPrefix;
- +static inline void
- +glnx_cleanup_auto_prefix_error (GLnxAutoErrorPrefix *prefix)
- +{
- + if (prefix->error && *(prefix->error))
- + g_prefix_error (prefix->error, "%s: ", prefix->prefix);
- +}
- +G_DEFINE_AUTO_CLEANUP_CLEAR_FUNC(GLnxAutoErrorPrefix, glnx_cleanup_auto_prefix_error)
- +#define GLNX_AUTO_PREFIX_ERROR(text, error) \
- + G_GNUC_UNUSED g_auto(GLnxAutoErrorPrefix) _GLNX_MAKE_ANONYMOUS(_glnxautoprefixerror_) = { text, error }
- +
- +/* Set @error using the value of `g_strerror (errno)`.
- + *
- + * This function returns %FALSE so it can be used conveniently in a single
- + * statement:
- + *
- + * ```
- + * if (unlinkat (fd, somepathname) < 0)
- + * return glnx_throw_errno (error);
- + * ```
- + */
- +static inline gboolean
- +glnx_throw_errno (GError **error)
- +{
- + /* Save the value of errno, in case one of the
- + * intermediate function calls happens to set it.
- + */
- + int errsv = errno;
- + g_set_error_literal (error, G_IO_ERROR,
- + g_io_error_from_errno (errsv),
- + g_strerror (errsv));
- + /* We also restore the value of errno, since that's
- + * what was done in a long-ago libgsystem commit
- + * https://git.gnome.org/browse/libgsystem/commit/?id=ed106741f7a0596dc8b960b31fdae671d31d666d
- + * but I certainly can't remember now why I did that.
- + */
- + errno = errsv;
- + return FALSE;
- +}
- +
- +/* Like glnx_throw_errno(), but yields a NULL pointer. */
- +#define glnx_null_throw_errno(error) \
- + ({glnx_throw_errno (error); NULL;})
- +
- +/* Implementation detail of glnx_throw_errno_prefix() */
- +void glnx_real_set_prefix_error_from_errno_va (GError **error,
- + gint errsv,
- + const char *format,
- + va_list args) G_GNUC_PRINTF (3,0);
- +
- +gboolean glnx_throw_errno_prefix (GError **error, const char *fmt, ...) G_GNUC_PRINTF (2,3);
- +
- +/* Like glnx_throw_errno_prefix(), but yields a NULL pointer. */
- +#define glnx_null_throw_errno_prefix(error, args...) \
- + ({glnx_throw_errno_prefix (error, args); NULL;})
- +
- +/* BEGIN LEGACY APIS */
- +
- +#define glnx_set_error_from_errno(error) \
- + do { \
- + glnx_throw_errno (error); \
- + } while (0);
- +
- +#define glnx_set_prefix_error_from_errno(error, format, args...) \
- + do { \
- + glnx_throw_errno_prefix (error, format, args); \
- + } while (0);
- +
- +G_END_DECLS
- diff -Nuar ostree-2018.8.orig/libglnx/glnx-fdio.c ostree-2018.8/libglnx/glnx-fdio.c
- --- ostree-2018.8.orig/libglnx/glnx-fdio.c 1970-01-01 02:00:00.000000000 +0200
- +++ ostree-2018.8/libglnx/glnx-fdio.c 2018-05-26 00:37:32.007018846 +0300
- @@ -0,0 +1,1106 @@
- +/* -*- mode: C; c-file-style: "gnu"; indent-tabs-mode: nil; -*-
- + *
- + * Copyright (C) 2014,2015 Colin Walters <walters@verbum.org>.
- + *
- + * Portions derived from systemd:
- + * Copyright 2010 Lennart Poettering
- + *
- + * This library is free software; you can redistribute it and/or
- + * modify it under the terms of the GNU Lesser General Public
- + * License as published by the Free Software Foundation; either
- + * version 2 of the License, or (at your option) any later version.
- + *
- + * This library is distributed in the hope that it will be useful,
- + * but WITHOUT ANY WARRANTY; without even the implied warranty of
- + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
- + * Lesser General Public License for more details.
- + *
- + * You should have received a copy of the GNU Lesser General Public
- + * License along with this library; if not, write to the
- + * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
- + * Boston, MA 02111-1307, USA.
- + */
- +
- +#include "config.h"
- +
- +#include <string.h>
- +#include <stdio.h>
- +#include <stdlib.h>
- +#include <stdint.h>
- +#include <stdbool.h>
- +#include <sys/ioctl.h>
- +#include <sys/sendfile.h>
- +#include <errno.h>
- +
- +#include <glnx-fdio.h>
- +#include <glnx-dirfd.h>
- +#include <glnx-errors.h>
- +#include <glnx-xattrs.h>
- +#include <glnx-backport-autoptr.h>
- +#include <glnx-local-alloc.h>
- +#include <glnx-missing.h>
- +
- +/* The standardized version of BTRFS_IOC_CLONE */
- +#ifndef FICLONE
- +#define FICLONE _IOW(0x94, 9, int)
- +#endif
- +
- +/* Returns the number of chars needed to format variables of the
- + * specified type as a decimal string. Adds in extra space for a
- + * negative '-' prefix (hence works correctly on signed
- + * types). Includes space for the trailing NUL. */
- +#define DECIMAL_STR_MAX(type) \
- + (2+(sizeof(type) <= 1 ? 3 : \
- + sizeof(type) <= 2 ? 5 : \
- + sizeof(type) <= 4 ? 10 : \
- + sizeof(type) <= 8 ? 20 : sizeof(int[-2*(sizeof(type) > 8)])))
- +
- +gboolean
- +glnx_stdio_file_flush (FILE *f, GError **error)
- +{
- + if (fflush (f) != 0)
- + return glnx_throw_errno_prefix (error, "fflush");
- + if (ferror (f) != 0)
- + return glnx_throw_errno_prefix (error, "ferror");
- + return TRUE;
- +}
- +
- +/* An implementation of renameat2(..., RENAME_NOREPLACE)
- + * with fallback to a non-atomic version.
- + */
- +int
- +glnx_renameat2_noreplace (int olddirfd, const char *oldpath,
- + int newdirfd, const char *newpath)
- +{
- +#ifndef ENABLE_WRPSEUDO_COMPAT
- + if (renameat2 (olddirfd, oldpath, newdirfd, newpath, RENAME_NOREPLACE) < 0)
- + {
- + if (G_IN_SET(errno, EINVAL, ENOSYS))
- + {
- + /* Fall through */
- + }
- + else
- + {
- + return -1;
- + }
- + }
- + else
- + return TRUE;
- +#endif
- +
- + if (linkat (olddirfd, oldpath, newdirfd, newpath, 0) < 0)
- + return -1;
- +
- + if (unlinkat (olddirfd, oldpath, 0) < 0)
- + return -1;
- +
- + return 0;
- +}
- +
- +static gboolean
- +rename_file_noreplace_at (int olddirfd, const char *oldpath,
- + int newdirfd, const char *newpath,
- + gboolean ignore_eexist,
- + GError **error)
- +{
- + if (glnx_renameat2_noreplace (olddirfd, oldpath,
- + newdirfd, newpath) < 0)
- + {
- + if (errno == EEXIST && ignore_eexist)
- + {
- + (void) unlinkat (olddirfd, oldpath, 0);
- + return TRUE;
- + }
- + else
- + return glnx_throw_errno_prefix (error, "renameat");
- + }
- + return TRUE;
- +}
- +
- +/* An implementation of renameat2(..., RENAME_EXCHANGE)
- + * with fallback to a non-atomic version.
- + */
- +int
- +glnx_renameat2_exchange (int olddirfd, const char *oldpath,
- + int newdirfd, const char *newpath)
- +{
- +#ifndef ENABLE_WRPSEUDO_COMPAT
- + if (renameat2 (olddirfd, oldpath, newdirfd, newpath, RENAME_EXCHANGE) == 0)
- + return 0;
- + else
- + {
- + if (G_IN_SET(errno, ENOSYS, EINVAL))
- + {
- + /* Fall through */
- + }
- + else
- + {
- + return -1;
- + }
- + }
- +#endif
- +
- + /* Fallback */
- + { char *old_tmp_name_buf = glnx_strjoina (oldpath, ".XXXXXX");
- + /* This obviously isn't race-free, but doing better gets tricky, since if
- + * we're here the kernel isn't likely to support RENAME_NOREPLACE either.
- + * Anyways, upgrade the kernel. Failing that, avoid use of this function in
- + * shared subdirectories like /tmp.
- + */
- + glnx_gen_temp_name (old_tmp_name_buf);
- + const char *old_tmp_name = old_tmp_name_buf;
- +
- + /* Move old out of the way */
- + if (renameat (olddirfd, oldpath, olddirfd, old_tmp_name) < 0)
- + return -1;
- + /* Now move new into its place */
- + if (renameat (newdirfd, newpath, olddirfd, oldpath) < 0)
- + return -1;
- + /* And finally old(tmp) into new */
- + if (renameat (olddirfd, old_tmp_name, newdirfd, newpath) < 0)
- + return -1;
- + }
- + return 0;
- +}
- +
- +/* Deallocate a tmpfile, closing the fd and deleting the path, if any. This is
- + * normally called by default by the autocleanup attribute, but you can also
- + * invoke this directly.
- + */
- +void
- +glnx_tmpfile_clear (GLnxTmpfile *tmpf)
- +{
- + /* Support being passed NULL so we work nicely in a GPtrArray */
- + if (!tmpf)
- + return;
- + if (!tmpf->initialized)
- + return;
- + glnx_close_fd (&tmpf->fd);
- + /* If ->path is set, we're likely aborting due to an error. Clean it up */
- + if (tmpf->path)
- + {
- + (void) unlinkat (tmpf->src_dfd, tmpf->path, 0);
- + g_free (tmpf->path);
- + }
- + tmpf->initialized = FALSE;
- +}
- +
- +static gboolean
- +open_tmpfile_core (int dfd, const char *subpath,
- + int flags,
- + GLnxTmpfile *out_tmpf,
- + GError **error)
- +{
- + /* Picked this to match mkstemp() */
- + const guint mode = 0600;
- +
- + dfd = glnx_dirfd_canonicalize (dfd);
- +
- + /* Creates a temporary file, that shall be renamed to "target"
- + * later. If possible, this uses O_TMPFILE – in which case
- + * "ret_path" will be returned as NULL. If not possible a the
- + * tempoary path name used is returned in "ret_path". Use
- + * link_tmpfile() below to rename the result after writing the file
- + * in full. */
- +#if defined(O_TMPFILE) && !defined(DISABLE_OTMPFILE) && !defined(ENABLE_WRPSEUDO_COMPAT)
- + {
- + glnx_autofd int fd = openat (dfd, subpath, O_TMPFILE|flags, mode);
- + if (fd == -1 && !(G_IN_SET(errno, ENOSYS, EISDIR, EOPNOTSUPP)))
- + return glnx_throw_errno_prefix (error, "open(O_TMPFILE)");
- + if (fd != -1)
- + {
- + /* Workaround for https://sourceware.org/bugzilla/show_bug.cgi?id=17523
- + * See also https://github.com/ostreedev/ostree/issues/991
- + */
- + if (fchmod (fd, mode) < 0)
- + return glnx_throw_errno_prefix (error, "fchmod");
- + out_tmpf->initialized = TRUE;
- + out_tmpf->src_dfd = dfd; /* Copied; caller must keep open */
- + out_tmpf->fd = glnx_steal_fd (&fd);
- + out_tmpf->path = NULL;
- + return TRUE;
- + }
- + }
- + /* Fallthrough */
- +#endif
- +
- + const guint count_max = 100;
- + { g_autofree char *tmp = g_strconcat (subpath, "/tmp.XXXXXX", NULL);
- +
- + for (int count = 0; count < count_max; count++)
- + {
- + glnx_gen_temp_name (tmp);
- +
- + glnx_autofd int fd = openat (dfd, tmp, O_CREAT|O_EXCL|O_NOFOLLOW|O_NOCTTY|flags, mode);
- + if (fd < 0)
- + {
- + if (errno == EEXIST)
- + continue;
- + else
- + return glnx_throw_errno_prefix (error, "Creating temp file");
- + }
- + else
- + {
- + out_tmpf->initialized = TRUE;
- + out_tmpf->src_dfd = dfd; /* Copied; caller must keep open */
- + out_tmpf->fd = glnx_steal_fd (&fd);
- + out_tmpf->path = g_steal_pointer (&tmp);
- + return TRUE;
- + }
- + }
- + }
- + g_set_error (error, G_IO_ERROR, G_IO_ERROR_EXISTS,
- + "Exhausted %u attempts to create temporary file", count_max);
- + return FALSE;
- +}
- +
- +/* Allocate a temporary file, using Linux O_TMPFILE if available. The file mode
- + * will be 0600.
- + *
- + * The result will be stored in @out_tmpf, which is caller allocated
- + * so you can store it on the stack in common scenarios.
- + *
- + * The directory fd @dfd must live at least as long as the output @out_tmpf.
- + */
- +gboolean
- +glnx_open_tmpfile_linkable_at (int dfd,
- + const char *subpath,
- + int flags,
- + GLnxTmpfile *out_tmpf,
- + GError **error)
- +{
- + /* Don't allow O_EXCL, as that has a special meaning for O_TMPFILE;
- + * it's used for glnx_open_anonymous_tmpfile().
- + */
- + g_return_val_if_fail ((flags & O_EXCL) == 0, FALSE);
- +
- + return open_tmpfile_core (dfd, subpath, flags, out_tmpf, error);
- +}
- +
- +/* A variant of `glnx_open_tmpfile_linkable_at()` which doesn't support linking.
- + * Useful for true temporary storage. The fd will be allocated in /var/tmp to
- + * ensure maximum storage space.
- + */
- +gboolean
- +glnx_open_anonymous_tmpfile (int flags,
- + GLnxTmpfile *out_tmpf,
- + GError **error)
- +{
- + /* Add in O_EXCL */
- + if (!open_tmpfile_core (AT_FDCWD, "/var/tmp", flags | O_EXCL, out_tmpf, error))
- + return FALSE;
- + if (out_tmpf->path)
- + {
- + (void) unlinkat (out_tmpf->src_dfd, out_tmpf->path, 0);
- + g_clear_pointer (&out_tmpf->path, g_free);
- + }
- + out_tmpf->anonymous = TRUE;
- + out_tmpf->src_dfd = -1;
- + return TRUE;
- +}
- +
- +/* Use this after calling glnx_open_tmpfile_linkable_at() to give
- + * the file its final name (link into place).
- + */
- +gboolean
- +glnx_link_tmpfile_at (GLnxTmpfile *tmpf,
- + GLnxLinkTmpfileReplaceMode mode,
- + int target_dfd,
- + const char *target,
- + GError **error)
- +{
- + const gboolean replace = (mode == GLNX_LINK_TMPFILE_REPLACE);
- + const gboolean ignore_eexist = (mode == GLNX_LINK_TMPFILE_NOREPLACE_IGNORE_EXIST);
- +
- + g_return_val_if_fail (!tmpf->anonymous, FALSE);
- + g_return_val_if_fail (tmpf->fd >= 0, FALSE);
- + g_return_val_if_fail (tmpf->src_dfd == AT_FDCWD || tmpf->src_dfd >= 0, FALSE);
- +
- + /* Unlike the original systemd code, this function also supports
- + * replacing existing files.
- + */
- +
- + /* We have `tmpfile_path` for old systems without O_TMPFILE. */
- + if (tmpf->path)
- + {
- + if (replace)
- + {
- + /* We have a regular tempfile, we're overwriting - this is a
- + * simple renameat().
- + */
- + if (renameat (tmpf->src_dfd, tmpf->path, target_dfd, target) < 0)
- + return glnx_throw_errno_prefix (error, "renameat");
- + }
- + else
- + {
- + /* We need to use renameat2(..., NOREPLACE) or emulate it */
- + if (!rename_file_noreplace_at (tmpf->src_dfd, tmpf->path, target_dfd, target,
- + ignore_eexist,
- + error))
- + return FALSE;
- + }
- + /* Now, clear the pointer so we don't try to unlink it */
- + g_clear_pointer (&tmpf->path, g_free);
- + }
- + else
- + {
- + /* This case we have O_TMPFILE, so our reference to it is via /proc/self/fd */
- + char proc_fd_path[strlen("/proc/self/fd/") + DECIMAL_STR_MAX(tmpf->fd) + 1];
- +
- + sprintf (proc_fd_path, "/proc/self/fd/%i", tmpf->fd);
- +
- + if (replace)
- + {
- + /* In this case, we had our temp file atomically hidden, but now
- + * we need to make it visible in the FS so we can do a rename.
- + * Ideally, linkat() would gain AT_REPLACE or so.
- + */
- + /* TODO - avoid double alloca, we can just alloca a copy of
- + * the pathname plus space for tmp.XXXXX */
- + char *dnbuf = strdupa (target);
- + const char *dn = dirname (dnbuf);
- + char *tmpname_buf = glnx_strjoina (dn, "/tmp.XXXXXX");
- +
- + const guint count_max = 100;
- + guint count;
- + for (count = 0; count < count_max; count++)
- + {
- + glnx_gen_temp_name (tmpname_buf);
- +
- + if (linkat (AT_FDCWD, proc_fd_path, target_dfd, tmpname_buf, AT_SYMLINK_FOLLOW) < 0)
- + {
- + if (errno == EEXIST)
- + continue;
- + else
- + return glnx_throw_errno_prefix (error, "linkat");
- + }
- + else
- + break;
- + }
- + if (count == count_max)
- + {
- + g_set_error (error, G_IO_ERROR, G_IO_ERROR_EXISTS,
- + "Exhausted %u attempts to create temporary file", count);
- + return FALSE;
- + }
- + if (!glnx_renameat (target_dfd, tmpname_buf, target_dfd, target, error))
- + {
- + /* This is currently the only case where we need to have
- + * a cleanup unlinkat() still with O_TMPFILE.
- + */
- + (void) unlinkat (target_dfd, tmpname_buf, 0);
- + return FALSE;
- + }
- + }
- + else
- + {
- + if (linkat (AT_FDCWD, proc_fd_path, target_dfd, target, AT_SYMLINK_FOLLOW) < 0)
- + {
- + if (errno == EEXIST && mode == GLNX_LINK_TMPFILE_NOREPLACE_IGNORE_EXIST)
- + ;
- + else
- + return glnx_throw_errno_prefix (error, "linkat");
- + }
- + }
- +
- + }
- + return TRUE;
- +}
- +
- +/**
- + * glnx_openat_rdonly:
- + * @dfd: File descriptor for origin directory
- + * @path: Pathname, relative to @dfd
- + * @follow: Whether or not to follow symbolic links in the final component
- + * @out_fd: (out): File descriptor
- + * @error: Error
- + *
- + * Use openat() to open a file, with flags `O_RDONLY | O_CLOEXEC | O_NOCTTY`.
- + * Like the other libglnx wrappers, will use `TEMP_FAILURE_RETRY` and
- + * also includes @path in @error in case of failure.
- + */
- +gboolean
- +glnx_openat_rdonly (int dfd,
- + const char *path,
- + gboolean follow,
- + int *out_fd,
- + GError **error)
- +{
- + int flags = O_RDONLY | O_CLOEXEC | O_NOCTTY;
- + if (!follow)
- + flags |= O_NOFOLLOW;
- + int fd = TEMP_FAILURE_RETRY (openat (dfd, path, flags));
- + if (fd == -1)
- + return glnx_throw_errno_prefix (error, "openat(%s)", path);
- + *out_fd = fd;
- + return TRUE;
- +}
- +
- +static guint8*
- +glnx_fd_readall_malloc (int fd,
- + gsize *out_len,
- + gboolean nul_terminate,
- + GCancellable *cancellable,
- + GError **error)
- +{
- + const guint maxreadlen = 4096;
- +
- + struct stat stbuf;
- + if (!glnx_fstat (fd, &stbuf, error))
- + return FALSE;
- +
- + gsize buf_allocated;
- + if (S_ISREG (stbuf.st_mode) && stbuf.st_size > 0)
- + buf_allocated = stbuf.st_size;
- + else
- + buf_allocated = 16;
- +
- + g_autofree guint8* buf = g_malloc (buf_allocated);
- +
- + gsize buf_size = 0;
- + while (TRUE)
- + {
- + gsize readlen = MIN (buf_allocated - buf_size, maxreadlen);
- +
- + if (g_cancellable_set_error_if_cancelled (cancellable, error))
- + return FALSE;
- +
- + gssize bytes_read;
- + do
- + bytes_read = read (fd, buf + buf_size, readlen);
- + while (G_UNLIKELY (bytes_read == -1 && errno == EINTR));
- + if (G_UNLIKELY (bytes_read == -1))
- + return glnx_null_throw_errno (error);
- + if (bytes_read == 0)
- + break;
- +
- + buf_size += bytes_read;
- + if (buf_allocated - buf_size < maxreadlen)
- + buf = g_realloc (buf, buf_allocated *= 2);
- + }
- +
- + if (nul_terminate)
- + {
- + if (buf_allocated - buf_size == 0)
- + buf = g_realloc (buf, buf_allocated + 1);
- + buf[buf_size] = '\0';
- + }
- +
- + *out_len = buf_size;
- + return g_steal_pointer (&buf);
- +}
- +
- +/**
- + * glnx_fd_readall_bytes:
- + * @fd: A file descriptor
- + * @cancellable: Cancellable:
- + * @error: Error
- + *
- + * Read all data from file descriptor @fd into a #GBytes. It's
- + * recommended to only use this for small files.
- + *
- + * Returns: (transfer full): A newly allocated #GBytes
- + */
- +GBytes *
- +glnx_fd_readall_bytes (int fd,
- + GCancellable *cancellable,
- + GError **error)
- +{
- + gsize len;
- + guint8 *buf = glnx_fd_readall_malloc (fd, &len, FALSE, cancellable, error);
- + if (!buf)
- + return NULL;
- + return g_bytes_new_take (buf, len);
- +}
- +
- +/**
- + * glnx_fd_readall_utf8:
- + * @fd: A file descriptor
- + * @out_len: (out): Returned length
- + * @cancellable: Cancellable:
- + * @error: Error
- + *
- + * Read all data from file descriptor @fd, validating
- + * the result as UTF-8.
- + *
- + * Returns: (transfer full): A string validated as UTF-8, or %NULL on error.
- + */
- +char *
- +glnx_fd_readall_utf8 (int fd,
- + gsize *out_len,
- + GCancellable *cancellable,
- + GError **error)
- +{
- + gsize len;
- + g_autofree guint8 *buf = glnx_fd_readall_malloc (fd, &len, TRUE, cancellable, error);
- + if (!buf)
- + return FALSE;
- +
- + if (!g_utf8_validate ((char*)buf, len, NULL))
- + {
- + g_set_error (error,
- + G_IO_ERROR,
- + G_IO_ERROR_INVALID_DATA,
- + "Invalid UTF-8");
- + return FALSE;
- + }
- +
- + if (out_len)
- + *out_len = len;
- + return (char*)g_steal_pointer (&buf);
- +}
- +
- +/**
- + * glnx_file_get_contents_utf8_at:
- + * @dfd: Directory file descriptor
- + * @subpath: Path relative to @dfd
- + * @out_len: (out) (allow-none): Optional length
- + * @cancellable: Cancellable
- + * @error: Error
- + *
- + * Read the entire contents of the file referred
- + * to by @dfd and @subpath, validate the result as UTF-8.
- + * The length is optionally stored in @out_len.
- + *
- + * Returns: (transfer full): UTF-8 validated text, or %NULL on error
- + */
- +char *
- +glnx_file_get_contents_utf8_at (int dfd,
- + const char *subpath,
- + gsize *out_len,
- + GCancellable *cancellable,
- + GError **error)
- +{
- + dfd = glnx_dirfd_canonicalize (dfd);
- +
- + glnx_autofd int fd = -1;
- + if (!glnx_openat_rdonly (dfd, subpath, TRUE, &fd, error))
- + return NULL;
- +
- + gsize len;
- + g_autofree char *buf = glnx_fd_readall_utf8 (fd, &len, cancellable, error);
- + if (G_UNLIKELY(!buf))
- + return FALSE;
- +
- + if (out_len)
- + *out_len = len;
- + return g_steal_pointer (&buf);
- +}
- +
- +/**
- + * glnx_readlinkat_malloc:
- + * @dfd: Directory file descriptor
- + * @subpath: Subpath
- + * @cancellable: Cancellable
- + * @error: Error
- + *
- + * Read the value of a symlink into a dynamically
- + * allocated buffer.
- + */
- +char *
- +glnx_readlinkat_malloc (int dfd,
- + const char *subpath,
- + GCancellable *cancellable,
- + GError **error)
- +{
- + dfd = glnx_dirfd_canonicalize (dfd);
- +
- + size_t l = 100;
- + for (;;)
- + {
- + g_autofree char *c = g_malloc (l);
- + ssize_t n = TEMP_FAILURE_RETRY (readlinkat (dfd, subpath, c, l-1));
- + if (n < 0)
- + return glnx_null_throw_errno_prefix (error, "readlinkat");
- +
- + if ((size_t) n < l-1)
- + {
- + c[n] = 0;
- + return g_steal_pointer (&c);
- + }
- +
- + l *= 2;
- + }
- +
- + g_assert_not_reached ();
- +}
- +
- +static gboolean
- +copy_symlink_at (int src_dfd,
- + const char *src_subpath,
- + const struct stat *src_stbuf,
- + int dest_dfd,
- + const char *dest_subpath,
- + GLnxFileCopyFlags copyflags,
- + GCancellable *cancellable,
- + GError **error)
- +{
- + g_autofree char *buf = glnx_readlinkat_malloc (src_dfd, src_subpath, cancellable, error);
- + if (!buf)
- + return FALSE;
- +
- + if (TEMP_FAILURE_RETRY (symlinkat (buf, dest_dfd, dest_subpath)) != 0)
- + return glnx_throw_errno_prefix (error, "symlinkat");
- +
- + if (!(copyflags & GLNX_FILE_COPY_NOXATTRS))
- + {
- + g_autoptr(GVariant) xattrs = NULL;
- +
- + if (!glnx_dfd_name_get_all_xattrs (src_dfd, src_subpath, &xattrs,
- + cancellable, error))
- + return FALSE;
- +
- + if (!glnx_dfd_name_set_all_xattrs (dest_dfd, dest_subpath, xattrs,
- + cancellable, error))
- + return FALSE;
- + }
- +
- + if (TEMP_FAILURE_RETRY (fchownat (dest_dfd, dest_subpath,
- + src_stbuf->st_uid, src_stbuf->st_gid,
- + AT_SYMLINK_NOFOLLOW)) != 0)
- + return glnx_throw_errno_prefix (error, "fchownat");
- +
- + return TRUE;
- +}
- +
- +#define COPY_BUFFER_SIZE (16*1024)
- +
- +/* Most of the code below is from systemd, but has been reindented to GNU style,
- + * and changed to use POSIX error conventions (return -1, set errno) to more
- + * conveniently fit in with the rest of libglnx.
- + */
- +
- +/* Like write(), but loop until @nbytes are written, or an error
- + * occurs.
- + *
- + * On error, -1 is returned an @errno is set. NOTE: This is an
- + * API change from previous versions of this function.
- + */
- +int
- +glnx_loop_write(int fd, const void *buf, size_t nbytes)
- +{
- + g_return_val_if_fail (fd >= 0, -1);
- + g_return_val_if_fail (buf, -1);
- +
- + errno = 0;
- +
- + const uint8_t *p = buf;
- + while (nbytes > 0)
- + {
- + ssize_t k = write(fd, p, nbytes);
- + if (k < 0)
- + {
- + if (errno == EINTR)
- + continue;
- +
- + return -1;
- + }
- +
- + if (k == 0) /* Can't really happen */
- + {
- + errno = EIO;
- + return -1;
- + }
- +
- + p += k;
- + nbytes -= k;
- + }
- +
- + return 0;
- +}
- +
- +/* Read from @fdf until EOF, writing to @fdt. If max_bytes is -1, a full-file
- + * clone will be attempted. Otherwise Linux copy_file_range(), sendfile()
- + * syscall will be attempted. If none of those work, this function will do a
- + * plain read()/write() loop.
- + *
- + * The file descriptor @fdf must refer to a regular file.
- + *
- + * If provided, @max_bytes specifies the maximum number of bytes to read from @fdf.
- + * On error, this function returns `-1` and @errno will be set.
- + */
- +int
- +glnx_regfile_copy_bytes (int fdf, int fdt, off_t max_bytes)
- +{
- + /* Last updates from systemd as of commit 6bda23dd6aaba50cf8e3e6024248cf736cc443ca */
- + static int have_cfr = -1; /* -1 means unknown */
- + bool try_cfr = have_cfr != 0;
- + static int have_sendfile = -1; /* -1 means unknown */
- + bool try_sendfile = have_sendfile != 0;
- +
- + g_return_val_if_fail (fdf >= 0, -1);
- + g_return_val_if_fail (fdt >= 0, -1);
- + g_return_val_if_fail (max_bytes >= -1, -1);
- +
- + /* If we've requested to copy the whole range, try a full-file clone first.
- + */
- + if (max_bytes == (off_t) -1)
- + {
- + if (ioctl (fdt, FICLONE, fdf) == 0)
- + return 0;
- + /* Fall through */
- + struct stat stbuf;
- +
- + /* Gather the size so we can provide the whole thing at once to
- + * copy_file_range() or sendfile().
- + */
- + if (fstat (fdf, &stbuf) < 0)
- + return -1;
- + max_bytes = stbuf.st_size;
- + }
- +
- + while (TRUE)
- + {
- + ssize_t n;
- +
- + /* First, try copy_file_range(). Note this is an inlined version of
- + * try_copy_file_range() from systemd upstream, which works better since
- + * we use POSIX errno style.
- + */
- + if (try_cfr)
- + {
- + n = copy_file_range (fdf, NULL, fdt, NULL, max_bytes, 0u);
- + if (n < 0)
- + {
- + if (errno == ENOSYS)
- + {
- + /* No cfr in kernel, mark as permanently unavailable
- + * and fall through to sendfile().
- + */
- + have_cfr = 0;
- + try_cfr = false;
- + }
- + else if (errno == EXDEV)
- + /* We won't try cfr again for this run, but let's be
- + * conservative and not mark it as available/unavailable until
- + * we know for sure.
- + */
- + try_cfr = false;
- + else
- + return -1;
- + }
- + else
- + {
- + /* cfr worked, mark it as available */
- + if (have_cfr == -1)
- + have_cfr = 1;
- +
- + if (n == 0) /* EOF */
- + break;
- + else
- + /* Success! */
- + goto next;
- + }
- + }
- +
- + /* Next try sendfile(); this version is also changed from systemd upstream
- + * to match the same logic we have for copy_file_range().
- + */
- + if (try_sendfile)
- + {
- + n = sendfile (fdt, fdf, NULL, max_bytes);
- + if (n < 0)
- + {
- + if (G_IN_SET (errno, EINVAL, ENOSYS))
- + {
- + /* No sendfile(), or it doesn't work on regular files.
- + * Mark it as permanently unavailable, and fall through
- + * to plain read()/write().
- + */
- + have_sendfile = 0;
- + try_sendfile = false;
- + }
- + else
- + return -1;
- + }
- + else
- + {
- + /* sendfile() worked, mark it as available */
- + if (have_sendfile == -1)
- + have_sendfile = 1;
- +
- + if (n == 0) /* EOF */
- + break;
- + else if (n > 0)
- + /* Succcess! */
- + goto next;
- + }
- + }
- +
- + /* As a fallback just copy bits by hand */
- + { size_t m = COPY_BUFFER_SIZE;
- + if (max_bytes != (off_t) -1)
- + {
- + if ((off_t) m > max_bytes)
- + m = (size_t) max_bytes;
- + }
- + char buf[m];
- +
- + n = TEMP_FAILURE_RETRY (read (fdf, buf, m));
- + if (n < 0)
- + return -1;
- + if (n == 0) /* EOF */
- + break;
- +
- + if (glnx_loop_write (fdt, buf, (size_t) n) < 0)
- + return -1;
- + }
- +
- + next:
- + if (max_bytes != (off_t) -1)
- + {
- + g_assert_cmpint (max_bytes, >=, n);
- + max_bytes -= n;
- + if (max_bytes == 0)
- + break;
- + }
- + }
- +
- + return 0;
- +}
- +
- +/**
- + * glnx_file_copy_at:
- + * @src_dfd: Source directory fd
- + * @src_subpath: Subpath relative to @src_dfd
- + * @src_stbuf: (allow-none): Optional stat buffer for source; if a stat() has already been done
- + * @dest_dfd: Target directory fd
- + * @dest_subpath: Destination name
- + * @copyflags: Flags
- + * @cancellable: cancellable
- + * @error: Error
- + *
- + * Perform a full copy of the regular file or symbolic link from @src_subpath to
- + * @dest_subpath; if @src_subpath is anything other than a regular file or
- + * symbolic link, an error will be returned.
- + *
- + * If the source is a regular file and the destination exists as a symbolic
- + * link, the symbolic link will not be followed; rather the link itself will be
- + * replaced. Related to this: for regular files, when `GLNX_FILE_COPY_OVERWRITE`
- + * is specified, this function always uses `O_TMPFILE` (if available) and does a
- + * rename-into-place rather than `open(O_TRUNC)`.
- + */
- +gboolean
- +glnx_file_copy_at (int src_dfd,
- + const char *src_subpath,
- + struct stat *src_stbuf,
- + int dest_dfd,
- + const char *dest_subpath,
- + GLnxFileCopyFlags copyflags,
- + GCancellable *cancellable,
- + GError **error)
- +{
- + /* Canonicalize dfds */
- + src_dfd = glnx_dirfd_canonicalize (src_dfd);
- + dest_dfd = glnx_dirfd_canonicalize (dest_dfd);
- +
- + if (g_cancellable_set_error_if_cancelled (cancellable, error))
- + return FALSE;
- +
- + /* Automatically do stat() if no stat buffer was supplied */
- + struct stat local_stbuf;
- + if (!src_stbuf)
- + {
- + if (!glnx_fstatat (src_dfd, src_subpath, &local_stbuf, AT_SYMLINK_NOFOLLOW, error))
- + return FALSE;
- + src_stbuf = &local_stbuf;
- + }
- +
- + /* For symlinks, defer entirely to copy_symlink_at() */
- + if (S_ISLNK (src_stbuf->st_mode))
- + {
- + return copy_symlink_at (src_dfd, src_subpath, src_stbuf,
- + dest_dfd, dest_subpath,
- + copyflags,
- + cancellable, error);
- + }
- + else if (!S_ISREG (src_stbuf->st_mode))
- + {
- + g_set_error (error, G_IO_ERROR, G_IO_ERROR_NOT_SUPPORTED,
- + "Cannot copy non-regular/non-symlink file: %s", src_subpath);
- + return FALSE;
- + }
- +
- + /* Regular file path below here */
- +
- + glnx_autofd int src_fd = -1;
- + if (!glnx_openat_rdonly (src_dfd, src_subpath, FALSE, &src_fd, error))
- + return FALSE;
- +
- + /* Open a tmpfile for dest. Particularly for AT_FDCWD calls, we really want to
- + * open in the target directory, otherwise we may not be able to link.
- + */
- + g_auto(GLnxTmpfile) tmp_dest = { 0, };
- + { char *dnbuf = strdupa (dest_subpath);
- + const char *dn = dirname (dnbuf);
- + if (!glnx_open_tmpfile_linkable_at (dest_dfd, dn, O_WRONLY | O_CLOEXEC,
- + &tmp_dest, error))
- + return FALSE;
- + }
- +
- + if (glnx_regfile_copy_bytes (src_fd, tmp_dest.fd, (off_t) -1) < 0)
- + return glnx_throw_errno_prefix (error, "regfile copy");
- +
- + if (fchown (tmp_dest.fd, src_stbuf->st_uid, src_stbuf->st_gid) != 0)
- + return glnx_throw_errno_prefix (error, "fchown");
- +
- + if (!(copyflags & GLNX_FILE_COPY_NOXATTRS))
- + {
- + g_autoptr(GVariant) xattrs = NULL;
- +
- + if (!glnx_fd_get_all_xattrs (src_fd, &xattrs,
- + cancellable, error))
- + return FALSE;
- +
- + if (!glnx_fd_set_all_xattrs (tmp_dest.fd, xattrs,
- + cancellable, error))
- + return FALSE;
- + }
- +
- + /* Always chmod after setting xattrs, in case the file has mode 0400 or less,
- + * like /etc/shadow. Linux currently allows write() on non-writable open files
- + * but not fsetxattr().
- + */
- + if (fchmod (tmp_dest.fd, src_stbuf->st_mode & 07777) != 0)
- + return glnx_throw_errno_prefix (error, "fchmod");
- +
- + struct timespec ts[2];
- + ts[0] = src_stbuf->st_atim;
- + ts[1] = src_stbuf->st_mtim;
- + (void) futimens (tmp_dest.fd, ts);
- +
- + if (copyflags & GLNX_FILE_COPY_DATASYNC)
- + {
- + if (fdatasync (tmp_dest.fd) < 0)
- + return glnx_throw_errno_prefix (error, "fdatasync");
- + }
- +
- + const GLnxLinkTmpfileReplaceMode replacemode =
- + (copyflags & GLNX_FILE_COPY_OVERWRITE) ?
- + GLNX_LINK_TMPFILE_REPLACE :
- + GLNX_LINK_TMPFILE_NOREPLACE;
- +
- + if (!glnx_link_tmpfile_at (&tmp_dest, replacemode, dest_dfd, dest_subpath, error))
- + return FALSE;
- +
- + return TRUE;
- +}
- +
- +/**
- + * glnx_file_replace_contents_at:
- + * @dfd: Directory fd
- + * @subpath: Subpath
- + * @buf: (array len=len) (element-type guint8): File contents
- + * @len: Length (if `-1`, assume @buf is `NUL` terminated)
- + * @flags: Flags
- + * @cancellable: Cancellable
- + * @error: Error
- + *
- + * Create a new file, atomically replacing the contents of @subpath
- + * (relative to @dfd) with @buf. By default, if the file already
- + * existed, fdatasync() will be used before rename() to ensure stable
- + * contents. This and other behavior can be controlled via @flags.
- + *
- + * Note that no metadata from the existing file is preserved, such as
- + * uid/gid or extended attributes. The default mode will be `0666`,
- + * modified by umask.
- + */
- +gboolean
- +glnx_file_replace_contents_at (int dfd,
- + const char *subpath,
- + const guint8 *buf,
- + gsize len,
- + GLnxFileReplaceFlags flags,
- + GCancellable *cancellable,
- + GError **error)
- +{
- + return glnx_file_replace_contents_with_perms_at (dfd, subpath, buf, len,
- + (mode_t) -1, (uid_t) -1, (gid_t) -1,
- + flags, cancellable, error);
- +}
- +
- +/**
- + * glnx_file_replace_contents_with_perms_at:
- + * @dfd: Directory fd
- + * @subpath: Subpath
- + * @buf: (array len=len) (element-type guint8): File contents
- + * @len: Length (if `-1`, assume @buf is `NUL` terminated)
- + * @mode: File mode; if `-1`, use `0666 - umask`
- + * @flags: Flags
- + * @cancellable: Cancellable
- + * @error: Error
- + *
- + * Like glnx_file_replace_contents_at(), but also supports
- + * setting mode, and uid/gid.
- + */
- +gboolean
- +glnx_file_replace_contents_with_perms_at (int dfd,
- + const char *subpath,
- + const guint8 *buf,
- + gsize len,
- + mode_t mode,
- + uid_t uid,
- + gid_t gid,
- + GLnxFileReplaceFlags flags,
- + GCancellable *cancellable,
- + GError **error)
- +{
- + char *dnbuf = strdupa (subpath);
- + const char *dn = dirname (dnbuf);
- +
- + dfd = glnx_dirfd_canonicalize (dfd);
- +
- + /* With O_TMPFILE we can't use umask, and we can't sanely query the
- + * umask...let's assume something relatively standard.
- + */
- + if (mode == (mode_t) -1)
- + mode = 0644;
- +
- + g_auto(GLnxTmpfile) tmpf = { 0, };
- + if (!glnx_open_tmpfile_linkable_at (dfd, dn, O_WRONLY | O_CLOEXEC,
- + &tmpf, error))
- + return FALSE;
- +
- + if (len == -1)
- + len = strlen ((char*)buf);
- +
- + if (!glnx_try_fallocate (tmpf.fd, 0, len, error))
- + return FALSE;
- +
- + if (glnx_loop_write (tmpf.fd, buf, len) < 0)
- + return glnx_throw_errno_prefix (error, "write");
- +
- + if (!(flags & GLNX_FILE_REPLACE_NODATASYNC))
- + {
- + struct stat stbuf;
- + gboolean do_sync;
- +
- + if (!glnx_fstatat_allow_noent (dfd, subpath, &stbuf, AT_SYMLINK_NOFOLLOW, error))
- + return FALSE;
- + if (errno == ENOENT)
- + do_sync = (flags & GLNX_FILE_REPLACE_DATASYNC_NEW) > 0;
- + else
- + do_sync = TRUE;
- +
- + if (do_sync)
- + {
- + if (fdatasync (tmpf.fd) != 0)
- + return glnx_throw_errno_prefix (error, "fdatasync");
- + }
- + }
- +
- + if (uid != (uid_t) -1)
- + {
- + if (fchown (tmpf.fd, uid, gid) != 0)
- + return glnx_throw_errno_prefix (error, "fchown");
- + }
- +
- + if (fchmod (tmpf.fd, mode) != 0)
- + return glnx_throw_errno_prefix (error, "fchmod");
- +
- + if (!glnx_link_tmpfile_at (&tmpf, GLNX_LINK_TMPFILE_REPLACE,
- + dfd, subpath, error))
- + return FALSE;
- +
- + return TRUE;
- +}
- diff -Nuar ostree-2018.8.orig/libglnx/glnx-fdio.h ostree-2018.8/libglnx/glnx-fdio.h
- --- ostree-2018.8.orig/libglnx/glnx-fdio.h 1970-01-01 02:00:00.000000000 +0200
- +++ ostree-2018.8/libglnx/glnx-fdio.h 2018-05-26 00:37:32.007018846 +0300
- @@ -0,0 +1,369 @@
- +/* -*- mode: C; c-file-style: "gnu"; indent-tabs-mode: nil; -*-
- + *
- + * Copyright (C) 2014,2015 Colin Walters <walters@verbum.org>.
- + *
- + * This library is free software; you can redistribute it and/or
- + * modify it under the terms of the GNU Lesser General Public
- + * License as published by the Free Software Foundation; either
- + * version 2 of the License, or (at your option) any later version.
- + *
- + * This library is distributed in the hope that it will be useful,
- + * but WITHOUT ANY WARRANTY; without even the implied warranty of
- + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
- + * Lesser General Public License for more details.
- + *
- + * You should have received a copy of the GNU Lesser General Public
- + * License along with this library; if not, write to the
- + * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
- + * Boston, MA 02111-1307, USA.
- + */
- +
- +#pragma once
- +
- +#include <glnx-backport-autocleanups.h>
- +#include <gio/gfiledescriptorbased.h>
- +#include <limits.h>
- +#include <dirent.h>
- +#include <sys/stat.h>
- +#include <fcntl.h>
- +#include <string.h>
- +#include <stdio.h>
- +#include <sys/xattr.h>
- +// For dirname(), and previously basename()
- +#include <libgen.h>
- +
- +#include <glnx-macros.h>
- +#include <glnx-errors.h>
- +
- +G_BEGIN_DECLS
- +
- +/* Irritatingly, g_basename() which is what we want
- + * is deprecated.
- + */
- +static inline
- +const char *glnx_basename (const char *path)
- +{
- + gchar *base = strrchr (path, G_DIR_SEPARATOR);
- +
- + if (base)
- + return base + 1;
- +
- + return path;
- +}
- +
- +/* Utilities for standard FILE* */
- +static inline void
- +glnx_stdio_file_cleanup (void *filep)
- +{
- + FILE *f = filep;
- + if (f)
- + fclose (f);
- +}
- +G_DEFINE_AUTOPTR_CLEANUP_FUNC(FILE, glnx_stdio_file_cleanup)
- +
- +/**
- + * glnx_stdio_file_flush:
- + * Call fflush() and check ferror().
- + */
- +gboolean
- +glnx_stdio_file_flush (FILE *f, GError **error);
- +
- +typedef struct {
- + gboolean initialized;
- + gboolean anonymous;
- + int src_dfd;
- + int fd;
- + char *path;
- +} GLnxTmpfile;
- +void glnx_tmpfile_clear (GLnxTmpfile *tmpf);
- +G_DEFINE_AUTO_CLEANUP_CLEAR_FUNC(GLnxTmpfile, glnx_tmpfile_clear)
- +
- +gboolean
- +glnx_open_anonymous_tmpfile (int flags,
- + GLnxTmpfile *out_tmpf,
- + GError **error);
- +
- +gboolean
- +glnx_open_tmpfile_linkable_at (int dfd,
- + const char *subpath,
- + int flags,
- + GLnxTmpfile *out_tmpf,
- + GError **error);
- +
- +typedef enum {
- + GLNX_LINK_TMPFILE_REPLACE,
- + GLNX_LINK_TMPFILE_NOREPLACE,
- + GLNX_LINK_TMPFILE_NOREPLACE_IGNORE_EXIST
- +} GLnxLinkTmpfileReplaceMode;
- +
- +gboolean
- +glnx_link_tmpfile_at (GLnxTmpfile *tmpf,
- + GLnxLinkTmpfileReplaceMode flags,
- + int target_dfd,
- + const char *target,
- + GError **error);
- +
- +gboolean
- +glnx_openat_rdonly (int dfd,
- + const char *path,
- + gboolean follow,
- + int *out_fd,
- + GError **error);
- +
- +GBytes *
- +glnx_fd_readall_bytes (int fd,
- + GCancellable *cancellable,
- + GError **error);
- +
- +char *
- +glnx_fd_readall_utf8 (int fd,
- + gsize *out_len,
- + GCancellable *cancellable,
- + GError **error);
- +
- +char *
- +glnx_file_get_contents_utf8_at (int dfd,
- + const char *subpath,
- + gsize *out_len,
- + GCancellable *cancellable,
- + GError **error);
- +
- +/**
- + * GLnxFileReplaceFlags:
- + * @GLNX_FILE_REPLACE_DATASYNC_NEW: Call fdatasync() even if the file did not exist
- + * @GLNX_FILE_REPLACE_NODATASYNC: Never call fdatasync()
- + *
- + * Flags controlling file replacement.
- + */
- +typedef enum {
- + GLNX_FILE_REPLACE_DATASYNC_NEW = (1 << 0),
- + GLNX_FILE_REPLACE_NODATASYNC = (1 << 1),
- +} GLnxFileReplaceFlags;
- +
- +gboolean
- +glnx_file_replace_contents_at (int dfd,
- + const char *subpath,
- + const guint8 *buf,
- + gsize len,
- + GLnxFileReplaceFlags flags,
- + GCancellable *cancellable,
- + GError **error);
- +
- +gboolean
- +glnx_file_replace_contents_with_perms_at (int dfd,
- + const char *subpath,
- + const guint8 *buf,
- + gsize len,
- + mode_t mode,
- + uid_t uid,
- + gid_t gid,
- + GLnxFileReplaceFlags flags,
- + GCancellable *cancellable,
- + GError **error);
- +
- +char *
- +glnx_readlinkat_malloc (int dfd,
- + const char *subpath,
- + GCancellable *cancellable,
- + GError **error);
- +
- +int
- +glnx_loop_write (int fd, const void *buf, size_t nbytes);
- +
- +int
- +glnx_regfile_copy_bytes (int fdf, int fdt, off_t max_bytes);
- +
- +typedef enum {
- + GLNX_FILE_COPY_OVERWRITE = (1 << 0),
- + GLNX_FILE_COPY_NOXATTRS = (1 << 1),
- + GLNX_FILE_COPY_DATASYNC = (1 << 2)
- +} GLnxFileCopyFlags;
- +
- +gboolean
- +glnx_file_copy_at (int src_dfd,
- + const char *src_subpath,
- + struct stat *src_stbuf,
- + int dest_dfd,
- + const char *dest_subpath,
- + GLnxFileCopyFlags copyflags,
- + GCancellable *cancellable,
- + GError **error);
- +
- +int glnx_renameat2_noreplace (int olddirfd, const char *oldpath,
- + int newdirfd, const char *newpath);
- +int glnx_renameat2_exchange (int olddirfd, const char *oldpath,
- + int newdirfd, const char *newpath);
- +
- +/**
- + * glnx_try_fallocate:
- + * @fd: File descriptor
- + * @size: Size
- + * @error: Error
- + *
- + * Wrapper for Linux fallocate(). Explicitly ignores a @size of zero.
- + * Also, will silently do nothing if the underlying filesystem doesn't
- + * support it. Use this instead of posix_fallocate(), since the glibc fallback
- + * is bad: https://sourceware.org/bugzilla/show_bug.cgi?id=18515
- + */
- +static inline gboolean
- +glnx_try_fallocate (int fd,
- + off_t offset,
- + off_t size,
- + GError **error)
- +{
- + /* This is just nicer than throwing an error */
- + if (size == 0)
- + return TRUE;
- +
- + if (fallocate (fd, 0, offset, size) < 0)
- + {
- + if (G_IN_SET(errno, ENOSYS, EOPNOTSUPP))
- + ; /* Ignore */
- + else
- + return glnx_throw_errno_prefix (error, "fallocate");
- + }
- +
- + return TRUE;
- +}
- +
- +/**
- + * glnx_fstat:
- + * @fd: FD to stat
- + * @buf: (out caller-allocates): Return location for stat details
- + * @error: Return location for a #GError, or %NULL
- + *
- + * Wrapper around fstat() which adds #GError support and ensures that it retries
- + * on %EINTR.
- + *
- + * Returns: %TRUE on success, %FALSE otherwise
- + * Since: UNRELEASED
- + */
- +static inline gboolean
- +glnx_fstat (int fd,
- + struct stat *buf,
- + GError **error)
- +{
- + if (TEMP_FAILURE_RETRY (fstat (fd, buf)) != 0)
- + return glnx_throw_errno_prefix (error, "fstat");
- + return TRUE;
- +}
- +
- +/**
- + * glnx_fchmod:
- + * @fd: FD
- + * @mode: Mode
- + * @error: Return location for a #GError, or %NULL
- + *
- + * Wrapper around fchmod() which adds #GError support and ensures that it
- + * retries on %EINTR.
- + *
- + * Returns: %TRUE on success, %FALSE otherwise
- + * Since: UNRELEASED
- + */
- +static inline gboolean
- +glnx_fchmod (int fd,
- + mode_t mode,
- + GError **error)
- +{
- + if (TEMP_FAILURE_RETRY (fchmod (fd, mode)) != 0)
- + return glnx_throw_errno_prefix (error, "fchmod");
- + return TRUE;
- +}
- +
- +/**
- + * glnx_fstatat:
- + * @dfd: Directory FD to stat beneath
- + * @path: Path to stat beneath @dfd
- + * @buf: (out caller-allocates): Return location for stat details
- + * @flags: Flags to pass to fstatat()
- + * @error: Return location for a #GError, or %NULL
- + *
- + * Wrapper around fstatat() which adds #GError support and ensures that it
- + * retries on %EINTR.
- + *
- + * Returns: %TRUE on success, %FALSE otherwise
- + * Since: UNRELEASED
- + */
- +static inline gboolean
- +glnx_fstatat (int dfd,
- + const gchar *path,
- + struct stat *buf,
- + int flags,
- + GError **error)
- +{
- + if (TEMP_FAILURE_RETRY (fstatat (dfd, path, buf, flags)) != 0)
- + return glnx_throw_errno_prefix (error, "fstatat(%s)", path);
- + return TRUE;
- +}
- +
- +/**
- + * glnx_fstatat_allow_noent:
- + * @dfd: Directory FD to stat beneath
- + * @path: Path to stat beneath @dfd
- + * @buf: (out caller-allocates) (allow-none): Return location for stat details
- + * @flags: Flags to pass to fstatat()
- + * @error: Return location for a #GError, or %NULL
- + *
- + * Like glnx_fstatat(), but handles `ENOENT` in a non-error way. Instead,
- + * on success `errno` will be zero, otherwise it will be preserved. Hence
- + * you can test `if (errno == 0)` to conditionalize on the file existing,
- + * or `if (errno == ENOENT)` for non-existence.
- + *
- + * Returns: %TRUE on success, %FALSE otherwise (errno is preserved)
- + * Since: UNRELEASED
- + */
- +static inline gboolean
- +glnx_fstatat_allow_noent (int dfd,
- + const char *path,
- + struct stat *out_buf,
- + int flags,
- + GError **error)
- +{
- + G_GNUC_UNUSED struct stat unused_stbuf;
- + if (TEMP_FAILURE_RETRY (fstatat (dfd, path, out_buf ? out_buf : &unused_stbuf, flags)) != 0)
- + {
- + if (errno != ENOENT)
- + return glnx_throw_errno_prefix (error, "fstatat(%s)", path);
- + /* Note we preserve errno as ENOENT */
- + }
- + else
- + errno = 0;
- + return TRUE;
- +}
- +
- +/**
- + * glnx_renameat:
- + *
- + * Wrapper around renameat() which adds #GError support and ensures that it
- + * retries on %EINTR.
- + */
- +static inline gboolean
- +glnx_renameat (int src_dfd,
- + const gchar *src_path,
- + int dest_dfd,
- + const gchar *dest_path,
- + GError **error)
- +{
- + if (TEMP_FAILURE_RETRY (renameat (src_dfd, src_path, dest_dfd, dest_path)) != 0)
- + return glnx_throw_errno_prefix (error, "renameat(%s, %s)", src_path, dest_path);
- + return TRUE;
- +}
- +
- +/**
- + * glnx_unlinkat:
- + *
- + * Wrapper around unlinkat() which adds #GError support and ensures that it
- + * retries on %EINTR.
- + */
- +static inline gboolean
- +glnx_unlinkat (int dfd,
- + const gchar *path,
- + int flags,
- + GError **error)
- +{
- + if (TEMP_FAILURE_RETRY (unlinkat (dfd, path, flags)) != 0)
- + return glnx_throw_errno_prefix (error, "unlinkat(%s)", path);
- + return TRUE;
- +}
- +
- +G_END_DECLS
- diff -Nuar ostree-2018.8.orig/libglnx/glnx-local-alloc.c ostree-2018.8/libglnx/glnx-local-alloc.c
- --- ostree-2018.8.orig/libglnx/glnx-local-alloc.c 1970-01-01 02:00:00.000000000 +0200
- +++ ostree-2018.8/libglnx/glnx-local-alloc.c 2018-05-26 00:37:32.007018846 +0300
- @@ -0,0 +1,72 @@
- +/* -*- mode: C; c-file-style: "gnu"; indent-tabs-mode: nil; -*-
- + *
- + * Copyright (C) 2012,2015 Colin Walters <walters@verbum.org>
- + *
- + * This library is free software; you can redistribute it and/or
- + * modify it under the terms of the GNU Lesser General Public
- + * License as published by the Free Software Foundation; either
- + * version 2 of the License, or (at your option) any later version.
- + *
- + * This library is distributed in the hope that it will be useful,
- + * but WITHOUT ANY WARRANTY; without even the implied warranty of
- + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
- + * Lesser General Public License for more details.
- + *
- + * You should have received a copy of the GNU Lesser General Public
- + * License along with this library; if not, write to the
- + * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
- + * Boston, MA 02111-1307, USA.
- + */
- +
- +#include "config.h"
- +
- +#include "glnx-local-alloc.h"
- +
- +/**
- + * SECTION:glnxlocalalloc
- + * @title: GLnx local allocation
- + * @short_description: Release local variables automatically when they go out of scope
- + *
- + * These macros leverage the GCC extension __attribute__ ((cleanup))
- + * to allow calling a cleanup function such as g_free() when a
- + * variable goes out of scope. See <ulink
- + * url="http://gcc.gnu.org/onlinedocs/gcc/Variable-Attributes.html">
- + * for more information on the attribute.
- + *
- + * The provided macros make it easy to use the cleanup attribute for
- + * types that come with GLib. The primary two are #glnx_free and
- + * #glnx_unref_object, which correspond to g_free() and
- + * g_object_unref(), respectively.
- + *
- + * The rationale behind this is that particularly when handling error
- + * paths, it can be very tricky to ensure the right variables are
- + * freed. With this, one simply applies glnx_unref_object to a
- + * locally-allocated #GFile for example, and it will be automatically
- + * unreferenced when it goes out of scope.
- + *
- + * Note - you should only use these macros for <emphasis>stack
- + * allocated</emphasis> variables. They don't provide garbage
- + * collection or let you avoid freeing things. They're simply a
- + * compiler assisted deterministic mechanism for calling a cleanup
- + * function when a stack frame ends.
- + *
- + * <example id="gs-lfree"><title>Calling g_free automatically</title>
- + * <programlisting>
- + *
- + * GFile *
- + * create_file (GError **error)
- + * {
- + * glnx_free char *random_id = NULL;
- + *
- + * if (!prepare_file (error))
- + * return NULL;
- + *
- + * random_id = alloc_random_id ();
- + *
- + * return create_file_real (error);
- + * // Note that random_id is freed here automatically
- + * }
- + * </programlisting>
- + * </example>
- + *
- + */
- diff -Nuar ostree-2018.8.orig/libglnx/glnx-local-alloc.h ostree-2018.8/libglnx/glnx-local-alloc.h
- --- ostree-2018.8.orig/libglnx/glnx-local-alloc.h 1970-01-01 02:00:00.000000000 +0200
- +++ ostree-2018.8/libglnx/glnx-local-alloc.h 2018-05-26 00:37:32.007018846 +0300
- @@ -0,0 +1,91 @@
- +/* -*- mode: C; c-file-style: "gnu"; indent-tabs-mode: nil; -*-
- + *
- + * Copyright (C) 2012,2015 Colin Walters <walters@verbum.org>.
- + *
- + * This library is free software; you can redistribute it and/or
- + * modify it under the terms of the GNU Lesser General Public
- + * License as published by the Free Software Foundation; either
- + * version 2 of the License, or (at your option) any later version.
- + *
- + * This library is distributed in the hope that it will be useful,
- + * but WITHOUT ANY WARRANTY; without even the implied warranty of
- + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
- + * Lesser General Public License for more details.
- + *
- + * You should have received a copy of the GNU Lesser General Public
- + * License along with this library; if not, write to the
- + * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
- + * Boston, MA 02111-1307, USA.
- + */
- +
- +#pragma once
- +
- +#include <gio/gio.h>
- +#include <errno.h>
- +
- +G_BEGIN_DECLS
- +
- +/**
- + * glnx_unref_object:
- + *
- + * Call g_object_unref() on a variable location when it goes out of
- + * scope. Note that unlike g_object_unref(), the variable may be
- + * %NULL.
- + */
- +#define glnx_unref_object __attribute__ ((cleanup(glnx_local_obj_unref)))
- +static inline void
- +glnx_local_obj_unref (void *v)
- +{
- + GObject *o = *(GObject **)v;
- + if (o)
- + g_object_unref (o);
- +}
- +#define glnx_unref_object __attribute__ ((cleanup(glnx_local_obj_unref)))
- +
- +static inline int
- +glnx_steal_fd (int *fdp)
- +{
- + int fd = *fdp;
- + *fdp = -1;
- + return fd;
- +}
- +
- +/**
- + * glnx_close_fd:
- + * @fdp: Pointer to fd
- + *
- + * Effectively `close (glnx_steal_fd (&fd))`. Also
- + * asserts that `close()` did not raise `EBADF` - encountering
- + * that error is usually a critical bug in the program.
- + */
- +static inline void
- +glnx_close_fd (int *fdp)
- +{
- + int errsv;
- +
- + g_assert (fdp);
- +
- + int fd = glnx_steal_fd (fdp);
- + if (fd >= 0)
- + {
- + errsv = errno;
- + if (close (fd) < 0)
- + g_assert (errno != EBADF);
- + errno = errsv;
- + }
- +}
- +
- +/**
- + * glnx_fd_close:
- + *
- + * Deprecated in favor of `glnx_autofd`.
- + */
- +#define glnx_fd_close __attribute__((cleanup(glnx_close_fd)))
- +/**
- + * glnx_autofd:
- + *
- + * Call close() on a variable location when it goes out of scope.
- + */
- +#define glnx_autofd __attribute__((cleanup(glnx_close_fd)))
- +
- +G_END_DECLS
- diff -Nuar ostree-2018.8.orig/libglnx/glnx-lockfile.c ostree-2018.8/libglnx/glnx-lockfile.c
- --- ostree-2018.8.orig/libglnx/glnx-lockfile.c 1970-01-01 02:00:00.000000000 +0200
- +++ ostree-2018.8/libglnx/glnx-lockfile.c 2018-05-26 00:37:32.007018846 +0300
- @@ -0,0 +1,179 @@
- +/*-*- Mode: C; c-basic-offset: 8; indent-tabs-mode: nil -*-*/
- +
- +/***
- + This file is part of systemd.
- + Now copied into libglnx:
- + - Use GError
- +
- + Copyright 2010 Lennart Poettering
- + Copyright 2015 Colin Walters <walters@verbum.org>
- +
- + systemd is free software; you can redistribute it and/or modify it
- + under the terms of the GNU Lesser General Public License as published by
- + the Free Software Foundation; either version 2.1 of the License, or
- + (at your option) any later version.
- +
- + systemd is distributed in the hope that it will be useful, but
- + WITHOUT ANY WARRANTY; without even the implied warranty of
- + MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
- + Lesser General Public License for more details.
- +
- + You should have received a copy of the GNU Lesser General Public License
- + along with systemd; If not, see <http://www.gnu.org/licenses/>.
- +***/
- +
- +#include "config.h"
- +
- +#include <stdlib.h>
- +#include <stdbool.h>
- +#include <errno.h>
- +#include <string.h>
- +#include <stdio.h>
- +#include <limits.h>
- +#include <unistd.h>
- +#include <sys/types.h>
- +#include <sys/file.h>
- +#include <sys/stat.h>
- +#include <fcntl.h>
- +
- +#include "glnx-lockfile.h"
- +#include "glnx-errors.h"
- +#include "glnx-fdio.h"
- +#include "glnx-backport-autocleanups.h"
- +#include "glnx-local-alloc.h"
- +
- +#define newa(t, n) ((t*) alloca(sizeof(t)*(n)))
- +
- +/**
- + * glnx_make_lock_file:
- + * @dfd: Directory file descriptor (if not `AT_FDCWD`, must have lifetime `>=` @out_lock)
- + * @p: Path
- + * @operation: one of `LOCK_SH`, `LOCK_EX`, `LOCK_UN`, as passed to flock()
- + * @out_lock: (out) (caller allocates): Return location for lock
- + * @error: Error
- + *
- + * Block until a lock file named @p (relative to @dfd) can be created,
- + * using the flags in @operation, returning the lock data in the
- + * caller-allocated location @out_lock.
- + *
- + * This API wraps new-style process locking if available, otherwise
- + * falls back to BSD locks.
- + */
- +gboolean
- +glnx_make_lock_file(int dfd, const char *p, int operation, GLnxLockFile *out_lock, GError **error) {
- + glnx_autofd int fd = -1;
- + g_autofree char *t = NULL;
- + int r;
- +
- + /*
- + * We use UNPOSIX locks if they are available. They have nice
- + * semantics, and are mostly compatible with NFS. However,
- + * they are only available on new kernels. When we detect we
- + * are running on an older kernel, then we fall back to good
- + * old BSD locks. They also have nice semantics, but are
- + * slightly problematic on NFS, where they are upgraded to
- + * POSIX locks, even though locally they are orthogonal to
- + * POSIX locks.
- + */
- +
- + t = g_strdup(p);
- +
- + for (;;) {
- +#ifdef F_OFD_SETLK
- + struct flock fl = {
- + .l_type = (operation & ~LOCK_NB) == LOCK_EX ? F_WRLCK : F_RDLCK,
- + .l_whence = SEEK_SET,
- + };
- +#endif
- + struct stat st;
- +
- + fd = openat(dfd, p, O_CREAT|O_RDWR|O_NOFOLLOW|O_CLOEXEC|O_NOCTTY, 0600);
- + if (fd < 0)
- + return glnx_throw_errno(error);
- +
- + /* Unfortunately, new locks are not in RHEL 7.1 glibc */
- +#ifdef F_OFD_SETLK
- + r = fcntl(fd, (operation & LOCK_NB) ? F_OFD_SETLK : F_OFD_SETLKW, &fl);
- +#else
- + r = -1;
- + errno = EINVAL;
- +#endif
- + if (r < 0) {
- +
- + /* If the kernel is too old, use good old BSD locks */
- + if (errno == EINVAL)
- + r = flock(fd, operation);
- +
- + if (r < 0)
- + return glnx_throw_errno_prefix (error, "flock");
- + }
- +
- + /* If we acquired the lock, let's check if the file
- + * still exists in the file system. If not, then the
- + * previous exclusive owner removed it and then closed
- + * it. In such a case our acquired lock is worthless,
- + * hence try again. */
- +
- + if (!glnx_fstat (fd, &st, error))
- + return FALSE;
- + if (st.st_nlink > 0)
- + break;
- +
- + glnx_close_fd (&fd);
- + }
- +
- + /* Note that if this is not AT_FDCWD, the caller takes responsibility
- + * for the fd's lifetime being >= that of the lock.
- + */
- + out_lock->initialized = TRUE;
- + out_lock->dfd = dfd;
- + out_lock->path = g_steal_pointer (&t);
- + out_lock->fd = glnx_steal_fd (&fd);
- + out_lock->operation = operation;
- + return TRUE;
- +}
- +
- +void glnx_release_lock_file(GLnxLockFile *f) {
- + int r;
- +
- + if (!(f && f->initialized))
- + return;
- +
- + if (f->path) {
- +
- + /* If we are the exclusive owner we can safely delete
- + * the lock file itself. If we are not the exclusive
- + * owner, we can try becoming it. */
- +
- + if (f->fd >= 0 &&
- + (f->operation & ~LOCK_NB) == LOCK_SH) {
- +#ifdef F_OFD_SETLK
- + static const struct flock fl = {
- + .l_type = F_WRLCK,
- + .l_whence = SEEK_SET,
- + };
- +
- + r = fcntl(f->fd, F_OFD_SETLK, &fl);
- +#else
- + r = -1;
- + errno = EINVAL;
- +#endif
- + if (r < 0 && errno == EINVAL)
- + r = flock(f->fd, LOCK_EX|LOCK_NB);
- +
- + if (r >= 0)
- + f->operation = LOCK_EX|LOCK_NB;
- + }
- +
- + if ((f->operation & ~LOCK_NB) == LOCK_EX) {
- + (void) unlinkat(f->dfd, f->path, 0);
- + }
- +
- + g_free(f->path);
- + f->path = NULL;
- + }
- +
- + glnx_close_fd (&f->fd);
- + f->operation = 0;
- + f->initialized = FALSE;
- +}
- diff -Nuar ostree-2018.8.orig/libglnx/glnx-lockfile.h ostree-2018.8/libglnx/glnx-lockfile.h
- --- ostree-2018.8.orig/libglnx/glnx-lockfile.h 1970-01-01 02:00:00.000000000 +0200
- +++ ostree-2018.8/libglnx/glnx-lockfile.h 2018-05-26 00:37:32.007018846 +0300
- @@ -0,0 +1,40 @@
- +/*-*- Mode: C; c-basic-offset: 8; indent-tabs-mode: nil -*-*/
- +
- +#pragma once
- +
- +/***
- + This file is part of systemd.
- +
- + Copyright 2011 Lennart Poettering
- + Copyright 2015 Colin Walters <walters@verbum.org>
- +
- + systemd is free software; you can redistribute it and/or modify it
- + under the terms of the GNU Lesser General Public License as published by
- + the Free Software Foundation; either version 2.1 of the License, or
- + (at your option) any later version.
- +
- + systemd is distributed in the hope that it will be useful, but
- + WITHOUT ANY WARRANTY; without even the implied warranty of
- + MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
- + Lesser General Public License for more details.
- +
- + You should have received a copy of the GNU Lesser General Public License
- + along with systemd; If not, see <http://www.gnu.org/licenses/>.
- +***/
- +
- +#include "config.h"
- +
- +#include "glnx-backport-autoptr.h"
- +
- +typedef struct GLnxLockFile {
- + gboolean initialized;
- + int dfd;
- + char *path;
- + int fd;
- + int operation;
- +} GLnxLockFile;
- +
- +gboolean glnx_make_lock_file(int dfd, const char *p, int operation, GLnxLockFile *ret, GError **error);
- +void glnx_release_lock_file(GLnxLockFile *f);
- +
- +G_DEFINE_AUTO_CLEANUP_CLEAR_FUNC(GLnxLockFile, glnx_release_lock_file)
- diff -Nuar ostree-2018.8.orig/libglnx/glnx-macros.h ostree-2018.8/libglnx/glnx-macros.h
- --- ostree-2018.8.orig/libglnx/glnx-macros.h 1970-01-01 02:00:00.000000000 +0200
- +++ ostree-2018.8/libglnx/glnx-macros.h 2018-05-26 00:37:32.007018846 +0300
- @@ -0,0 +1,189 @@
- +/* -*- mode: C; c-file-style: "gnu"; indent-tabs-mode: nil; -*-
- + *
- + * Copyright (C) 2017 Colin Walters <walters@verbum.org>
- + * With original source from systemd:
- + * Copyright 2010 Lennart Poettering
- + *
- + * This library is free software; you can redistribute it and/or
- + * modify it under the terms of the GNU Lesser General Public
- + * License as published by the Free Software Foundation; either
- + * version 2 of the License, or (at your option) any later version.
- + *
- + * This library is distributed in the hope that it will be useful,
- + * but WITHOUT ANY WARRANTY; without even the implied warranty of
- + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
- + * Lesser General Public License for more details.
- + *
- + * You should have received a copy of the GNU Lesser General Public
- + * License along with this library; if not, write to the
- + * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
- + * Boston, MA 02111-1307, USA.
- + */
- +
- +#pragma once
- +
- +#include <stdlib.h>
- +#include <string.h>
- +#include <gio/gio.h>
- +
- +G_BEGIN_DECLS
- +
- +/* All of these are for C only. */
- +#ifndef __GI_SCANNER__
- +
- +/* Taken from https://github.com/systemd/systemd/src/basic/string-util.h
- + * at revision v228-666-gcf6c8c4
- + */
- +#define glnx_strjoina(a, ...) \
- + ({ \
- + const char *_appendees_[] = { a, __VA_ARGS__ }; \
- + char *_d_, *_p_; \
- + size_t _len_ = 0; \
- + unsigned _i_; \
- + for (_i_ = 0; _i_ < G_N_ELEMENTS(_appendees_) && _appendees_[_i_]; _i_++) \
- + _len_ += strlen(_appendees_[_i_]); \
- + _p_ = _d_ = alloca(_len_ + 1); \
- + for (_i_ = 0; _i_ < G_N_ELEMENTS(_appendees_) && _appendees_[_i_]; _i_++) \
- + _p_ = stpcpy(_p_, _appendees_[_i_]); \
- + *_p_ = 0; \
- + _d_; \
- + })
- +
- +#ifndef G_IN_SET
- +
- +/* Infrastructure for `G_IN_SET`; this code is copied from
- + * systemd's macro.h - please treat that version as canonical
- + * and submit patches first to systemd.
- + */
- +#define _G_INSET_CASE_F(X) case X:
- +#define _G_INSET_CASE_F_1(CASE, X) _G_INSET_CASE_F(X)
- +#define _G_INSET_CASE_F_2(CASE, X, ...) CASE(X) _G_INSET_CASE_F_1(CASE, __VA_ARGS__)
- +#define _G_INSET_CASE_F_3(CASE, X, ...) CASE(X) _G_INSET_CASE_F_2(CASE, __VA_ARGS__)
- +#define _G_INSET_CASE_F_4(CASE, X, ...) CASE(X) _G_INSET_CASE_F_3(CASE, __VA_ARGS__)
- +#define _G_INSET_CASE_F_5(CASE, X, ...) CASE(X) _G_INSET_CASE_F_4(CASE, __VA_ARGS__)
- +#define _G_INSET_CASE_F_6(CASE, X, ...) CASE(X) _G_INSET_CASE_F_5(CASE, __VA_ARGS__)
- +#define _G_INSET_CASE_F_7(CASE, X, ...) CASE(X) _G_INSET_CASE_F_6(CASE, __VA_ARGS__)
- +#define _G_INSET_CASE_F_8(CASE, X, ...) CASE(X) _G_INSET_CASE_F_7(CASE, __VA_ARGS__)
- +#define _G_INSET_CASE_F_9(CASE, X, ...) CASE(X) _G_INSET_CASE_F_8(CASE, __VA_ARGS__)
- +#define _G_INSET_CASE_F_10(CASE, X, ...) CASE(X) _G_INSET_CASE_F_9(CASE, __VA_ARGS__)
- +#define _G_INSET_CASE_F_11(CASE, X, ...) CASE(X) _G_INSET_CASE_F_10(CASE, __VA_ARGS__)
- +#define _G_INSET_CASE_F_12(CASE, X, ...) CASE(X) _G_INSET_CASE_F_11(CASE, __VA_ARGS__)
- +#define _G_INSET_CASE_F_13(CASE, X, ...) CASE(X) _G_INSET_CASE_F_12(CASE, __VA_ARGS__)
- +#define _G_INSET_CASE_F_14(CASE, X, ...) CASE(X) _G_INSET_CASE_F_13(CASE, __VA_ARGS__)
- +#define _G_INSET_CASE_F_15(CASE, X, ...) CASE(X) _G_INSET_CASE_F_14(CASE, __VA_ARGS__)
- +#define _G_INSET_CASE_F_16(CASE, X, ...) CASE(X) _G_INSET_CASE_F_15(CASE, __VA_ARGS__)
- +#define _G_INSET_CASE_F_17(CASE, X, ...) CASE(X) _G_INSET_CASE_F_16(CASE, __VA_ARGS__)
- +#define _G_INSET_CASE_F_18(CASE, X, ...) CASE(X) _G_INSET_CASE_F_17(CASE, __VA_ARGS__)
- +#define _G_INSET_CASE_F_19(CASE, X, ...) CASE(X) _G_INSET_CASE_F_18(CASE, __VA_ARGS__)
- +#define _G_INSET_CASE_F_20(CASE, X, ...) CASE(X) _G_INSET_CASE_F_19(CASE, __VA_ARGS__)
- +
- +#define _G_INSET_GET_CASE_F(_1,_2,_3,_4,_5,_6,_7,_8,_9,_10,_11,_12,_13,_14,_15,_16,_17,_18,_19,_20,NAME,...) NAME
- +#define _G_INSET_FOR_EACH_MAKE_CASE(...) \
- + _G_INSET_GET_CASE_F(__VA_ARGS__,_G_INSET_CASE_F_20,_G_INSET_CASE_F_19,_G_INSET_CASE_F_18,_G_INSET_CASE_F_17,_G_INSET_CASE_F_16,_G_INSET_CASE_F_15,_G_INSET_CASE_F_14,_G_INSET_CASE_F_13,_G_INSET_CASE_F_12,_G_INSET_CASE_F_11, \
- + _G_INSET_CASE_F_10,_G_INSET_CASE_F_9,_G_INSET_CASE_F_8,_G_INSET_CASE_F_7,_G_INSET_CASE_F_6,_G_INSET_CASE_F_5,_G_INSET_CASE_F_4,_G_INSET_CASE_F_3,_G_INSET_CASE_F_2,_G_INSET_CASE_F_1) \
- + (_G_INSET_CASE_F,__VA_ARGS__)
- +
- +/* Note: claiming the name here even though it isn't upstream yet
- + * https://bugzilla.gnome.org/show_bug.cgi?id=783751
- + */
- +/**
- + * G_IN_SET:
- + * @x: Integer (or smaller) sized value
- + * @...: Elements to compare
- + *
- + * It's quite common to test whether or not `char` values or Unix @errno (among) others
- + * are members of a small set. Normally one has to choose to either use `if (x == val || x == otherval ...)`
- + * or a `switch` statement. This macro is useful to reduce duplication in the first case,
- + * where one can write simply `if (G_IN_SET (x, val, otherval))`, and avoid the verbosity
- + * that the `switch` statement requires.
- + */
- +#define G_IN_SET(x, ...) \
- + ({ \
- + gboolean _g_inset_found = FALSE; \
- + /* If the build breaks in the line below, you need to extend the case macros */ \
- + static G_GNUC_UNUSED char _static_assert__macros_need_to_be_extended[20 - sizeof((int[]){__VA_ARGS__})/sizeof(int)]; \
- + switch(x) { \
- + _G_INSET_FOR_EACH_MAKE_CASE(__VA_ARGS__) \
- + _g_inset_found = TRUE; \
- + break; \
- + default: \
- + break; \
- + } \
- + _g_inset_found; \
- + })
- +
- +#endif /* ifndef G_IN_SET */
- +
- +#define _GLNX_CONCAT(a, b) a##b
- +#define _GLNX_CONCAT_INDIRECT(a, b) _GLNX_CONCAT(a, b)
- +#define _GLNX_MAKE_ANONYMOUS(a) _GLNX_CONCAT_INDIRECT(a, __COUNTER__)
- +
- +#define _GLNX_HASH_TABLE_FOREACH_IMPL_KV(guard, ht, it, kt, k, vt, v) \
- + gboolean guard = TRUE; \
- + G_STATIC_ASSERT (sizeof (kt) == sizeof (void*)); \
- + G_STATIC_ASSERT (sizeof (vt) == sizeof (void*)); \
- + for (GHashTableIter it; \
- + guard && ({ g_hash_table_iter_init (&it, ht), TRUE; }); \
- + guard = FALSE) \
- + for (kt k; guard; guard = FALSE) \
- + for (vt v; g_hash_table_iter_next (&it, (gpointer)&k, (gpointer)&v);)
- +
- +
- +/* Cleaner method to iterate over a GHashTable. I.e. rather than
- + *
- + * gpointer k, v;
- + * GHashTableIter it;
- + * g_hash_table_iter_init (&it, table);
- + * while (g_hash_table_iter_next (&it, &k, &v))
- + * {
- + * const char *str = k;
- + * GPtrArray *arr = v;
- + * ...
- + * }
- + *
- + * you can simply do
- + *
- + * GLNX_HASH_TABLE_FOREACH_IT (table, it, const char*, str, GPtrArray*, arr)
- + * {
- + * ...
- + * }
- + *
- + * All variables are scoped within the loop. You may use the `it` variable as
- + * usual, e.g. to remove an element using g_hash_table_iter_remove(&it). There
- + * are shorter variants for the more common cases where you do not need access
- + * to the iterator or to keys/values:
- + *
- + * GLNX_HASH_TABLE_FOREACH (table, const char*, str) { ... }
- + * GLNX_HASH_TABLE_FOREACH_V (table, MyData*, data) { ... }
- + * GLNX_HASH_TABLE_FOREACH_KV (table, const char*, str, MyData*, data) { ... }
- + *
- + */
- +#define GLNX_HASH_TABLE_FOREACH_IT(ht, it, kt, k, vt, v) \
- + _GLNX_HASH_TABLE_FOREACH_IMPL_KV( \
- + _GLNX_MAKE_ANONYMOUS(_glnx_ht_iter_guard_), ht, it, kt, k, vt, v)
- +
- +/* Variant of GLNX_HASH_TABLE_FOREACH without having to specify an iterator. An
- + * anonymous iterator will be created. */
- +#define GLNX_HASH_TABLE_FOREACH_KV(ht, kt, k, vt, v) \
- + _GLNX_HASH_TABLE_FOREACH_IMPL_KV( \
- + _GLNX_MAKE_ANONYMOUS(_glnx_ht_iter_guard_), ht, \
- + _GLNX_MAKE_ANONYMOUS(_glnx_ht_iter_it_), kt, k, vt, v)
- +
- +/* Variant of GLNX_HASH_TABLE_FOREACH_KV which omits unpacking keys. */
- +#define GLNX_HASH_TABLE_FOREACH_V(ht, vt, v) \
- + _GLNX_HASH_TABLE_FOREACH_IMPL_KV( \
- + _GLNX_MAKE_ANONYMOUS(_glnx_ht_iter_guard_), ht, \
- + _GLNX_MAKE_ANONYMOUS(_glnx_ht_iter_it_), \
- + gpointer, _GLNX_MAKE_ANONYMOUS(_glnx_ht_iter_v_), \
- + vt, v)
- +
- +/* Variant of GLNX_HASH_TABLE_FOREACH_KV which omits unpacking vals. */
- +#define GLNX_HASH_TABLE_FOREACH(ht, kt, k) \
- + _GLNX_HASH_TABLE_FOREACH_IMPL_KV( \
- + _GLNX_MAKE_ANONYMOUS(_glnx_ht_iter_guard_), ht, \
- + _GLNX_MAKE_ANONYMOUS(_glnx_ht_iter_it_), kt, k, \
- + gpointer, _GLNX_MAKE_ANONYMOUS(_glnx_ht_iter_v_))
- +
- +#endif /* GI_SCANNER */
- +
- +G_END_DECLS
- diff -Nuar ostree-2018.8.orig/libglnx/glnx-missing.h ostree-2018.8/libglnx/glnx-missing.h
- --- ostree-2018.8.orig/libglnx/glnx-missing.h 1970-01-01 02:00:00.000000000 +0200
- +++ ostree-2018.8/libglnx/glnx-missing.h 2018-05-26 00:37:32.007018846 +0300
- @@ -0,0 +1,95 @@
- +#pragma once
- +
- +/***
- + This file was originally part of systemd.
- +
- + Copyright 2010 Lennart Poettering
- +
- + systemd is free software; you can redistribute it and/or modify it
- + under the terms of the GNU Lesser General Public License as published by
- + the Free Software Foundation; either version 2.1 of the License, or
- + (at your option) any later version.
- +
- + systemd is distributed in the hope that it will be useful, but
- + WITHOUT ANY WARRANTY; without even the implied warranty of
- + MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
- + Lesser General Public License for more details.
- +
- + You should have received a copy of the GNU Lesser General Public License
- + along with systemd; If not, see <http://www.gnu.org/licenses/>.
- +***/
- +
- +/* Missing glibc definitions to access certain kernel APIs.
- + This file is last updated from systemd git:
- +
- + commit 71e5200f94b22589922704aa4abdf95d4fe2e528
- + Author: Daniel Mack <daniel@zonque.org>
- + AuthorDate: Tue Oct 18 17:57:10 2016 +0200
- + Commit: Lennart Poettering <lennart@poettering.net>
- + CommitDate: Fri Sep 22 15:24:54 2017 +0200
- +
- + Add abstraction model for BPF programs
- +*/
- +
- +#include <errno.h>
- +#include <fcntl.h>
- +#include <stdlib.h>
- +#include <sys/resource.h>
- +#include <sys/syscall.h>
- +#include <uchar.h>
- +#include <unistd.h>
- +
- +/* The precise definition of __O_TMPFILE is arch specific; use the
- + * values defined by the kernel (note: some are hexa, some are octal,
- + * duplicated as-is from the kernel definitions):
- + * - alpha, parisc, sparc: each has a specific value;
- + * - others: they use the "generic" value.
- + */
- +
- +#ifndef __O_TMPFILE
- +#if defined(__alpha__)
- +#define __O_TMPFILE 0100000000
- +#elif defined(__parisc__) || defined(__hppa__)
- +#define __O_TMPFILE 0400000000
- +#elif defined(__sparc__) || defined(__sparc64__)
- +#define __O_TMPFILE 0x2000000
- +#else
- +#define __O_TMPFILE 020000000
- +#endif
- +#endif
- +
- +/* a horrid kludge trying to make sure that this will fail on old kernels */
- +#ifndef O_TMPFILE
- +#define O_TMPFILE (__O_TMPFILE | O_DIRECTORY)
- +#endif
- +
- +#ifndef RENAME_NOREPLACE
- +#define RENAME_NOREPLACE (1 << 0)
- +#endif
- +#ifndef RENAME_EXCHANGE
- +#define RENAME_EXCHANGE (1 << 1)
- +#endif
- +
- +#ifndef F_LINUX_SPECIFIC_BASE
- +#define F_LINUX_SPECIFIC_BASE 1024
- +#endif
- +
- +#ifndef F_ADD_SEALS
- +#define F_ADD_SEALS (F_LINUX_SPECIFIC_BASE + 9)
- +#define F_GET_SEALS (F_LINUX_SPECIFIC_BASE + 10)
- +
- +#define F_SEAL_SEAL 0x0001 /* prevent further seals from being set */
- +#define F_SEAL_SHRINK 0x0002 /* prevent file from shrinking */
- +#define F_SEAL_GROW 0x0004 /* prevent file from growing */
- +#define F_SEAL_WRITE 0x0008 /* prevent writes */
- +#endif
- +
- +#ifndef MFD_ALLOW_SEALING
- +#define MFD_ALLOW_SEALING 0x0002U
- +#endif
- +
- +#ifndef MFD_CLOEXEC
- +#define MFD_CLOEXEC 0x0001U
- +#endif
- +
- +#include "glnx-missing-syscall.h"
- diff -Nuar ostree-2018.8.orig/libglnx/glnx-missing-syscall.h ostree-2018.8/libglnx/glnx-missing-syscall.h
- --- ostree-2018.8.orig/libglnx/glnx-missing-syscall.h 1970-01-01 02:00:00.000000000 +0200
- +++ ostree-2018.8/libglnx/glnx-missing-syscall.h 2018-05-26 00:37:32.007018846 +0300
- @@ -0,0 +1,156 @@
- +/***
- + This file was originally part of systemd.
- +
- + Copyright 2010 Lennart Poettering
- + Copyright 2016 Zbigniew Jędrzejewski-Szmek
- +
- + systemd is free software; you can redistribute it and/or modify it
- + under the terms of the GNU Lesser General Public License as published by
- + the Free Software Foundation; either version 2.1 of the License, or
- + (at your option) any later version.
- +
- + systemd is distributed in the hope that it will be useful, but
- + WITHOUT ANY WARRANTY; without even the implied warranty of
- + MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
- + Lesser General Public License for more details.
- +
- + You should have received a copy of the GNU Lesser General Public License
- + along with systemd; If not, see <http://www.gnu.org/licenses/>.
- +***/
- +
- +/* Missing glibc definitions to access certain kernel APIs.
- + This file is last updated from systemd git:
- +
- + commit 71e5200f94b22589922704aa4abdf95d4fe2e528
- + Author: Daniel Mack <daniel@zonque.org>
- + AuthorDate: Tue Oct 18 17:57:10 2016 +0200
- + Commit: Lennart Poettering <lennart@poettering.net>
- + CommitDate: Fri Sep 22 15:24:54 2017 +0200
- +
- + Add abstraction model for BPF programs
- +*/
- +
- +#include "config.h"
- +
- +#if !HAVE_DECL_RENAMEAT2
- +# ifndef __NR_renameat2
- +# if defined __x86_64__
- +# define __NR_renameat2 316
- +# elif defined __arm__
- +# define __NR_renameat2 382
- +# elif defined __aarch64__
- +# define __NR_renameat2 276
- +# elif defined _MIPS_SIM
- +# if _MIPS_SIM == _MIPS_SIM_ABI32
- +# define __NR_renameat2 4351
- +# endif
- +# if _MIPS_SIM == _MIPS_SIM_NABI32
- +# define __NR_renameat2 6315
- +# endif
- +# if _MIPS_SIM == _MIPS_SIM_ABI64
- +# define __NR_renameat2 5311
- +# endif
- +# elif defined __i386__
- +# define __NR_renameat2 353
- +# elif defined __powerpc64__
- +# define __NR_renameat2 357
- +# elif defined __s390__ || defined __s390x__
- +# define __NR_renameat2 347
- +# elif defined __arc__
- +# define __NR_renameat2 276
- +# else
- +# warning "__NR_renameat2 unknown for your architecture"
- +# endif
- +# endif
- +
- +static inline int renameat2(int oldfd, const char *oldname, int newfd, const char *newname, unsigned flags) {
- +# ifdef __NR_renameat2
- + return syscall(__NR_renameat2, oldfd, oldname, newfd, newname, flags);
- +# else
- + errno = ENOSYS;
- + return -1;
- +# endif
- +}
- +#endif
- +
- +#if !HAVE_DECL_MEMFD_CREATE
- +# ifndef __NR_memfd_create
- +# if defined __x86_64__
- +# define __NR_memfd_create 319
- +# elif defined __arm__
- +# define __NR_memfd_create 385
- +# elif defined __aarch64__
- +# define __NR_memfd_create 279
- +# elif defined __s390__
- +# define __NR_memfd_create 350
- +# elif defined _MIPS_SIM
- +# if _MIPS_SIM == _MIPS_SIM_ABI32
- +# define __NR_memfd_create 4354
- +# endif
- +# if _MIPS_SIM == _MIPS_SIM_NABI32
- +# define __NR_memfd_create 6318
- +# endif
- +# if _MIPS_SIM == _MIPS_SIM_ABI64
- +# define __NR_memfd_create 5314
- +# endif
- +# elif defined __i386__
- +# define __NR_memfd_create 356
- +# elif defined __arc__
- +# define __NR_memfd_create 279
- +# else
- +# warning "__NR_memfd_create unknown for your architecture"
- +# endif
- +# endif
- +
- +static inline int memfd_create(const char *name, unsigned int flags) {
- +# ifdef __NR_memfd_create
- + return syscall(__NR_memfd_create, name, flags);
- +# else
- + errno = ENOSYS;
- + return -1;
- +# endif
- +}
- +#endif
- +
- +/* Copied from systemd git:
- + commit 6bda23dd6aaba50cf8e3e6024248cf736cc443ca
- + Author: Yu Watanabe <watanabe.yu+github@gmail.com>
- + AuthorDate: Thu Jul 27 20:22:54 2017 +0900
- + Commit: Zbigniew Jędrzejewski-Szmek <zbyszek@in.waw.pl>
- + CommitDate: Thu Jul 27 07:22:54 2017 -0400
- +*/
- +#if !HAVE_DECL_COPY_FILE_RANGE
- +# ifndef __NR_copy_file_range
- +# if defined(__x86_64__)
- +# define __NR_copy_file_range 326
- +# elif defined(__i386__)
- +# define __NR_copy_file_range 377
- +# elif defined __s390__
- +# define __NR_copy_file_range 375
- +# elif defined __arm__
- +# define __NR_copy_file_range 391
- +# elif defined __aarch64__
- +# define __NR_copy_file_range 285
- +# elif defined __powerpc__
- +# define __NR_copy_file_range 379
- +# elif defined __arc__
- +# define __NR_copy_file_range 285
- +# else
- +# warning "__NR_copy_file_range not defined for your architecture"
- +# endif
- +# endif
- +
- +static inline ssize_t missing_copy_file_range(int fd_in, loff_t *off_in,
- + int fd_out, loff_t *off_out,
- + size_t len,
- + unsigned int flags) {
- +# ifdef __NR_copy_file_range
- + return syscall(__NR_copy_file_range, fd_in, off_in, fd_out, off_out, len, flags);
- +# else
- + errno = ENOSYS;
- + return -1;
- +# endif
- +}
- +
- +# define copy_file_range missing_copy_file_range
- +#endif
- diff -Nuar ostree-2018.8.orig/libglnx/glnx-shutil.c ostree-2018.8/libglnx/glnx-shutil.c
- --- ostree-2018.8.orig/libglnx/glnx-shutil.c 1970-01-01 02:00:00.000000000 +0200
- +++ ostree-2018.8/libglnx/glnx-shutil.c 2018-05-26 00:37:32.007018846 +0300
- @@ -0,0 +1,260 @@
- +/* -*- mode: C; c-file-style: "gnu"; indent-tabs-mode: nil; -*-
- + *
- + * Copyright (C) 2014,2015 Colin Walters <walters@verbum.org>.
- + *
- + * This library is free software; you can redistribute it and/or
- + * modify it under the terms of the GNU Lesser General Public
- + * License as published by the Free Software Foundation; either
- + * version 2 of the License, or (at your option) any later version.
- + *
- + * This library is distributed in the hope that it will be useful,
- + * but WITHOUT ANY WARRANTY; without even the implied warranty of
- + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
- + * Lesser General Public License for more details.
- + *
- + * You should have received a copy of the GNU Lesser General Public
- + * License along with this library; if not, write to the
- + * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
- + * Boston, MA 02111-1307, USA.
- + */
- +
- +#include "config.h"
- +
- +#include <string.h>
- +
- +#include <glnx-shutil.h>
- +#include <glnx-errors.h>
- +#include <glnx-local-alloc.h>
- +
- +static gboolean
- +glnx_shutil_rm_rf_children (GLnxDirFdIterator *dfd_iter,
- + GCancellable *cancellable,
- + GError **error)
- +{
- + struct dirent *dent;
- +
- + while (TRUE)
- + {
- + if (!glnx_dirfd_iterator_next_dent_ensure_dtype (dfd_iter, &dent, cancellable, error))
- + return FALSE;
- + if (dent == NULL)
- + break;
- +
- + if (dent->d_type == DT_DIR)
- + {
- + g_auto(GLnxDirFdIterator) child_dfd_iter = { 0, };
- +
- + if (!glnx_dirfd_iterator_init_at (dfd_iter->fd, dent->d_name, FALSE,
- + &child_dfd_iter, error))
- + return FALSE;
- +
- + if (!glnx_shutil_rm_rf_children (&child_dfd_iter, cancellable, error))
- + return FALSE;
- +
- + if (unlinkat (dfd_iter->fd, dent->d_name, AT_REMOVEDIR) == -1)
- + return glnx_throw_errno_prefix (error, "unlinkat");
- + }
- + else
- + {
- + if (unlinkat (dfd_iter->fd, dent->d_name, 0) == -1)
- + {
- + if (errno != ENOENT)
- + return glnx_throw_errno_prefix (error, "unlinkat");
- + }
- + }
- + }
- +
- + return TRUE;
- +}
- +
- +/**
- + * glnx_shutil_rm_rf_at:
- + * @dfd: A directory file descriptor, or `AT_FDCWD` or `-1` for current
- + * @path: Path
- + * @cancellable: Cancellable
- + * @error: Error
- + *
- + * Recursively delete the filename referenced by the combination of
- + * the directory fd @dfd and @path; it may be a file or directory. No
- + * error is thrown if @path does not exist.
- + */
- +gboolean
- +glnx_shutil_rm_rf_at (int dfd,
- + const char *path,
- + GCancellable *cancellable,
- + GError **error)
- +{
- + dfd = glnx_dirfd_canonicalize (dfd);
- +
- +
- + /* With O_NOFOLLOW first */
- + glnx_autofd int target_dfd =
- + openat (dfd, path, O_RDONLY | O_NONBLOCK | O_DIRECTORY | O_CLOEXEC | O_NOFOLLOW);
- +
- + if (target_dfd == -1)
- + {
- + int errsv = errno;
- + if (errsv == ENOENT)
- + {
- + ;
- + }
- + else if (errsv == ENOTDIR || errsv == ELOOP)
- + {
- + if (unlinkat (dfd, path, 0) != 0)
- + return glnx_throw_errno_prefix (error, "unlinkat");
- + }
- + else
- + return glnx_throw_errno_prefix (error, "open(%s)", path);
- + }
- + else
- + {
- + g_auto(GLnxDirFdIterator) dfd_iter = { 0, };
- + if (!glnx_dirfd_iterator_init_take_fd (&target_dfd, &dfd_iter, error))
- + return FALSE;
- +
- + if (!glnx_shutil_rm_rf_children (&dfd_iter, cancellable, error))
- + return FALSE;
- +
- + if (unlinkat (dfd, path, AT_REMOVEDIR) == -1)
- + {
- + if (errno != ENOENT)
- + return glnx_throw_errno_prefix (error, "unlinkat");
- + }
- + }
- +
- + return TRUE;
- +}
- +
- +static gboolean
- +mkdir_p_at_internal (int dfd,
- + char *path,
- + int mode,
- + GCancellable *cancellable,
- + GError **error)
- +{
- + gboolean did_recurse = FALSE;
- +
- + if (g_cancellable_set_error_if_cancelled (cancellable, error))
- + return FALSE;
- +
- + again:
- + if (mkdirat (dfd, path, mode) == -1)
- + {
- + if (errno == ENOENT)
- + {
- + char *lastslash;
- +
- + g_assert (!did_recurse);
- +
- + lastslash = strrchr (path, '/');
- + if (lastslash == NULL)
- + {
- + /* This can happen if @dfd was deleted between being opened and
- + * passed to mkdir_p_at_internal(). */
- + return glnx_throw_errno_prefix (error, "mkdir(%s)", path);
- + }
- +
- + /* Note we can mutate the buffer as we dup'd it */
- + *lastslash = '\0';
- +
- + if (!glnx_shutil_mkdir_p_at (dfd, path, mode,
- + cancellable, error))
- + return FALSE;
- +
- + /* Now restore it for another mkdir attempt */
- + *lastslash = '/';
- +
- + did_recurse = TRUE;
- + goto again;
- + }
- + else if (errno == EEXIST)
- + {
- + /* Fall through; it may not have been a directory,
- + * but we'll find that out on the next call up.
- + */
- + }
- + else
- + return glnx_throw_errno_prefix (error, "mkdir(%s)", path);
- + }
- +
- + return TRUE;
- +}
- +
- +/**
- + * glnx_shutil_mkdir_p_at:
- + * @dfd: Directory fd
- + * @path: Directory path to be created
- + * @mode: Mode for newly created directories
- + * @cancellable: Cancellable
- + * @error: Error
- + *
- + * Similar to g_mkdir_with_parents(), except operates relative to the
- + * directory fd @dfd.
- + *
- + * See also glnx_ensure_dir() for a non-recursive version.
- + *
- + * This will return %G_IO_ERROR_NOT_FOUND if @dfd has been deleted since being
- + * opened. It may return other errors from mkdirat() in other situations.
- + */
- +gboolean
- +glnx_shutil_mkdir_p_at (int dfd,
- + const char *path,
- + int mode,
- + GCancellable *cancellable,
- + GError **error)
- +{
- + struct stat stbuf;
- + char *buf;
- +
- + /* Fast path stat to see whether it already exists */
- + if (fstatat (dfd, path, &stbuf, AT_SYMLINK_NOFOLLOW) == 0)
- + {
- + /* Note early return */
- + if (S_ISDIR (stbuf.st_mode))
- + return TRUE;
- + }
- +
- + buf = strdupa (path);
- +
- + if (!mkdir_p_at_internal (dfd, buf, mode, cancellable, error))
- + return FALSE;
- +
- + return TRUE;
- +}
- +
- +/**
- + * glnx_shutil_mkdir_p_at_open:
- + * @dfd: Directory fd
- + * @path: Directory path to be created
- + * @mode: Mode for newly created directories
- + * @out_dfd: (out caller-allocates): Return location for an FD to @dfd/@path,
- + * or `-1` on error
- + * @cancellable: (nullable): Cancellable, or %NULL
- + * @error: Return location for a #GError, or %NULL
- + *
- + * Similar to glnx_shutil_mkdir_p_at(), except it opens the resulting directory
- + * and returns a directory FD to it. Currently, this is not guaranteed to be
- + * race-free.
- + *
- + * Returns: %TRUE on success, %FALSE otherwise
- + * Since: UNRELEASED
- + */
- +gboolean
- +glnx_shutil_mkdir_p_at_open (int dfd,
- + const char *path,
- + int mode,
- + int *out_dfd,
- + GCancellable *cancellable,
- + GError **error)
- +{
- + /* FIXME: It’s not possible to eliminate the race here until
- + * openat(O_DIRECTORY | O_CREAT) works (and returns a directory rather than a
- + * file). It appears to be not supported in current kernels. (Tested with
- + * 4.10.10-200.fc25.x86_64.) */
- + *out_dfd = -1;
- +
- + if (!glnx_shutil_mkdir_p_at (dfd, path, mode, cancellable, error))
- + return FALSE;
- +
- + return glnx_opendirat (dfd, path, TRUE, out_dfd, error);
- +}
- diff -Nuar ostree-2018.8.orig/libglnx/glnx-shutil.h ostree-2018.8/libglnx/glnx-shutil.h
- --- ostree-2018.8.orig/libglnx/glnx-shutil.h 1970-01-01 02:00:00.000000000 +0200
- +++ ostree-2018.8/libglnx/glnx-shutil.h 2018-05-26 00:37:32.008018846 +0300
- @@ -0,0 +1,48 @@
- +/* -*- mode: C; c-file-style: "gnu"; indent-tabs-mode: nil; -*-
- + *
- + * Copyright (C) 2014,2015 Colin Walters <walters@verbum.org>.
- + *
- + * This library is free software; you can redistribute it and/or
- + * modify it under the terms of the GNU Lesser General Public
- + * License as published by the Free Software Foundation; either
- + * version 2 of the License, or (at your option) any later version.
- + *
- + * This library is distributed in the hope that it will be useful,
- + * but WITHOUT ANY WARRANTY; without even the implied warranty of
- + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
- + * Lesser General Public License for more details.
- + *
- + * You should have received a copy of the GNU Lesser General Public
- + * License along with this library; if not, write to the
- + * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
- + * Boston, MA 02111-1307, USA.
- + */
- +
- +#pragma once
- +
- +#include <glnx-dirfd.h>
- +
- +G_BEGIN_DECLS
- +
- +gboolean
- +glnx_shutil_rm_rf_at (int dfd,
- + const char *path,
- + GCancellable *cancellable,
- + GError **error);
- +
- +gboolean
- +glnx_shutil_mkdir_p_at (int dfd,
- + const char *path,
- + int mode,
- + GCancellable *cancellable,
- + GError **error);
- +
- +gboolean
- +glnx_shutil_mkdir_p_at_open (int dfd,
- + const char *path,
- + int mode,
- + int *out_dfd,
- + GCancellable *cancellable,
- + GError **error);
- +
- +G_END_DECLS
- diff -Nuar ostree-2018.8.orig/libglnx/glnx-xattrs.c ostree-2018.8/libglnx/glnx-xattrs.c
- --- ostree-2018.8.orig/libglnx/glnx-xattrs.c 1970-01-01 02:00:00.000000000 +0200
- +++ ostree-2018.8/libglnx/glnx-xattrs.c 2018-05-26 00:37:32.008018846 +0300
- @@ -0,0 +1,444 @@
- +/* -*- mode: C; c-file-style: "gnu"; indent-tabs-mode: nil; -*-
- + *
- + * Copyright (C) 2014,2015 Colin Walters <walters@verbum.org>.
- + *
- + * This library is free software; you can redistribute it and/or
- + * modify it under the terms of the GNU Lesser General Public
- + * License as published by the Free Software Foundation; either
- + * version 2 of the License, or (at your option) any later version.
- + *
- + * This library is distributed in the hope that it will be useful,
- + * but WITHOUT ANY WARRANTY; without even the implied warranty of
- + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
- + * Lesser General Public License for more details.
- + *
- + * You should have received a copy of the GNU Lesser General Public
- + * License along with this library; if not, write to the
- + * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
- + * Boston, MA 02111-1307, USA.
- + */
- +
- +#include "config.h"
- +
- +#include <string.h>
- +#include <stdio.h>
- +
- +#include <glnx-macros.h>
- +#include <glnx-xattrs.h>
- +#include <glnx-errors.h>
- +#include <glnx-local-alloc.h>
- +
- +static GVariant *
- +variant_new_ay_bytes (GBytes *bytes)
- +{
- + gsize size;
- + gconstpointer data;
- + data = g_bytes_get_data (bytes, &size);
- + g_bytes_ref (bytes);
- + return g_variant_new_from_data (G_VARIANT_TYPE ("ay"), data, size,
- + TRUE, (GDestroyNotify)g_bytes_unref, bytes);
- +}
- +
- +static char *
- +canonicalize_xattrs (char *xattr_string,
- + size_t len)
- +{
- + char *p;
- + GSList *xattrs = NULL;
- + GSList *iter;
- + GString *result;
- +
- + result = g_string_new (0);
- +
- + p = xattr_string;
- + while (p < xattr_string+len)
- + {
- + xattrs = g_slist_prepend (xattrs, p);
- + p += strlen (p) + 1;
- + }
- +
- + xattrs = g_slist_sort (xattrs, (GCompareFunc) strcmp);
- + for (iter = xattrs; iter; iter = iter->next) {
- + g_string_append (result, iter->data);
- + g_string_append_c (result, '\0');
- + }
- +
- + g_slist_free (xattrs);
- + return g_string_free (result, FALSE);
- +}
- +
- +static gboolean
- +read_xattr_name_array (const char *path,
- + int fd,
- + const char *xattrs,
- + size_t len,
- + GVariantBuilder *builder,
- + GError **error)
- +{
- + gboolean ret = FALSE;
- + const char *p;
- + int r;
- + const char *funcstr;
- +
- + g_assert (path != NULL || fd != -1);
- +
- + funcstr = fd != -1 ? "fgetxattr" : "lgetxattr";
- +
- + for (p = xattrs; p < xattrs+len; p = p + strlen (p) + 1)
- + {
- + ssize_t bytes_read;
- + g_autofree char *buf = NULL;
- + g_autoptr(GBytes) bytes = NULL;
- +
- + again:
- + if (fd != -1)
- + bytes_read = fgetxattr (fd, p, NULL, 0);
- + else
- + bytes_read = lgetxattr (path, p, NULL, 0);
- + if (bytes_read < 0)
- + {
- + if (errno == ENODATA)
- + continue;
- +
- + glnx_set_prefix_error_from_errno (error, "%s", funcstr);
- + goto out;
- + }
- + if (bytes_read == 0)
- + continue;
- +
- + buf = g_malloc (bytes_read);
- + if (fd != -1)
- + r = fgetxattr (fd, p, buf, bytes_read);
- + else
- + r = lgetxattr (path, p, buf, bytes_read);
- + if (r < 0)
- + {
- + if (errno == ERANGE)
- + {
- + g_free (g_steal_pointer (&buf));
- + goto again;
- + }
- + else if (errno == ENODATA)
- + continue;
- +
- + glnx_set_prefix_error_from_errno (error, "%s", funcstr);
- + goto out;
- + }
- +
- + bytes = g_bytes_new_take (g_steal_pointer (&buf), bytes_read);
- + g_variant_builder_add (builder, "(@ay@ay)",
- + g_variant_new_bytestring (p),
- + variant_new_ay_bytes (bytes));
- + }
- +
- + ret = TRUE;
- + out:
- + return ret;
- +}
- +
- +static gboolean
- +get_xattrs_impl (const char *path,
- + int fd,
- + GVariant **out_xattrs,
- + GCancellable *cancellable,
- + GError **error)
- +{
- + gboolean ret = FALSE;
- + ssize_t bytes_read, real_size;
- + g_autofree char *xattr_names = NULL;
- + g_autofree char *xattr_names_canonical = NULL;
- + GVariantBuilder builder;
- + gboolean builder_initialized = FALSE;
- + g_autoptr(GVariant) ret_xattrs = NULL;
- +
- + g_assert (path != NULL || fd != -1);
- +
- + g_variant_builder_init (&builder, G_VARIANT_TYPE ("a(ayay)"));
- + builder_initialized = TRUE;
- +
- + again:
- + if (path)
- + bytes_read = llistxattr (path, NULL, 0);
- + else
- + bytes_read = flistxattr (fd, NULL, 0);
- +
- + if (bytes_read < 0)
- + {
- + if (errno != ENOTSUP)
- + {
- + glnx_set_prefix_error_from_errno (error, "%s", "llistxattr");
- + goto out;
- + }
- + }
- + else if (bytes_read > 0)
- + {
- + xattr_names = g_malloc (bytes_read);
- + if (path)
- + real_size = llistxattr (path, xattr_names, bytes_read);
- + else
- + real_size = flistxattr (fd, xattr_names, bytes_read);
- + if (real_size < 0)
- + {
- + if (errno == ERANGE)
- + {
- + g_free (xattr_names);
- + goto again;
- + }
- + glnx_set_prefix_error_from_errno (error, "%s", "llistxattr");
- + goto out;
- + }
- + else if (real_size > 0)
- + {
- + xattr_names_canonical = canonicalize_xattrs (xattr_names, real_size);
- +
- + if (!read_xattr_name_array (path, fd, xattr_names_canonical, real_size, &builder, error))
- + goto out;
- + }
- + }
- +
- + ret_xattrs = g_variant_builder_end (&builder);
- + builder_initialized = FALSE;
- + g_variant_ref_sink (ret_xattrs);
- +
- + ret = TRUE;
- + if (out_xattrs)
- + *out_xattrs = g_steal_pointer (&ret_xattrs);
- + out:
- + if (!builder_initialized)
- + g_variant_builder_clear (&builder);
- + return ret;
- +}
- +
- +/**
- + * glnx_fd_get_all_xattrs:
- + * @fd: a file descriptor
- + * @out_xattrs: (out): A new #GVariant containing the extended attributes
- + * @cancellable: Cancellable
- + * @error: Error
- + *
- + * Read all extended attributes from @fd in a canonical sorted order, and
- + * set @out_xattrs with the result.
- + *
- + * If the filesystem does not support extended attributes, @out_xattrs
- + * will have 0 elements, and this function will return successfully.
- + */
- +gboolean
- +glnx_fd_get_all_xattrs (int fd,
- + GVariant **out_xattrs,
- + GCancellable *cancellable,
- + GError **error)
- +{
- + return get_xattrs_impl (NULL, fd, out_xattrs,
- + cancellable, error);
- +}
- +
- +/**
- + * glnx_dfd_name_get_all_xattrs:
- + * @dfd: Parent directory file descriptor
- + * @name: File name
- + * @out_xattrs: (out): Extended attribute set
- + * @cancellable: Cancellable
- + * @error: Error
- + *
- + * Load all extended attributes for the file named @name residing in
- + * directory @dfd.
- + */
- +gboolean
- +glnx_dfd_name_get_all_xattrs (int dfd,
- + const char *name,
- + GVariant **out_xattrs,
- + GCancellable *cancellable,
- + GError **error)
- +{
- + if (G_IN_SET(dfd, AT_FDCWD, -1))
- + {
- + return get_xattrs_impl (name, -1, out_xattrs, cancellable, error);
- + }
- + else
- + {
- + char buf[PATH_MAX];
- + /* A workaround for the lack of lgetxattrat(), thanks to Florian Weimer:
- + * https://mail.gnome.org/archives/ostree-list/2014-February/msg00017.html
- + */
- + snprintf (buf, sizeof (buf), "/proc/self/fd/%d/%s", dfd, name);
- + return get_xattrs_impl (buf, -1, out_xattrs, cancellable, error);
- + }
- +}
- +
- +static gboolean
- +set_all_xattrs_for_path (const char *path,
- + GVariant *xattrs,
- + GCancellable *cancellable,
- + GError **error)
- +{
- + const guint n = g_variant_n_children (xattrs);
- + for (guint i = 0; i < n; i++)
- + {
- + const guint8* name;
- + g_autoptr(GVariant) value = NULL;
- + g_variant_get_child (xattrs, i, "(^&ay@ay)",
- + &name, &value);
- +
- + gsize value_len;
- + const guint8* value_data = g_variant_get_fixed_array (value, &value_len, 1);
- +
- + if (lsetxattr (path, (char*)name, (char*)value_data, value_len, 0) < 0)
- + return glnx_throw_errno_prefix (error, "lsetxattr");
- + }
- +
- + return TRUE;
- +}
- +
- +/**
- + * glnx_dfd_name_set_all_xattrs:
- + * @dfd: Parent directory file descriptor
- + * @name: File name
- + * @xattrs: Extended attribute set
- + * @cancellable: Cancellable
- + * @error: Error
- + *
- + * Set all extended attributes for the file named @name residing in
- + * directory @dfd.
- + */
- +gboolean
- +glnx_dfd_name_set_all_xattrs (int dfd,
- + const char *name,
- + GVariant *xattrs,
- + GCancellable *cancellable,
- + GError **error)
- +{
- + if (G_IN_SET(dfd, AT_FDCWD, -1))
- + {
- + return set_all_xattrs_for_path (name, xattrs, cancellable, error);
- + }
- + else
- + {
- + char buf[PATH_MAX];
- + /* A workaround for the lack of lsetxattrat(), thanks to Florian Weimer:
- + * https://mail.gnome.org/archives/ostree-list/2014-February/msg00017.html
- + */
- + snprintf (buf, sizeof (buf), "/proc/self/fd/%d/%s", dfd, name);
- + return set_all_xattrs_for_path (buf, xattrs, cancellable, error);
- + }
- +}
- +
- +/**
- + * glnx_fd_set_all_xattrs:
- + * @fd: File descriptor
- + * @xattrs: Extended attributes
- + * @cancellable: Cancellable
- + * @error: Error
- + *
- + * For each attribute in @xattrs, set its value on the file or
- + * directory referred to by @fd. This function does not remove any
- + * attributes not in @xattrs.
- + */
- +gboolean
- +glnx_fd_set_all_xattrs (int fd,
- + GVariant *xattrs,
- + GCancellable *cancellable,
- + GError **error)
- +{
- + const guint n = g_variant_n_children (xattrs);
- + for (guint i = 0; i < n; i++)
- + {
- + const guint8* name;
- + g_autoptr(GVariant) value = NULL;
- + g_variant_get_child (xattrs, i, "(^&ay@ay)",
- + &name, &value);
- +
- + gsize value_len;
- + const guint8* value_data = g_variant_get_fixed_array (value, &value_len, 1);
- +
- + if (TEMP_FAILURE_RETRY (fsetxattr (fd, (char*)name, (char*)value_data, value_len, 0)) < 0)
- + return glnx_throw_errno_prefix (error, "fsetxattr");
- + }
- +
- + return TRUE;
- +}
- +
- +/**
- + * glnx_lgetxattrat:
- + * @dfd: Directory file descriptor
- + * @subpath: Subpath
- + * @attribute: Extended attribute to retrieve
- + * @error: Error
- + *
- + * Retrieve an extended attribute value, relative to a directory file
- + * descriptor.
- + */
- +GBytes *
- +glnx_lgetxattrat (int dfd,
- + const char *subpath,
- + const char *attribute,
- + GError **error)
- +{
- + char pathbuf[PATH_MAX];
- + snprintf (pathbuf, sizeof (pathbuf), "/proc/self/fd/%d/%s", dfd, subpath);
- +
- + ssize_t bytes_read, real_size;
- + if (TEMP_FAILURE_RETRY (bytes_read = lgetxattr (pathbuf, attribute, NULL, 0)) < 0)
- + return glnx_null_throw_errno_prefix (error, "lgetxattr");
- +
- + g_autofree guint8 *buf = g_malloc (bytes_read);
- + if (TEMP_FAILURE_RETRY (real_size = lgetxattr (pathbuf, attribute, buf, bytes_read)) < 0)
- + return glnx_null_throw_errno_prefix (error, "lgetxattr");
- +
- + return g_bytes_new_take (g_steal_pointer (&buf), real_size);
- +}
- +
- +/**
- + * glnx_fgetxattr_bytes:
- + * @fd: Directory file descriptor
- + * @attribute: Extended attribute to retrieve
- + * @error: Error
- + *
- + * Returns: (transfer full): An extended attribute value, or %NULL on error
- + */
- +GBytes *
- +glnx_fgetxattr_bytes (int fd,
- + const char *attribute,
- + GError **error)
- +{
- + ssize_t bytes_read, real_size;
- +
- + if (TEMP_FAILURE_RETRY (bytes_read = fgetxattr (fd, attribute, NULL, 0)) < 0)
- + return glnx_null_throw_errno_prefix (error, "fgetxattr");
- +
- + g_autofree guint8 *buf = g_malloc (bytes_read);
- + if (TEMP_FAILURE_RETRY (real_size = fgetxattr (fd, attribute, buf, bytes_read)) < 0)
- + return glnx_null_throw_errno_prefix (error, "fgetxattr");
- +
- + return g_bytes_new_take (g_steal_pointer (&buf), real_size);
- +}
- +
- +/**
- + * glnx_lsetxattrat:
- + * @dfd: Directory file descriptor
- + * @subpath: Path
- + * @attribute: An attribute name
- + * @value: (array length=len) (element-type guint8): Attribute value
- + * @len: Length of @value
- + * @flags: Flags, containing either XATTR_CREATE or XATTR_REPLACE
- + * @error: Error
- + *
- + * Set an extended attribute, relative to a directory file descriptor.
- + */
- +gboolean
- +glnx_lsetxattrat (int dfd,
- + const char *subpath,
- + const char *attribute,
- + const guint8 *value,
- + gsize len,
- + int flags,
- + GError **error)
- +{
- + char pathbuf[PATH_MAX];
- + snprintf (pathbuf, sizeof (pathbuf), "/proc/self/fd/%d/%s", dfd, subpath);
- +
- + if (TEMP_FAILURE_RETRY (lsetxattr (subpath, attribute, value, len, flags)) < 0)
- + return glnx_throw_errno_prefix (error, "lsetxattr");
- +
- + return TRUE;
- +}
- +
- diff -Nuar ostree-2018.8.orig/libglnx/glnx-xattrs.h ostree-2018.8/libglnx/glnx-xattrs.h
- --- ostree-2018.8.orig/libglnx/glnx-xattrs.h 1970-01-01 02:00:00.000000000 +0200
- +++ ostree-2018.8/libglnx/glnx-xattrs.h 2018-05-26 00:37:32.008018846 +0300
- @@ -0,0 +1,78 @@
- +/* -*- mode: C; c-file-style: "gnu"; indent-tabs-mode: nil; -*-
- + *
- + * Copyright (C) 2014,2015 Colin Walters <walters@verbum.org>.
- + *
- + * This library is free software; you can redistribute it and/or
- + * modify it under the terms of the GNU Lesser General Public
- + * License as published by the Free Software Foundation; either
- + * version 2 of the License, or (at your option) any later version.
- + *
- + * This library is distributed in the hope that it will be useful,
- + * but WITHOUT ANY WARRANTY; without even the implied warranty of
- + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
- + * Lesser General Public License for more details.
- + *
- + * You should have received a copy of the GNU Lesser General Public
- + * License along with this library; if not, write to the
- + * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
- + * Boston, MA 02111-1307, USA.
- + */
- +
- +#pragma once
- +
- +#include <glnx-backport-autocleanups.h>
- +#include <limits.h>
- +#include <dirent.h>
- +#include <sys/stat.h>
- +#include <fcntl.h>
- +#include <sys/xattr.h>
- +
- +G_BEGIN_DECLS
- +
- +gboolean
- +glnx_dfd_name_get_all_xattrs (int dfd,
- + const char *name,
- + GVariant **out_xattrs,
- + GCancellable *cancellable,
- + GError **error);
- +
- +gboolean
- +glnx_fd_get_all_xattrs (int fd,
- + GVariant **out_xattrs,
- + GCancellable *cancellable,
- + GError **error);
- +
- +gboolean
- +glnx_dfd_name_set_all_xattrs (int dfd,
- + const char *name,
- + GVariant *xattrs,
- + GCancellable *cancellable,
- + GError **error);
- +
- +gboolean
- +glnx_fd_set_all_xattrs (int fd,
- + GVariant *xattrs,
- + GCancellable *cancellable,
- + GError **error);
- +
- +GBytes *
- +glnx_lgetxattrat (int dfd,
- + const char *subpath,
- + const char *attribute,
- + GError **error);
- +
- +GBytes *
- +glnx_fgetxattr_bytes (int fd,
- + const char *attribute,
- + GError **error);
- +
- +gboolean
- +glnx_lsetxattrat (int dfd,
- + const char *subpath,
- + const char *attribute,
- + const guint8 *value,
- + gsize len,
- + int flags,
- + GError **error);
- +
- +G_END_DECLS
- diff -Nuar ostree-2018.8.orig/libglnx/libglnx.doap ostree-2018.8/libglnx/libglnx.doap
- --- ostree-2018.8.orig/libglnx/libglnx.doap 1970-01-01 02:00:00.000000000 +0200
- +++ ostree-2018.8/libglnx/libglnx.doap 2018-05-26 00:37:32.008018846 +0300
- @@ -0,0 +1,31 @@
- +<?xml version="1.0" encoding="UTF-8"?>
- +<Project xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#"
- + xmlns:rdfs="http://www.w3.org/2000/01/rdf-schema#"
- + xmlns:foaf="http://xmlns.com/foaf/0.1/"
- + xmlns:gnome="http://api.gnome.org/doap-extensions#"
- + xmlns="http://usefulinc.com/ns/doap#">
- +
- + <name>libglnx</name>
- + <shortname>libglnx</shortname>
- +
- + <shortdesc xml:lang="en">"Copylib" for system service modules using GLib with Linux</shortdesc>
- +
- + <description xml:lang="en">This module is intended for use by
- + infrastructure code using GLib that is also Linux specific, such as
- + ostree, NetworkManager, and others.
- + </description>
- +
- + <license rdf:resource="http://usefulinc.com/doap/licenses/lgpl" />
- + <mailing-list rdf:resource="mailto:desktop-devel-list@gnome.org" />
- +
- + <programming-language>C</programming-language>
- +
- + <maintainer>
- + <foaf:Person>
- + <foaf:name>Colin Walters</foaf:name>
- + <foaf:mbox rdf:resource="mailto:walters@verbum.org"/>
- + <gnome:userid>walters</gnome:userid>
- + </foaf:Person>
- + </maintainer>
- +
- +</Project>
- diff -Nuar ostree-2018.8.orig/libglnx/libglnx.h ostree-2018.8/libglnx/libglnx.h
- --- ostree-2018.8.orig/libglnx/libglnx.h 1970-01-01 02:00:00.000000000 +0200
- +++ ostree-2018.8/libglnx/libglnx.h 2018-05-26 00:37:32.008018846 +0300
- @@ -0,0 +1,40 @@
- +/* -*- mode: C; c-file-style: "gnu"; indent-tabs-mode: nil; -*-
- + *
- + * Copyright (C) 2012,2013,2015 Colin Walters <walters@verbum.org>.
- + *
- + * This library is free software; you can redistribute it and/or
- + * modify it under the terms of the GNU Lesser General Public
- + * License as published by the Free Software Foundation; either
- + * version 2 of the License, or (at your option) any later version.
- + *
- + * This library is distributed in the hope that it will be useful,
- + * but WITHOUT ANY WARRANTY; without even the implied warranty of
- + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
- + * Lesser General Public License for more details.
- + *
- + * You should have received a copy of the GNU Lesser General Public
- + * License along with this library; if not, write to the
- + * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
- + * Boston, MA 02111-1307, USA.
- + */
- +
- +#pragma once
- +
- +#include <gio/gio.h>
- +
- +G_BEGIN_DECLS
- +
- +#include <glnx-macros.h>
- +#include <glnx-missing.h>
- +#include <glnx-local-alloc.h>
- +#include <glnx-backport-autocleanups.h>
- +#include <glnx-backports.h>
- +#include <glnx-lockfile.h>
- +#include <glnx-errors.h>
- +#include <glnx-dirfd.h>
- +#include <glnx-shutil.h>
- +#include <glnx-xattrs.h>
- +#include <glnx-console.h>
- +#include <glnx-fdio.h>
- +
- +G_END_DECLS
- diff -Nuar ostree-2018.8.orig/libglnx/libglnx.m4 ostree-2018.8/libglnx/libglnx.m4
- --- ostree-2018.8.orig/libglnx/libglnx.m4 1970-01-01 02:00:00.000000000 +0200
- +++ ostree-2018.8/libglnx/libglnx.m4 2018-05-26 00:37:32.008018846 +0300
- @@ -0,0 +1,33 @@
- +AC_DEFUN([LIBGLNX_CONFIGURE],
- +[
- +AC_CHECK_DECLS([
- + renameat2,
- + memfd_create,
- + copy_file_range],
- + [], [], [[
- +#include <sys/types.h>
- +#include <unistd.h>
- +#include <sys/mount.h>
- +#include <fcntl.h>
- +#include <sched.h>
- +#include <linux/loop.h>
- +#include <linux/random.h>
- +#include <sys/mman.h>
- +]])
- +
- +AC_ARG_ENABLE(otmpfile,
- + [AS_HELP_STRING([--disable-otmpfile],
- + [Disable use of O_TMPFILE [default=no]])],,
- + [enable_otmpfile=yes])
- +AS_IF([test $enable_otmpfile = yes], [], [
- + AC_DEFINE([DISABLE_OTMPFILE], 1, [Define if we should avoid using O_TMPFILE])])
- +
- +AC_ARG_ENABLE(wrpseudo-compat,
- + [AS_HELP_STRING([--enable-wrpseudo-compat],
- + [Disable use syscall() and filesystem calls to for compatibility with wrpseudo [default=no]])],,
- + [enable_wrpseudo_compat=no])
- +AS_IF([test $enable_wrpseudo_compat = no], [], [
- + AC_DEFINE([ENABLE_WRPSEUDO_COMPAT], 1, [Define if we should be compatible with wrpseudo])])
- +
- +dnl end LIBGLNX_CONFIGURE
- +])
- diff -Nuar ostree-2018.8.orig/libglnx/Makefile-libglnx.am ostree-2018.8/libglnx/Makefile-libglnx.am
- --- ostree-2018.8.orig/libglnx/Makefile-libglnx.am 1970-01-01 02:00:00.000000000 +0200
- +++ ostree-2018.8/libglnx/Makefile-libglnx.am 2018-05-26 00:37:32.006018846 +0300
- @@ -0,0 +1,78 @@
- +# Copyright (C) 2015 Colin Walters <walters@verbum.org>
- +#
- +# This library is free software; you can redistribute it and/or
- +# modify it under the terms of the GNU Lesser General Public
- +# License as published by the Free Software Foundation; either
- +# version 2 of the License, or (at your option) any later version.
- +#
- +# This library is distributed in the hope that it will be useful,
- +# but WITHOUT ANY WARRANTY; without even the implied warranty of
- +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
- +# Lesser General Public License for more details.
- +#
- +# You should have received a copy of the GNU Lesser General Public
- +# License along with this library; if not, write to the
- +# Free Software Foundation, Inc., 59 Temple Place - Suite 330,
- +# Boston, MA 02111-1307, USA.
- +
- +EXTRA_DIST += \
- + $(libglnx_srcpath)/README.md \
- + $(libglnx_srcpath)/COPYING \
- + $(libglnx_srcpath)/libglnx.m4 \
- + $(NULL)
- +
- +libglnx_la_SOURCES = \
- + $(libglnx_srcpath)/glnx-macros.h \
- + $(libglnx_srcpath)/glnx-backport-autocleanups.h \
- + $(libglnx_srcpath)/glnx-backport-autoptr.h \
- + $(libglnx_srcpath)/glnx-backports.h \
- + $(libglnx_srcpath)/glnx-backports.c \
- + $(libglnx_srcpath)/glnx-local-alloc.h \
- + $(libglnx_srcpath)/glnx-local-alloc.c \
- + $(libglnx_srcpath)/glnx-errors.h \
- + $(libglnx_srcpath)/glnx-errors.c \
- + $(libglnx_srcpath)/glnx-console.h \
- + $(libglnx_srcpath)/glnx-console.c \
- + $(libglnx_srcpath)/glnx-dirfd.h \
- + $(libglnx_srcpath)/glnx-dirfd.c \
- + $(libglnx_srcpath)/glnx-fdio.h \
- + $(libglnx_srcpath)/glnx-fdio.c \
- + $(libglnx_srcpath)/glnx-lockfile.h \
- + $(libglnx_srcpath)/glnx-lockfile.c \
- + $(libglnx_srcpath)/glnx-missing-syscall.h \
- + $(libglnx_srcpath)/glnx-missing.h \
- + $(libglnx_srcpath)/glnx-xattrs.h \
- + $(libglnx_srcpath)/glnx-xattrs.c \
- + $(libglnx_srcpath)/glnx-shutil.h \
- + $(libglnx_srcpath)/glnx-shutil.c \
- + $(libglnx_srcpath)/libglnx.h \
- + $(libglnx_srcpath)/tests/libglnx-testlib.h \
- + $(NULL)
- +
- +libglnx_la_CFLAGS = $(AM_CFLAGS) $(libglnx_cflags)
- +libglnx_la_LDFLAGS = -avoid-version -Bsymbolic-functions -export-symbols-regex "^glnx_" -no-undefined -export-dynamic
- +libglnx_la_LIBADD = $(libglnx_libs)
- +
- +libglnx_tests = test-libglnx-xattrs test-libglnx-fdio test-libglnx-errors test-libglnx-macros test-libglnx-shutil
- +TESTS += $(libglnx_tests)
- +
- +check_PROGRAMS += $(libglnx_tests)
- +test_libglnx_xattrs_SOURCES = $(libglnx_srcpath)/tests/test-libglnx-xattrs.c
- +test_libglnx_xattrs_CFLAGS = $(AM_CFLAGS) $(libglnx_cflags)
- +test_libglnx_xattrs_LDADD = $(libglnx_libs) libglnx.la
- +
- +test_libglnx_fdio_SOURCES = $(libglnx_srcpath)/tests/test-libglnx-fdio.c
- +test_libglnx_fdio_CFLAGS = $(AM_CFLAGS) $(libglnx_cflags)
- +test_libglnx_fdio_LDADD = $(libglnx_libs) libglnx.la
- +
- +test_libglnx_errors_SOURCES = $(libglnx_srcpath)/tests/test-libglnx-errors.c
- +test_libglnx_errors_CFLAGS = $(AM_CFLAGS) $(libglnx_cflags)
- +test_libglnx_errors_LDADD = $(libglnx_libs) libglnx.la
- +
- +test_libglnx_macros_SOURCES = $(libglnx_srcpath)/tests/test-libglnx-macros.c
- +test_libglnx_macros_CFLAGS = $(AM_CFLAGS) $(libglnx_cflags)
- +test_libglnx_macros_LDADD = $(libglnx_libs) libglnx.la
- +
- +test_libglnx_shutil_SOURCES = $(libglnx_srcpath)/tests/test-libglnx-shutil.c
- +test_libglnx_shutil_CFLAGS = $(AM_CFLAGS) $(libglnx_cflags)
- +test_libglnx_shutil_LDADD = $(libglnx_libs) libglnx.la
- diff -Nuar ostree-2018.8.orig/libglnx/README.md ostree-2018.8/libglnx/README.md
- --- ostree-2018.8.orig/libglnx/README.md 1970-01-01 02:00:00.000000000 +0200
- +++ ostree-2018.8/libglnx/README.md 2018-05-26 00:37:32.006018846 +0300
- @@ -0,0 +1,52 @@
- +libglnx is the successor to libgsystem: https://git.gnome.org/browse/libgsystem
- +
- +It is for modules which depend on both GLib and Linux, intended to be
- +used as a git submodule.
- +
- +Features:
- +
- + - File APIs which use `openat()` like APIs, but also take a `GCancellable`
- + to support dynamic cancellation
- + - APIs also have a `GError` parameter
- + - High level "shutil", somewhat inspired by Python's
- + - A "console" API for tty output
- + - A backport of the GLib cleanup macros for projects which can't yet take
- + a dependency on 2.40.
- +
- +Why?
- +----
- +
- +There are multiple projects which have a hard dependency on Linux and
- +GLib, such as NetworkManager, ostree, flatpak, etc. It makes sense
- +for them to be able to share Linux-specific APIs.
- +
- +This module also contains some code taken from systemd, which has very
- +high quality LGPLv2+ shared library code, but most of the internal
- +shared library is private, and not namespaced.
- +
- +One could also compare this project to gnulib; the salient differences
- +there are that at least some of this module is eventually destined for
- +inclusion in GLib.
- +
- +Porting from libgsystem
- +-----------------------
- +
- +For all of the filesystem access code, libglnx exposes only
- +fd-relative API, not `GFile*`. It does use `GCancellable` where
- +applicable.
- +
- +For local allocation macros, you should start using the `g_auto`
- +macros from GLib. A backport is included in libglnx. There are a few
- +APIs not defined in GLib yet, such as `glnx_autofd`.
- +
- +`gs_transfer_out_value` is replaced by `g_steal_pointer`.
- +
- +Contributing
- +------------
- +
- +Currently there is not a Bugzilla product - one may be created
- +in the future. You can submit PRs against the Github mirror:
- +
- +https://github.com/GNOME/libglnx/pulls
- +
- +Or alternatively, email one of the maintainers directly.
- diff -Nuar ostree-2018.8.orig/libglnx/tests/libglnx-testlib.h ostree-2018.8/libglnx/tests/libglnx-testlib.h
- --- ostree-2018.8.orig/libglnx/tests/libglnx-testlib.h 1970-01-01 02:00:00.000000000 +0200
- +++ ostree-2018.8/libglnx/tests/libglnx-testlib.h 2018-05-26 00:37:32.008018846 +0300
- @@ -0,0 +1,34 @@
- +/* -*- mode: C; c-file-style: "gnu"; indent-tabs-mode: nil; -*-
- + *
- + * Copyright (C) 2017 Red Hat, Inc.
- + *
- + * This library is free software; you can redistribute it and/or
- + * modify it under the terms of the GNU Lesser General Public
- + * License as published by the Free Software Foundation; either
- + * version 2 of the License, or (at your option) any later version.
- + *
- + * This library is distributed in the hope that it will be useful,
- + * but WITHOUT ANY WARRANTY; without even the implied warranty of
- + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
- + * Lesser General Public License for more details.
- + *
- + * You should have received a copy of the GNU Lesser General Public
- + * License along with this library; if not, write to the
- + * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
- + * Boston, MA 02111-1307, USA.
- + */
- +
- +#pragma once
- +
- +typedef GError _GLnxTestAutoError;
- +static inline void
- +_glnx_test_auto_error_cleanup (_GLnxTestAutoError *autoerror)
- +{
- + g_assert_no_error (autoerror);
- + /* We could add a clear call here, but no point...we'll have aborted */
- +}
- +G_DEFINE_AUTOPTR_CLEANUP_FUNC(_GLnxTestAutoError, _glnx_test_auto_error_cleanup);
- +
- +#define _GLNX_TEST_DECLARE_ERROR(local_error, error) \
- + g_autoptr(_GLnxTestAutoError) local_error = NULL; \
- + GError **error = &local_error
- diff -Nuar ostree-2018.8.orig/libglnx/tests/test-libglnx-errors.c ostree-2018.8/libglnx/tests/test-libglnx-errors.c
- --- ostree-2018.8.orig/libglnx/tests/test-libglnx-errors.c 1970-01-01 02:00:00.000000000 +0200
- +++ ostree-2018.8/libglnx/tests/test-libglnx-errors.c 2018-05-26 00:37:32.008018846 +0300
- @@ -0,0 +1,183 @@
- +/* -*- mode: C; c-file-style: "gnu"; indent-tabs-mode: nil; -*-
- + *
- + * Copyright (C) 2017 Red Hat, Inc.
- + *
- + * This library is free software; you can redistribute it and/or
- + * modify it under the terms of the GNU Lesser General Public
- + * License as published by the Free Software Foundation; either
- + * version 2 of the License, or (at your option) any later version.
- + *
- + * This library is distributed in the hope that it will be useful,
- + * but WITHOUT ANY WARRANTY; without even the implied warranty of
- + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
- + * Lesser General Public License for more details.
- + *
- + * You should have received a copy of the GNU Lesser General Public
- + * License along with this library; if not, write to the
- + * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
- + * Boston, MA 02111-1307, USA.
- + */
- +
- +#include "config.h"
- +#include "libglnx.h"
- +#include <glib.h>
- +#include <stdlib.h>
- +#include <gio/gio.h>
- +#include <string.h>
- +
- +static void
- +test_error_throw (void)
- +{
- + g_autoptr(GError) error = NULL;
- +
- + g_assert (!glnx_throw (&error, "foo: %s %d", "hello", 42));
- + g_assert_error (error, G_IO_ERROR, G_IO_ERROR_FAILED);
- + g_assert_cmpstr (error->message, ==, "foo: hello 42");
- + g_clear_error (&error);
- +
- + gpointer dummy = glnx_null_throw (&error, "literal foo");
- + g_assert (dummy == NULL);
- + g_assert_error (error, G_IO_ERROR, G_IO_ERROR_FAILED);
- + g_assert_cmpstr (error->message, ==, "literal foo");
- + g_clear_error (&error);
- +
- + gpointer dummy2 = glnx_null_throw (&error, "foo: %s %d", "hola", 24);
- + g_assert (dummy2 == NULL);
- + g_assert_error (error, G_IO_ERROR, G_IO_ERROR_FAILED);
- + g_assert_cmpstr (error->message, ==, "foo: hola 24");
- + g_clear_error (&error);
- +}
- +
- +static void
- +test_error_errno (void)
- +{
- + g_autoptr(GError) error = NULL;
- + const char noent_path[] = "/enoent-this-should-not-exist";
- + int fd;
- +
- + fd = open (noent_path, O_RDONLY);
- + if (fd < 0)
- + {
- + g_assert (!glnx_throw_errno (&error));
- + g_assert_error (error, G_IO_ERROR, G_IO_ERROR_NOT_FOUND);
- + g_assert (!glnx_prefix_error (&error, "myprefix"));
- + g_assert_error (error, G_IO_ERROR, G_IO_ERROR_NOT_FOUND);
- + g_assert (g_str_has_prefix (error->message, "myprefix: "));
- + g_clear_error (&error);
- + }
- + else
- + g_assert_cmpint (fd, ==, -1);
- +
- + fd = open (noent_path, O_RDONLY);
- + if (fd < 0)
- + {
- + gpointer dummy = glnx_null_throw_errno (&error);
- + g_assert (dummy == NULL);
- + g_assert_error (error, G_IO_ERROR, G_IO_ERROR_NOT_FOUND);
- + dummy = glnx_prefix_error_null (&error, "myprefix");
- + g_assert (dummy == NULL);
- + g_assert_error (error, G_IO_ERROR, G_IO_ERROR_NOT_FOUND);
- + g_assert (g_str_has_prefix (error->message, "myprefix: "));
- + g_clear_error (&error);
- + }
- + else
- + g_assert_cmpint (fd, ==, -1);
- +
- + fd = open (noent_path, O_RDONLY);
- + if (fd < 0)
- + {
- + g_autofree char *expected_prefix = g_strdup_printf ("Failed to open %s", noent_path);
- + g_assert (!glnx_throw_errno_prefix (&error, "Failed to open %s", noent_path));
- + g_assert_error (error, G_IO_ERROR, G_IO_ERROR_NOT_FOUND);
- + g_assert (g_str_has_prefix (error->message, expected_prefix));
- + g_clear_error (&error);
- + /* And test the legacy wrapper */
- + glnx_set_prefix_error_from_errno (&error, "Failed to open %s", noent_path);
- + g_assert_error (error, G_IO_ERROR, G_IO_ERROR_NOT_FOUND);
- + g_assert (g_str_has_prefix (error->message, expected_prefix));
- + g_clear_error (&error);
- + }
- + else
- + g_assert_cmpint (fd, ==, -1);
- +
- + fd = open (noent_path, O_RDONLY);
- + if (fd < 0)
- + {
- + gpointer dummy = glnx_null_throw_errno_prefix (&error, "Failed to open file");
- + g_assert (dummy == NULL);
- + g_assert_error (error, G_IO_ERROR, G_IO_ERROR_NOT_FOUND);
- + g_assert (g_str_has_prefix (error->message, "Failed to open file"));
- + g_clear_error (&error);
- + }
- + else
- + g_assert_cmpint (fd, ==, -1);
- +
- + fd = open (noent_path, O_RDONLY);
- + if (fd < 0)
- + {
- + gpointer dummy = glnx_null_throw_errno_prefix (&error, "Failed to open %s", noent_path);
- + g_assert (dummy == NULL);
- + g_assert_error (error, G_IO_ERROR, G_IO_ERROR_NOT_FOUND);
- + g_assert (g_str_has_prefix (error->message, glnx_strjoina ("Failed to open ", noent_path)));
- + g_clear_error (&error);
- + }
- + else
- + g_assert_cmpint (fd, ==, -1);
- +}
- +
- +static void
- +test_error_auto_nothrow (GError **error)
- +{
- + GLNX_AUTO_PREFIX_ERROR("foo", error);
- + /* Side effect to avoid otherwise empty function */
- + g_assert_no_error (*error);
- +}
- +
- +static void
- +test_error_auto_throw (GError **error)
- +{
- + GLNX_AUTO_PREFIX_ERROR("foo", error);
- + (void) glnx_throw (error, "oops");
- +}
- +
- +static void
- +test_error_auto_throw_recurse (GError **error)
- +{
- + GLNX_AUTO_PREFIX_ERROR("foo", error);
- +
- + if (TRUE)
- + {
- + GLNX_AUTO_PREFIX_ERROR("bar", error);
- + (void) glnx_throw (error, "oops");
- + }
- +}
- +
- +static void
- +test_error_auto (void)
- +{
- + g_autoptr(GError) error = NULL;
- + test_error_auto_nothrow (&error);
- + g_assert_no_error (error);
- + test_error_auto_throw (&error);
- + g_assert_nonnull (error);
- + g_assert_cmpstr (error->message, ==, "foo: oops");
- + g_clear_error (&error);
- + test_error_auto_throw_recurse (&error);
- + g_assert_nonnull (error);
- + g_assert_cmpstr (error->message, ==, "foo: bar: oops");
- +}
- +
- +int main (int argc, char **argv)
- +{
- + int ret;
- +
- + g_test_init (&argc, &argv, NULL);
- +
- + g_test_add_func ("/error-throw", test_error_throw);
- + g_test_add_func ("/error-errno", test_error_errno);
- + g_test_add_func ("/error-auto", test_error_auto);
- +
- + ret = g_test_run();
- +
- + return ret;
- +}
- diff -Nuar ostree-2018.8.orig/libglnx/tests/test-libglnx-fdio.c ostree-2018.8/libglnx/tests/test-libglnx-fdio.c
- --- ostree-2018.8.orig/libglnx/tests/test-libglnx-fdio.c 1970-01-01 02:00:00.000000000 +0200
- +++ ostree-2018.8/libglnx/tests/test-libglnx-fdio.c 2018-05-26 00:37:32.008018846 +0300
- @@ -0,0 +1,254 @@
- +/* -*- mode: C; c-file-style: "gnu"; indent-tabs-mode: nil; -*-
- + *
- + * Copyright (C) 2017 Red Hat, Inc.
- + *
- + * This library is free software; you can redistribute it and/or
- + * modify it under the terms of the GNU Lesser General Public
- + * License as published by the Free Software Foundation; either
- + * version 2 of the License, or (at your option) any later version.
- + *
- + * This library is distributed in the hope that it will be useful,
- + * but WITHOUT ANY WARRANTY; without even the implied warranty of
- + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
- + * Lesser General Public License for more details.
- + *
- + * You should have received a copy of the GNU Lesser General Public
- + * License along with this library; if not, write to the
- + * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
- + * Boston, MA 02111-1307, USA.
- + */
- +
- +#include "config.h"
- +#include "libglnx.h"
- +#include <glib.h>
- +#include <stdlib.h>
- +#include <gio/gio.h>
- +#include <err.h>
- +#include <string.h>
- +
- +#include "libglnx-testlib.h"
- +
- +static gboolean
- +renameat_test_setup (int *out_srcfd, int *out_destfd,
- + GError **error)
- +{
- + glnx_autofd int srcfd = -1;
- + glnx_autofd int destfd = -1;
- +
- + (void) glnx_shutil_rm_rf_at (AT_FDCWD, "srcdir", NULL, NULL);
- + if (mkdir ("srcdir", 0755) < 0)
- + err (1, "mkdir");
- + if (!glnx_opendirat (AT_FDCWD, "srcdir", TRUE, &srcfd, error))
- + return FALSE;
- + (void) glnx_shutil_rm_rf_at (AT_FDCWD, "destdir", NULL, NULL);
- + if (mkdir ("destdir", 0755) < 0)
- + err (1, "mkdir");
- + if (!glnx_opendirat (AT_FDCWD, "destdir", TRUE, &destfd, error))
- + return FALSE;
- +
- + if (!glnx_file_replace_contents_at (srcfd, "foo", (guint8*)"foo contents", strlen ("foo contents"),
- + GLNX_FILE_REPLACE_NODATASYNC, NULL, error))
- + return FALSE;
- + if (!glnx_file_replace_contents_at (destfd, "bar", (guint8*)"bar contents", strlen ("bar contents"),
- + GLNX_FILE_REPLACE_NODATASYNC, NULL, error))
- + return FALSE;
- +
- + *out_srcfd = srcfd; srcfd = -1;
- + *out_destfd = destfd; destfd = -1;
- + return TRUE;
- +}
- +
- +static void
- +test_renameat2_noreplace (void)
- +{
- + _GLNX_TEST_DECLARE_ERROR(local_error, error);
- + glnx_autofd int srcfd = -1;
- + glnx_autofd int destfd = -1;
- + struct stat stbuf;
- +
- + if (!renameat_test_setup (&srcfd, &destfd, error))
- + return;
- +
- + if (glnx_renameat2_noreplace (srcfd, "foo", destfd, "bar") == 0)
- + g_assert_not_reached ();
- + else
- + {
- + g_assert_cmpint (errno, ==, EEXIST);
- + }
- +
- + if (glnx_renameat2_noreplace (srcfd, "foo", destfd, "baz") < 0)
- + return (void)glnx_throw_errno_prefix (error, "renameat");
- + if (!glnx_fstatat (destfd, "bar", &stbuf, AT_SYMLINK_NOFOLLOW, error))
- + return;
- +
- + if (fstatat (srcfd, "foo", &stbuf, AT_SYMLINK_NOFOLLOW) == 0)
- + g_assert_not_reached ();
- + else
- + g_assert_cmpint (errno, ==, ENOENT);
- +}
- +
- +static void
- +test_renameat2_exchange (void)
- +{
- + _GLNX_TEST_DECLARE_ERROR(local_error, error);
- +
- + glnx_autofd int srcfd = -1;
- + glnx_autofd int destfd = -1;
- + if (!renameat_test_setup (&srcfd, &destfd, error))
- + return;
- +
- + if (glnx_renameat2_exchange (AT_FDCWD, "srcdir", AT_FDCWD, "destdir") < 0)
- + return (void)glnx_throw_errno_prefix (error, "renameat");
- +
- + /* Ensure the dir fds are the same */
- + struct stat stbuf;
- + if (!glnx_fstatat (srcfd, "foo", &stbuf, AT_SYMLINK_NOFOLLOW, error))
- + return;
- + if (!glnx_fstatat (destfd, "bar", &stbuf, AT_SYMLINK_NOFOLLOW, error))
- + return;
- + /* But the dirs should be swapped */
- + if (!glnx_fstatat (AT_FDCWD, "destdir/foo", &stbuf, AT_SYMLINK_NOFOLLOW, error))
- + return;
- + if (!glnx_fstatat (AT_FDCWD, "srcdir/bar", &stbuf, AT_SYMLINK_NOFOLLOW, error))
- + return;
- +}
- +
- +static void
- +test_tmpfile (void)
- +{
- + _GLNX_TEST_DECLARE_ERROR(local_error, error);
- +
- + g_auto(GLnxTmpfile) tmpf = { 0, };
- + if (!glnx_open_tmpfile_linkable_at (AT_FDCWD, ".", O_WRONLY|O_CLOEXEC, &tmpf, error))
- + return;
- + if (glnx_loop_write (tmpf.fd, "foo", strlen ("foo")) < 0)
- + return (void)glnx_throw_errno_prefix (error, "write");
- + if (glnx_link_tmpfile_at (&tmpf, GLNX_LINK_TMPFILE_NOREPLACE, AT_FDCWD, "foo", error))
- + return;
- +}
- +
- +static void
- +test_stdio_file (void)
- +{
- + _GLNX_TEST_DECLARE_ERROR(local_error, error);
- + g_auto(GLnxTmpfile) tmpf = { 0, };
- + g_autoptr(FILE) f = NULL;
- +
- + if (!glnx_open_anonymous_tmpfile (O_RDWR|O_CLOEXEC, &tmpf, error))
- + return;
- + f = fdopen (tmpf.fd, "w");
- + tmpf.fd = -1; /* Ownership was transferred via fdopen() */
- + if (!f)
- + return (void)glnx_throw_errno_prefix (error, "fdopen");
- + if (fwrite ("hello", 1, strlen ("hello"), f) != strlen ("hello"))
- + return (void)glnx_throw_errno_prefix (error, "fwrite");
- + if (!glnx_stdio_file_flush (f, error))
- + return;
- +}
- +
- +static void
- +test_fstatat (void)
- +{
- + _GLNX_TEST_DECLARE_ERROR(local_error, error);
- + struct stat stbuf = { 0, };
- +
- + if (!glnx_fstatat_allow_noent (AT_FDCWD, ".", &stbuf, 0, error))
- + return;
- + g_assert_cmpint (errno, ==, 0);
- + g_assert_no_error (local_error);
- + g_assert (S_ISDIR (stbuf.st_mode));
- + if (!glnx_fstatat_allow_noent (AT_FDCWD, "nosuchfile", &stbuf, 0, error))
- + return;
- + g_assert_cmpint (errno, ==, ENOENT);
- + g_assert_no_error (local_error);
- +
- + /* test NULL parameter for stat */
- + if (!glnx_fstatat_allow_noent (AT_FDCWD, ".", NULL, 0, error))
- + return;
- + g_assert_cmpint (errno, ==, 0);
- + g_assert_no_error (local_error);
- + if (!glnx_fstatat_allow_noent (AT_FDCWD, "nosuchfile", NULL, 0, error))
- + return;
- + g_assert_cmpint (errno, ==, ENOENT);
- + g_assert_no_error (local_error);
- +}
- +
- +static void
- +test_filecopy (void)
- +{
- + _GLNX_TEST_DECLARE_ERROR(local_error, error);
- + const char foo[] = "foo";
- + struct stat stbuf;
- +
- + if (!glnx_ensure_dir (AT_FDCWD, "subdir", 0755, error))
- + return;
- +
- + if (!glnx_file_replace_contents_at (AT_FDCWD, foo, (guint8*)foo, sizeof (foo),
- + GLNX_FILE_REPLACE_NODATASYNC, NULL, error))
- + return;
- +
- + /* Copy it into both the same dir and a subdir */
- + if (!glnx_file_copy_at (AT_FDCWD, foo, NULL, AT_FDCWD, "bar",
- + GLNX_FILE_COPY_NOXATTRS, NULL, error))
- + return;
- + if (!glnx_file_copy_at (AT_FDCWD, foo, NULL, AT_FDCWD, "subdir/bar",
- + GLNX_FILE_COPY_NOXATTRS, NULL, error))
- + return;
- + if (!glnx_fstatat (AT_FDCWD, "subdir/bar", &stbuf, 0, error))
- + return;
- +
- + if (glnx_file_copy_at (AT_FDCWD, foo, NULL, AT_FDCWD, "bar",
- + GLNX_FILE_COPY_NOXATTRS, NULL, error))
- + g_assert_not_reached ();
- + g_assert_error (local_error, G_IO_ERROR, G_IO_ERROR_EXISTS);
- + g_clear_error (&local_error);
- +
- + if (!glnx_file_copy_at (AT_FDCWD, foo, NULL, AT_FDCWD, "bar",
- + GLNX_FILE_COPY_NOXATTRS | GLNX_FILE_COPY_OVERWRITE,
- + NULL, error))
- + return;
- +
- + if (symlinkat ("nosuchtarget", AT_FDCWD, "link") < 0)
- + return (void) glnx_throw_errno_prefix (error, "symlinkat");
- +
- + /* Shouldn't be able to overwrite a symlink without GLNX_FILE_COPY_OVERWRITE */
- + if (glnx_file_copy_at (AT_FDCWD, foo, NULL, AT_FDCWD, "link",
- + GLNX_FILE_COPY_NOXATTRS,
- + NULL, error))
- + g_assert_not_reached ();
- + g_assert_error (local_error, G_IO_ERROR, G_IO_ERROR_EXISTS);
- + g_clear_error (&local_error);
- +
- + /* Test overwriting symlink */
- + if (!glnx_file_copy_at (AT_FDCWD, foo, NULL, AT_FDCWD, "link",
- + GLNX_FILE_COPY_NOXATTRS | GLNX_FILE_COPY_OVERWRITE,
- + NULL, error))
- + return;
- +
- + if (!glnx_fstatat_allow_noent (AT_FDCWD, "nosuchtarget", &stbuf, AT_SYMLINK_NOFOLLOW, error))
- + return;
- + g_assert_cmpint (errno, ==, ENOENT);
- + g_assert_no_error (local_error);
- +
- + if (!glnx_fstatat (AT_FDCWD, "link", &stbuf, AT_SYMLINK_NOFOLLOW, error))
- + return;
- + g_assert (S_ISREG (stbuf.st_mode));
- +}
- +
- +int main (int argc, char **argv)
- +{
- + int ret;
- +
- + g_test_init (&argc, &argv, NULL);
- +
- + g_test_add_func ("/tmpfile", test_tmpfile);
- + g_test_add_func ("/stdio-file", test_stdio_file);
- + g_test_add_func ("/filecopy", test_filecopy);
- + g_test_add_func ("/renameat2-noreplace", test_renameat2_noreplace);
- + g_test_add_func ("/renameat2-exchange", test_renameat2_exchange);
- + g_test_add_func ("/fstat", test_fstatat);
- +
- + ret = g_test_run();
- +
- + return ret;
- +}
- diff -Nuar ostree-2018.8.orig/libglnx/tests/test-libglnx-macros.c ostree-2018.8/libglnx/tests/test-libglnx-macros.c
- --- ostree-2018.8.orig/libglnx/tests/test-libglnx-macros.c 1970-01-01 02:00:00.000000000 +0200
- +++ ostree-2018.8/libglnx/tests/test-libglnx-macros.c 2018-05-26 00:37:32.008018846 +0300
- @@ -0,0 +1,109 @@
- +/* -*- mode: C; c-file-style: "gnu"; indent-tabs-mode: nil; -*-
- + *
- + * Copyright (C) 2017 Red Hat, Inc.
- + *
- + * This library is free software; you can redistribute it and/or
- + * modify it under the terms of the GNU Lesser General Public
- + * License as published by the Free Software Foundation; either
- + * version 2 of the License, or (at your option) any later version.
- + *
- + * This library is distributed in the hope that it will be useful,
- + * but WITHOUT ANY WARRANTY; without even the implied warranty of
- + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
- + * Lesser General Public License for more details.
- + *
- + * You should have received a copy of the GNU Lesser General Public
- + * License along with this library; if not, write to the
- + * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
- + * Boston, MA 02111-1307, USA.
- + */
- +
- +#include "config.h"
- +#include "libglnx.h"
- +#include <glib.h>
- +#include <stdlib.h>
- +#include <gio/gio.h>
- +#include <string.h>
- +
- +static void
- +test_inset (void)
- +{
- + g_assert (G_IN_SET (7, 7));
- + g_assert (G_IN_SET (7, 42, 7));
- + g_assert (G_IN_SET (7, 7,42,3,9));
- + g_assert (G_IN_SET (42, 7,42,3,9));
- + g_assert (G_IN_SET (3, 7,42,3,9));
- + g_assert (G_IN_SET (9, 7,42,3,9));
- + g_assert (!G_IN_SET (8, 7,42,3,9));
- + g_assert (!G_IN_SET (-1, 7,42,3,9));
- + g_assert (G_IN_SET ('x', 'a', 'x', 'c'));
- + g_assert (!G_IN_SET ('y', 'a', 'x', 'c'));
- +}
- +
- +static void
- +test_hash_table_foreach (void)
- +{
- + /* use var names all different from the macro metavars to ensure proper
- + * substitution */
- + g_autoptr(GHashTable) table = g_hash_table_new (g_str_hash, g_str_equal);
- + const char *keys[] = {"key1", "key2"};
- + const char *vals[] = {"val1", "val2"};
- + g_hash_table_insert (table, (gpointer)keys[0], (gpointer)vals[0]);
- + g_hash_table_insert (table, (gpointer)keys[1], (gpointer)vals[1]);
- +
- + guint i = 0;
- + GLNX_HASH_TABLE_FOREACH_IT (table, it, const char*, key, const char*, val)
- + {
- + g_assert_cmpstr (key, ==, keys[i]);
- + g_assert_cmpstr (val, ==, vals[i]);
- + i++;
- + }
- + g_assert_cmpuint (i, ==, 2);
- +
- + i = 0;
- + GLNX_HASH_TABLE_FOREACH_IT (table, it, const char*, key, const char*, val)
- + {
- + g_hash_table_iter_remove (&it);
- + break;
- + }
- + g_assert_cmpuint (g_hash_table_size (table), ==, 1);
- +
- + g_hash_table_insert (table, (gpointer)keys[1], (gpointer)vals[1]);
- + g_assert_cmpuint (g_hash_table_size (table), ==, 1);
- +
- + g_hash_table_insert (table, (gpointer)keys[0], (gpointer)vals[0]);
- + g_assert_cmpuint (g_hash_table_size (table), ==, 2);
- +
- + i = 0;
- + GLNX_HASH_TABLE_FOREACH_KV (table, const char*, key, const char*, val)
- + {
- + g_assert_cmpstr (key, ==, keys[i]);
- + g_assert_cmpstr (val, ==, vals[i]);
- + i++;
- + }
- + g_assert_cmpuint (i, ==, 2);
- +
- + i = 0;
- + GLNX_HASH_TABLE_FOREACH (table, const char*, key)
- + {
- + g_assert_cmpstr (key, ==, keys[i]);
- + i++;
- + }
- + g_assert_cmpuint (i, ==, 2);
- +
- + i = 0;
- + GLNX_HASH_TABLE_FOREACH_V (table, const char*, val)
- + {
- + g_assert_cmpstr (val, ==, vals[i]);
- + i++;
- + }
- + g_assert_cmpuint (i, ==, 2);
- +}
- +
- +int main (int argc, char **argv)
- +{
- + g_test_init (&argc, &argv, NULL);
- + g_test_add_func ("/inset", test_inset);
- + g_test_add_func ("/hash_table_foreach", test_hash_table_foreach);
- + return g_test_run();
- +}
- diff -Nuar ostree-2018.8.orig/libglnx/tests/test-libglnx-shutil.c ostree-2018.8/libglnx/tests/test-libglnx-shutil.c
- --- ostree-2018.8.orig/libglnx/tests/test-libglnx-shutil.c 1970-01-01 02:00:00.000000000 +0200
- +++ ostree-2018.8/libglnx/tests/test-libglnx-shutil.c 2018-05-26 00:37:32.008018846 +0300
- @@ -0,0 +1,63 @@
- +/* -*- mode: C; c-file-style: "gnu"; indent-tabs-mode: nil; -*-
- + *
- + * Copyright © 2017 Endless Mobile, Inc.
- + *
- + * This library is free software; you can redistribute it and/or
- + * modify it under the terms of the GNU Lesser General Public
- + * License as published by the Free Software Foundation; either
- + * version 2 of the License, or (at your option) any later version.
- + *
- + * This library is distributed in the hope that it will be useful,
- + * but WITHOUT ANY WARRANTY; without even the implied warranty of
- + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
- + * Lesser General Public License for more details.
- + *
- + * You should have received a copy of the GNU Lesser General Public
- + * License along with this library; if not, write to the
- + * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
- + * Boston, MA 02111-1307, USA.
- + */
- +
- +#include "config.h"
- +#include "libglnx.h"
- +#include <glib.h>
- +#include <stdlib.h>
- +#include <gio/gio.h>
- +#include <err.h>
- +#include <string.h>
- +
- +#include "libglnx-testlib.h"
- +
- +static void
- +test_mkdir_p_enoent (void)
- +{
- + _GLNX_TEST_DECLARE_ERROR(local_error, error);
- + glnx_autofd int dfd = -1;
- +
- + if (!glnx_ensure_dir (AT_FDCWD, "test", 0755, error))
- + return;
- + if (!glnx_opendirat (AT_FDCWD, "test", FALSE, &dfd, error))
- + return;
- + if (rmdir ("test") < 0)
- + return (void) glnx_throw_errno_prefix (error, "rmdir(%s)", "test");
- +
- + /* This should fail with ENOENT. */
- + glnx_shutil_mkdir_p_at (dfd, "blah/baz", 0755, NULL, error);
- + g_assert_error (local_error, G_IO_ERROR, G_IO_ERROR_NOT_FOUND);
- + g_clear_error (&local_error);
- +}
- +
- +int
- +main (int argc,
- + char **argv)
- +{
- + int ret;
- +
- + g_test_init (&argc, &argv, NULL);
- +
- + g_test_add_func ("/mkdir-p/enoent", test_mkdir_p_enoent);
- +
- + ret = g_test_run();
- +
- + return ret;
- +}
- diff -Nuar ostree-2018.8.orig/libglnx/tests/test-libglnx-xattrs.c ostree-2018.8/libglnx/tests/test-libglnx-xattrs.c
- --- ostree-2018.8.orig/libglnx/tests/test-libglnx-xattrs.c 1970-01-01 02:00:00.000000000 +0200
- +++ ostree-2018.8/libglnx/tests/test-libglnx-xattrs.c 2018-05-26 00:37:32.008018846 +0300
- @@ -0,0 +1,283 @@
- +/* -*- mode: C; c-file-style: "gnu"; indent-tabs-mode: nil; -*-
- + *
- + * Copyright (C) 2017 Red Hat, Inc.
- + *
- + * This library is free software; you can redistribute it and/or
- + * modify it under the terms of the GNU Lesser General Public
- + * License as published by the Free Software Foundation; either
- + * version 2 of the License, or (at your option) any later version.
- + *
- + * This library is distributed in the hope that it will be useful,
- + * but WITHOUT ANY WARRANTY; without even the implied warranty of
- + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
- + * Lesser General Public License for more details.
- + *
- + * You should have received a copy of the GNU Lesser General Public
- + * License along with this library; if not, write to the
- + * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
- + * Boston, MA 02111-1307, USA.
- + */
- +
- +#include "config.h"
- +#include "libglnx.h"
- +#include <glib.h>
- +#include <stdlib.h>
- +#include <gio/gio.h>
- +#include <string.h>
- +
- +#define XATTR_THREAD_RUN_TIME_USECS (5 * G_USEC_PER_SEC)
- +
- +struct XattrWorker {
- + int dfd;
- + gboolean is_writer;
- + guint n_attrs_read;
- +};
- +
- +typedef enum {
- + WRITE_RUN_MUTATE,
- + WRITE_RUN_CREATE,
- +} WriteType;
- +
- +static gboolean
- +set_random_xattr_value (int fd, const char *name, GError **error)
- +{
- + const guint8 randxattrbyte = g_random_int_range (0, 256);
- + const guint32 randxattrvalue_len = (g_random_int () % 256) + 1; /* Picked to be not too small or large */
- + g_autofree char *randxattrvalue = g_malloc (randxattrvalue_len);
- +
- + memset (randxattrvalue, randxattrbyte, randxattrvalue_len);
- +
- + if (fsetxattr (fd, name, randxattrvalue, randxattrvalue_len, 0) < 0)
- + {
- + glnx_set_error_from_errno (error);
- + return FALSE;
- + }
- +
- + return TRUE;
- +}
- +
- +static gboolean
- +add_random_xattrs (int fd, GError **error)
- +{
- + const guint nattrs = MIN (2, g_random_int () % 16);
- +
- + for (guint i = 0; i < nattrs; i++)
- + {
- + guint32 randxattrname_v = g_random_int ();
- + g_autofree char *randxattrname = g_strdup_printf ("user.test%u", randxattrname_v);
- +
- + if (!set_random_xattr_value (fd, randxattrname, error))
- + return FALSE;
- + }
- +
- + return TRUE;
- +}
- +
- +static gboolean
- +do_write_run (GLnxDirFdIterator *dfd_iter, GError **error)
- +{
- + WriteType wtype = g_random_int () % 2;
- +
- + if (wtype == WRITE_RUN_CREATE)
- + {
- + guint32 randname_v = g_random_int ();
- + g_autofree char *randname = g_strdup_printf ("file%u", randname_v);
- + glnx_autofd int fd = -1;
- +
- + again:
- + fd = openat (dfd_iter->fd, randname, O_CREAT | O_EXCL, 0644);
- + if (fd < 0)
- + {
- + if (errno == EEXIST)
- + {
- + g_printerr ("Congratulations! I suggest purchasing a lottery ticket today!\n");
- + goto again;
- + }
- + else
- + {
- + glnx_set_error_from_errno (error);
- + return FALSE;
- + }
- + }
- +
- + if (!add_random_xattrs (fd, error))
- + return FALSE;
- + }
- + else if (wtype == WRITE_RUN_MUTATE)
- + {
- + while (TRUE)
- + {
- + struct dirent *dent;
- + if (!glnx_dirfd_iterator_next_dent (dfd_iter, &dent, NULL, error))
- + return FALSE;
- + if (!dent)
- + break;
- +
- + glnx_autofd int fd = -1;
- + if (!glnx_openat_rdonly (dfd_iter->fd, dent->d_name, FALSE, &fd, error))
- + return FALSE;
- +
- + g_autoptr(GVariant) current_xattrs = NULL;
- + if (!glnx_fd_get_all_xattrs (fd, ¤t_xattrs, NULL, error))
- + return FALSE;
- +
- + for (int i = 0; i < g_variant_n_children (current_xattrs); i++)
- + {
- + const char *name, *value;
- + g_variant_get_child (current_xattrs, i, "(^&ay^&ay)", &name, &value);
- +
- + /* We don't want to potentially test/change xattrs like security.selinux
- + * that were injected by the system.
- + */
- + if (!g_str_has_prefix (name, "user.test"))
- + continue;
- +
- + if (!set_random_xattr_value (fd, name, error))
- + return FALSE;
- + }
- + }
- + }
- + else
- + g_assert_not_reached ();
- +
- + return TRUE;
- +}
- +
- +static gboolean
- +do_read_run (GLnxDirFdIterator *dfd_iter,
- + guint *out_n_read,
- + GError **error)
- +{
- + guint nattrs = 0;
- + while (TRUE)
- + {
- + struct dirent *dent;
- + if (!glnx_dirfd_iterator_next_dent (dfd_iter, &dent, NULL, error))
- + return FALSE;
- + if (!dent)
- + break;
- +
- + glnx_autofd int fd = -1;
- + if (!glnx_openat_rdonly (dfd_iter->fd, dent->d_name, FALSE, &fd, error))
- + return FALSE;
- +
- + g_autoptr(GVariant) current_xattrs = NULL;
- + if (!glnx_fd_get_all_xattrs (fd, ¤t_xattrs, NULL, error))
- + return FALSE;
- +
- + /* We don't actually care about the values, just use the variable
- + * to avoid compiler warnings.
- + */
- + nattrs += g_variant_n_children (current_xattrs);
- + }
- +
- + *out_n_read = nattrs;
- + return TRUE;
- +}
- +
- +static gpointer
- +xattr_thread (gpointer data)
- +{
- + g_autoptr(GError) local_error = NULL;
- + GError **error = &local_error;
- + struct XattrWorker *worker = data;
- + guint64 end_time = g_get_monotonic_time () + XATTR_THREAD_RUN_TIME_USECS;
- + guint n_read = 0;
- +
- + while (g_get_monotonic_time () < end_time)
- + {
- + g_auto(GLnxDirFdIterator) dfd_iter = { 0, };
- +
- + if (!glnx_dirfd_iterator_init_at (worker->dfd, ".", TRUE, &dfd_iter, error))
- + goto out;
- +
- + if (worker->is_writer)
- + {
- + if (!do_write_run (&dfd_iter, error))
- + goto out;
- + }
- + else
- + {
- + if (!do_read_run (&dfd_iter, &n_read, error))
- + goto out;
- + }
- + }
- +
- + out:
- + g_assert_no_error (local_error);
- +
- + return GINT_TO_POINTER (n_read);
- +}
- +
- +static void
- +test_xattr_races (void)
- +{
- + /* If for some reason we're built in a VM which only has one vcpu, let's still
- + * at least make the test do something.
- + */
- + /* FIXME - this deadlocks for me on 4.9.4-201.fc25.x86_64, whether
- + * using overlayfs or xfs as source/dest.
- + */
- + const guint nprocs = MAX (4, g_get_num_processors ());
- + struct XattrWorker wdata[nprocs];
- + GThread *threads[nprocs];
- + g_autoptr(GError) local_error = NULL;
- + GError **error = &local_error;
- + g_auto(GLnxTmpDir) tmpdir = { 0, };
- + g_autofree char *tmpdir_path = g_strdup_printf ("%s/libglnx-xattrs-XXXXXX",
- + getenv ("TMPDIR") ?: "/var/tmp");
- + guint nread = 0;
- +
- + if (!glnx_mkdtempat (AT_FDCWD, tmpdir_path, 0700,
- + &tmpdir, error))
- + goto out;
- +
- + /* Support people building/testing on tmpfs https://github.com/flatpak/flatpak/issues/686 */
- + if (fsetxattr (tmpdir.fd, "user.test", "novalue", strlen ("novalue"), 0) < 0)
- + {
- + if (errno == EOPNOTSUPP)
- + {
- + g_test_skip ("no xattr support");
- + return;
- + }
- + else
- + {
- + glnx_set_error_from_errno (error);
- + goto out;
- + }
- + }
- +
- + for (guint i = 0; i < nprocs; i++)
- + {
- + struct XattrWorker *worker = &wdata[i];
- + worker->dfd = tmpdir.fd;
- + worker->is_writer = i % 2 == 0;
- + threads[i] = g_thread_new (NULL, xattr_thread, worker);
- + }
- +
- + for (guint i = 0; i < nprocs; i++)
- + {
- + if (wdata[i].is_writer)
- + (void) g_thread_join (threads[i]);
- + else
- + nread += GPOINTER_TO_UINT (g_thread_join (threads[i]));
- + }
- +
- + g_print ("Read %u xattrs race free!\n", nread);
- +
- + out:
- + g_assert_no_error (local_error);
- +}
- +
- +int main (int argc, char **argv)
- +{
- + int ret;
- +
- + g_test_init (&argc, &argv, NULL);
- +
- + g_test_add_func ("/xattr-races", test_xattr_races);
- +
- + ret = g_test_run();
- +
- + return ret;
- +}
|