123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085208620872088208920902091209220932094209520962097209820992100210121022103210421052106210721082109211021112112211321142115211621172118211921202121212221232124212521262127212821292130213121322133213421352136213721382139214021412142214321442145214621472148214921502151215221532154215521562157215821592160216121622163216421652166216721682169217021712172217321742175217621772178217921802181218221832184218521862187218821892190219121922193219421952196219721982199220022012202220322042205220622072208220922102211221222132214221522162217221822192220222122222223222422252226222722282229223022312232223322342235223622372238223922402241224222432244224522462247224822492250225122522253225422552256225722582259226022612262226322642265226622672268226922702271227222732274227522762277227822792280228122822283228422852286228722882289229022912292229322942295229622972298229923002301230223032304230523062307230823092310231123122313231423152316231723182319232023212322232323242325232623272328232923302331233223332334233523362337233823392340234123422343234423452346234723482349235023512352235323542355235623572358235923602361236223632364236523662367236823692370237123722373237423752376237723782379238023812382238323842385238623872388238923902391239223932394239523962397239823992400240124022403240424052406240724082409241024112412241324142415241624172418241924202421242224232424242524262427242824292430243124322433243424352436243724382439244024412442244324442445244624472448244924502451245224532454245524562457245824592460246124622463246424652466246724682469247024712472247324742475247624772478247924802481248224832484248524862487248824892490249124922493249424952496249724982499250025012502250325042505250625072508250925102511251225132514251525162517251825192520252125222523252425252526252725282529253025312532253325342535253625372538253925402541254225432544254525462547254825492550255125522553255425552556255725582559256025612562256325642565256625672568256925702571257225732574257525762577257825792580258125822583258425852586258725882589259025912592259325942595259625972598259926002601260226032604260526062607260826092610261126122613261426152616261726182619262026212622262326242625262626272628262926302631263226332634263526362637263826392640264126422643264426452646264726482649265026512652265326542655265626572658265926602661266226632664266526662667266826692670267126722673267426752676267726782679268026812682268326842685268626872688268926902691269226932694269526962697269826992700270127022703270427052706270727082709271027112712271327142715271627172718271927202721272227232724272527262727272827292730273127322733273427352736273727382739274027412742274327442745274627472748274927502751275227532754275527562757275827592760276127622763276427652766276727682769277027712772277327742775277627772778277927802781278227832784278527862787278827892790279127922793279427952796279727982799280028012802280328042805280628072808280928102811281228132814281528162817281828192820282128222823282428252826282728282829283028312832283328342835283628372838283928402841284228432844284528462847284828492850285128522853285428552856285728582859286028612862286328642865286628672868286928702871287228732874287528762877287828792880288128822883288428852886288728882889289028912892289328942895289628972898289929002901290229032904290529062907290829092910291129122913291429152916291729182919292029212922292329242925292629272928292929302931293229332934293529362937293829392940294129422943294429452946294729482949295029512952295329542955295629572958295929602961296229632964296529662967296829692970297129722973297429752976297729782979298029812982298329842985298629872988298929902991299229932994299529962997299829993000300130023003300430053006300730083009301030113012301330143015301630173018301930203021302230233024302530263027302830293030303130323033303430353036303730383039304030413042304330443045304630473048304930503051305230533054305530563057305830593060306130623063306430653066306730683069307030713072307330743075307630773078307930803081308230833084308530863087308830893090309130923093309430953096309730983099310031013102310331043105310631073108310931103111311231133114311531163117311831193120312131223123312431253126312731283129313031313132313331343135313631373138313931403141314231433144314531463147314831493150315131523153315431553156315731583159316031613162316331643165316631673168316931703171317231733174317531763177317831793180318131823183318431853186318731883189319031913192319331943195319631973198319932003201320232033204320532063207320832093210321132123213321432153216321732183219322032213222322332243225322632273228322932303231323232333234323532363237323832393240324132423243324432453246324732483249325032513252325332543255325632573258325932603261326232633264326532663267326832693270327132723273327432753276327732783279328032813282328332843285328632873288328932903291329232933294329532963297329832993300330133023303330433053306330733083309331033113312331333143315331633173318331933203321332233233324332533263327332833293330333133323333333433353336333733383339334033413342334333443345334633473348334933503351335233533354335533563357335833593360336133623363336433653366336733683369337033713372337333743375337633773378337933803381338233833384338533863387338833893390339133923393339433953396339733983399340034013402340334043405340634073408340934103411341234133414341534163417341834193420342134223423342434253426342734283429343034313432343334343435343634373438343934403441344234433444344534463447344834493450345134523453345434553456345734583459346034613462346334643465346634673468346934703471347234733474347534763477347834793480348134823483348434853486348734883489349034913492349334943495349634973498349935003501350235033504350535063507350835093510351135123513351435153516351735183519352035213522352335243525352635273528352935303531353235333534353535363537353835393540354135423543354435453546354735483549355035513552355335543555355635573558355935603561356235633564356535663567356835693570357135723573357435753576357735783579358035813582358335843585358635873588358935903591359235933594359535963597359835993600360136023603360436053606360736083609361036113612361336143615361636173618361936203621362236233624362536263627362836293630363136323633363436353636363736383639364036413642364336443645364636473648364936503651365236533654365536563657365836593660366136623663366436653666366736683669367036713672367336743675367636773678367936803681368236833684368536863687368836893690369136923693369436953696369736983699370037013702370337043705370637073708370937103711371237133714371537163717371837193720372137223723372437253726372737283729373037313732373337343735373637373738373937403741374237433744374537463747374837493750375137523753375437553756375737583759376037613762376337643765376637673768376937703771377237733774377537763777377837793780378137823783378437853786378737883789379037913792379337943795379637973798379938003801380238033804380538063807380838093810381138123813381438153816381738183819382038213822382338243825382638273828382938303831383238333834383538363837383838393840384138423843384438453846384738483849385038513852385338543855385638573858385938603861386238633864386538663867386838693870387138723873387438753876387738783879388038813882388338843885388638873888388938903891389238933894389538963897389838993900390139023903390439053906390739083909391039113912391339143915391639173918391939203921392239233924392539263927392839293930393139323933393439353936393739383939394039413942394339443945394639473948394939503951395239533954395539563957395839593960396139623963396439653966396739683969397039713972397339743975397639773978397939803981398239833984398539863987398839893990399139923993399439953996399739983999400040014002400340044005400640074008400940104011401240134014401540164017401840194020402140224023402440254026402740284029403040314032403340344035403640374038403940404041404240434044404540464047404840494050405140524053405440554056405740584059406040614062406340644065406640674068406940704071407240734074407540764077407840794080408140824083408440854086408740884089409040914092409340944095409640974098409941004101410241034104410541064107410841094110411141124113411441154116411741184119412041214122412341244125412641274128412941304131413241334134413541364137413841394140414141424143414441454146414741484149415041514152415341544155415641574158415941604161416241634164416541664167416841694170417141724173417441754176417741784179418041814182418341844185418641874188418941904191419241934194 |
- /*
- zipint.h -- internal declarations.
- Copyright (C) 1999-2008 Dieter Baron and Thomas Klausner
- This file is part of libzip, a library to manipulate ZIP archives.
- The authors can be contacted at <libzip@nih.at>
- Redistribution and use in source and binary forms, with or without
- modification, are permitted provided 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.
- 3. The names of the authors may not be used to endorse or promote
- products derived from this software without specific prior
- written permission.
- THIS SOFTWARE IS PROVIDED BY THE AUTHORS ``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 AUTHORS 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 <zlib.h>
- /*
- #ifdef _MSC_VER
- #define ZIP_EXTERN __declspec(dllimport)
- #endif
- */
- /*
- zip.h -- exported declarations.
- Copyright (C) 1999-2008 Dieter Baron and Thomas Klausner
- This file is part of libzip, a library to manipulate ZIP archives.
- The authors can be contacted at <libzip@nih.at>
- Redistribution and use in source and binary forms, with or without
- modification, are permitted provided 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.
- 3. The names of the authors may not be used to endorse or promote
- products derived from this software without specific prior
- written permission.
- THIS SOFTWARE IS PROVIDED BY THE AUTHORS ``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 AUTHORS 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 ZIP_EXTERN
- #define ZIP_EXTERN
- #endif
- #ifdef __cplusplus
- extern "C" {
- #endif
- #include <sys/types.h>
- #include <stdio.h>
- #include <time.h>
- /* flags for zip_open */
- #define ZIP_CREATE 1
- #define ZIP_EXCL 2
- #define ZIP_CHECKCONS 4
- /* flags for zip_name_locate, zip_fopen, zip_stat, ... */
- #define ZIP_FL_NOCASE 1 /* ignore case on name lookup */
- #define ZIP_FL_NODIR 2 /* ignore directory component */
- #define ZIP_FL_COMPRESSED 4 /* read compressed data */
- #define ZIP_FL_UNCHANGED 8 /* use original data, ignoring changes */
- #define ZIP_FL_RECOMPRESS 16 /* force recompression of data */
- /* archive global flags flags */
- #define ZIP_AFL_TORRENT 1 /* torrent zipped */
- /* libzip error codes */
- #define ZIP_ER_OK 0 /* N No error */
- #define ZIP_ER_MULTIDISK 1 /* N Multi-disk zip archives not supported */
- #define ZIP_ER_RENAME 2 /* S Renaming temporary file failed */
- #define ZIP_ER_CLOSE 3 /* S Closing zip archive failed */
- #define ZIP_ER_SEEK 4 /* S Seek error */
- #define ZIP_ER_READ 5 /* S Read error */
- #define ZIP_ER_WRITE 6 /* S Write error */
- #define ZIP_ER_CRC 7 /* N CRC error */
- #define ZIP_ER_ZIPCLOSED 8 /* N Containing zip archive was closed */
- #define ZIP_ER_NOENT 9 /* N No such file */
- #define ZIP_ER_EXISTS 10 /* N File already exists */
- #define ZIP_ER_OPEN 11 /* S Can't open file */
- #define ZIP_ER_TMPOPEN 12 /* S Failure to create temporary file */
- #define ZIP_ER_ZLIB 13 /* Z Zlib error */
- #define ZIP_ER_MEMORY 14 /* N Malloc failure */
- #define ZIP_ER_CHANGED 15 /* N Entry has been changed */
- #define ZIP_ER_COMPNOTSUPP 16 /* N Compression method not supported */
- #define ZIP_ER_EOF 17 /* N Premature EOF */
- #define ZIP_ER_INVAL 18 /* N Invalid argument */
- #define ZIP_ER_NOZIP 19 /* N Not a zip archive */
- #define ZIP_ER_INTERNAL 20 /* N Internal error */
- #define ZIP_ER_INCONS 21 /* N Zip archive inconsistent */
- #define ZIP_ER_REMOVE 22 /* S Can't remove file */
- #define ZIP_ER_DELETED 23 /* N Entry has been deleted */
- /* type of system error value */
- #define ZIP_ET_NONE 0 /* sys_err unused */
- #define ZIP_ET_SYS 1 /* sys_err is errno */
- #define ZIP_ET_ZLIB 2 /* sys_err is zlib error code */
- /* compression methods */
- #define ZIP_CM_DEFAULT -1 /* better of deflate or store */
- #define ZIP_CM_STORE 0 /* stored (uncompressed) */
- #define ZIP_CM_SHRINK 1 /* shrunk */
- #define ZIP_CM_REDUCE_1 2 /* reduced with factor 1 */
- #define ZIP_CM_REDUCE_2 3 /* reduced with factor 2 */
- #define ZIP_CM_REDUCE_3 4 /* reduced with factor 3 */
- #define ZIP_CM_REDUCE_4 5 /* reduced with factor 4 */
- #define ZIP_CM_IMPLODE 6 /* imploded */
- /* 7 - Reserved for Tokenizing compression algorithm */
- #define ZIP_CM_DEFLATE 8 /* deflated */
- #define ZIP_CM_DEFLATE64 9 /* deflate64 */
- #define ZIP_CM_PKWARE_IMPLODE 10 /* PKWARE imploding */
- /* 11 - Reserved by PKWARE */
- #define ZIP_CM_BZIP2 12 /* compressed using BZIP2 algorithm */
- /* 13 - Reserved by PKWARE */
- #define ZIP_CM_LZMA 14 /* LZMA (EFS) */
- /* 15-17 - Reserved by PKWARE */
- #define ZIP_CM_TERSE 18 /* compressed using IBM TERSE (new) */
- #define ZIP_CM_LZ77 19 /* IBM LZ77 z Architecture (PFS) */
- #define ZIP_CM_WAVPACK 97 /* WavPack compressed data */
- #define ZIP_CM_PPMD 98 /* PPMd version I, Rev 1 */
- /* encryption methods */
- #define ZIP_EM_NONE 0 /* not encrypted */
- #define ZIP_EM_TRAD_PKWARE 1 /* traditional PKWARE encryption */
- #if 0 /* Strong Encryption Header not parsed yet */
- #define ZIP_EM_DES 0x6601 /* strong encryption: DES */
- #define ZIP_EM_RC2_OLD 0x6602 /* strong encryption: RC2, version < 5.2 */
- #define ZIP_EM_3DES_168 0x6603
- #define ZIP_EM_3DES_112 0x6609
- #define ZIP_EM_AES_128 0x660e
- #define ZIP_EM_AES_192 0x660f
- #define ZIP_EM_AES_256 0x6610
- #define ZIP_EM_RC2 0x6702 /* strong encryption: RC2, version >= 5.2 */
- #define ZIP_EM_RC4 0x6801
- #endif
- #define ZIP_EM_UNKNOWN 0xffff /* unknown algorithm */
- typedef long myoff_t; /* XXX: 64 bit support */
- enum zip_source_cmd {
- ZIP_SOURCE_OPEN, /* prepare for reading */
- ZIP_SOURCE_READ, /* read data */
- ZIP_SOURCE_CLOSE, /* reading is done */
- ZIP_SOURCE_STAT, /* get meta information */
- ZIP_SOURCE_ERROR, /* get error information */
- ZIP_SOURCE_FREE /* cleanup and free resources */
- };
- typedef ssize_t (*zip_source_callback)(void *state, void *data,
- size_t len, enum zip_source_cmd cmd);
- struct zip_stat {
- const char *name; /* name of the file */
- int index; /* index within archive */
- unsigned int crc; /* crc of file data */
- time_t mtime; /* modification time */
- myoff_t size; /* size of file (uncompressed) */
- myoff_t comp_size; /* size of file (compressed) */
- unsigned short comp_method; /* compression method used */
- unsigned short encryption_method; /* encryption method used */
- };
- struct zip;
- struct zip_file;
- struct zip_source;
- ZIP_EXTERN int zip_add(struct zip *, const char *, struct zip_source *);
- ZIP_EXTERN int zip_add_dir(struct zip *, const char *);
- ZIP_EXTERN int zip_close(struct zip *);
- ZIP_EXTERN int zip_delete(struct zip *, int);
- ZIP_EXTERN void zip_error_clear(struct zip *);
- ZIP_EXTERN void zip_error_get(struct zip *, int *, int *);
- ZIP_EXTERN int zip_error_get_sys_type(int);
- ZIP_EXTERN int zip_error_to_str(char *, size_t, int, int);
- ZIP_EXTERN int zip_fclose(struct zip_file *);
- ZIP_EXTERN void zip_file_error_clear(struct zip_file *);
- ZIP_EXTERN void zip_file_error_get(struct zip_file *, int *, int *);
- ZIP_EXTERN const char *zip_file_strerror(struct zip_file *);
- ZIP_EXTERN struct zip_file *zip_fopen(struct zip *, const char *, int);
- ZIP_EXTERN struct zip_file *zip_fopen_index(struct zip *, int, int);
- ZIP_EXTERN ssize_t zip_fread(struct zip_file *, void *, size_t);
- ZIP_EXTERN const char *zip_get_archive_comment(struct zip *, int *, int);
- ZIP_EXTERN int zip_get_archive_flag(struct zip *, int, int);
- ZIP_EXTERN const char *zip_get_file_comment(struct zip *, int, int *, int);
- ZIP_EXTERN const char *zip_get_name(struct zip *, int, int);
- ZIP_EXTERN int zip_get_num_files(struct zip *);
- ZIP_EXTERN int zip_name_locate(struct zip *, const char *, int);
- ZIP_EXTERN struct zip *zip_open(const char *, int, int *);
- ZIP_EXTERN int zip_rename(struct zip *, int, const char *);
- ZIP_EXTERN int zip_replace(struct zip *, int, struct zip_source *);
- ZIP_EXTERN int zip_set_archive_comment(struct zip *, const char *, int);
- ZIP_EXTERN int zip_set_archive_flag(struct zip *, int, int);
- ZIP_EXTERN int zip_set_file_comment(struct zip *, int, const char *, int);
- ZIP_EXTERN struct zip_source *zip_source_buffer(struct zip *, const void *,
- myoff_t, int);
- ZIP_EXTERN struct zip_source *zip_source_file(struct zip *, const char *,
- myoff_t, myoff_t);
- ZIP_EXTERN struct zip_source *zip_source_filep(struct zip *, FILE *,
- myoff_t, myoff_t);
- ZIP_EXTERN void zip_source_free(struct zip_source *);
- ZIP_EXTERN struct zip_source *zip_source_function(struct zip *,
- zip_source_callback, void *);
- ZIP_EXTERN struct zip_source *zip_source_zip(struct zip *, struct zip *,
- int, int, myoff_t, myoff_t);
- ZIP_EXTERN int zip_stat(struct zip *, const char *, int, struct zip_stat *);
- ZIP_EXTERN int zip_stat_index(struct zip *, int, int, struct zip_stat *);
- ZIP_EXTERN void zip_stat_init(struct zip_stat *);
- ZIP_EXTERN const char *zip_strerror(struct zip *);
- ZIP_EXTERN int zip_unchange(struct zip *, int);
- ZIP_EXTERN int zip_unchange_all(struct zip *);
- ZIP_EXTERN int zip_unchange_archive(struct zip *);
- #ifdef __cplusplus
- }
- #endif
- /* config.h. Generated from config.h.in by configure. */
- /* config.h.in. Generated from configure.ac by autoheader. */
- /* Define to 1 if you have the declaration of `tzname', and to 0 if you don't.
- */
- /* #undef HAVE_DECL_TZNAME */
- #define HAVE_CONFIG_H 1
- /* Define to 1 if you have the <dlfcn.h> header file. */
- #define HAVE_DLFCN_H 1
- /* Define to 1 if you have the `fseeko' function. */
- #define HAVE_FSEEKO 1
- /* Define to 1 if you have the `ftello' function. */
- #define HAVE_FTELLO 1
- /* Define to 1 if you have the <inttypes.h> header file. */
- #define HAVE_INTTYPES_H 1
- /* Define to 1 if you have the `z' library (-lz). */
- #define HAVE_LIBZ 1
- /* Define to 1 if you have the <memory.h> header file. */
- #define HAVE_MEMORY_H 1
- /* Define to 1 if you have the `mkstemp' function. */
- #define HAVE_MKSTEMP 1
- /* Define to 1 if you have the `MoveFileExA' function. */
- /* #undef HAVE_MOVEFILEEXA */
- /* Define to 1 if you have the <stdint.h> header file. */
- #define HAVE_STDINT_H 1
- /* Define to 1 if you have the <stdlib.h> header file. */
- #define HAVE_STDLIB_H 1
- /* Define to 1 if you have the <strings.h> header file. */
- #define HAVE_STRINGS_H 1
- /* Define to 1 if you have the <string.h> header file. */
- #define HAVE_STRING_H 1
- /* Define to 1 if `tm_zone' is member of `struct tm'. */
- #ifdef WIN32
- #undef HAVE_STRUCT_TM_TM_ZONE
- #else
- #define HAVE_STRUCT_TM_TM_ZONE 1
- #endif
- /* Define to 1 if you have the <sys/stat.h> header file. */
- #define HAVE_SYS_STAT_H 1
- /* Define to 1 if you have the <sys/types.h> header file. */
- #define HAVE_SYS_TYPES_H 1
- /* Define to 1 if your `struct tm' has `tm_zone'. Deprecated, use
- `HAVE_STRUCT_TM_TM_ZONE' instead. */
- #define HAVE_TM_ZONE 1
- /* Define to 1 if you don't have `tm_zone' but do have the external array
- `tzname'. */
- /* #undef HAVE_TZNAME */
- /* Define to 1 if you have the <unistd.h> header file. */
- #define HAVE_UNISTD_H 1
- /* Define to 1 if your C compiler doesn't accept -c and -o together. */
- /* #undef NO_MINUS_C_MINUS_O */
- /* Name of package */
- #define PACKAGE "libzip"
- /* Define to the address where bug reports for this package should be sent. */
- #define PACKAGE_BUGREPORT "libzip@nih.at"
- /* Define to the full name of this package. */
- #define PACKAGE_NAME "libzip"
- /* Define to the full name and version of this package. */
- #define PACKAGE_STRING "libzip 0.9"
- /* Define to the one symbol short name of this package. */
- #define PACKAGE_TARNAME "libzip"
- /* Define to the version of this package. */
- #define PACKAGE_VERSION "0.9"
- /* Define to 1 if you have the ANSI C header files. */
- #define STDC_HEADERS 1
- /* Define to 1 if your <sys/time.h> declares `struct tm'. */
- /* #undef TM_IN_SYS_TIME */
- /* Version number of package */
- #define VERSION "0.9"
- #ifndef HAVE_MKSTEMP
- int _zip_mkstemp(char *);
- #define mkstemp _zip_mkstemp
- #endif
- #ifdef HAVE_MOVEFILEEXA
- #include <windows.h>
- #define _zip_rename(s, t) \
- (!MoveFileExA((s), (t), \
- MOVEFILE_COPY_ALLOWED|MOVEFILE_REPLACE_EXISTING))
- #else
- #define _zip_rename rename
- #endif
- #ifndef HAVE_FSEEKO
- #define fseeko(s, o, w) (fseek((s), (long int)(o), (w)))
- #endif
- #ifndef HAVE_FTELLO
- #define ftello(s) ((long)ftell((s)))
- #endif
- #define CENTRAL_MAGIC "PK\1\2"
- #define LOCAL_MAGIC "PK\3\4"
- #define EOCD_MAGIC "PK\5\6"
- #define DATADES_MAGIC "PK\7\8"
- #define TORRENT_SIG "TORRENTZIPPED-"
- #define TORRENT_SIG_LEN 14
- #define TORRENT_CRC_LEN 8
- #define TORRENT_MEM_LEVEL 8
- #define CDENTRYSIZE 46u
- #define LENTRYSIZE 30
- #define MAXCOMLEN 65536
- #define EOCDLEN 22
- #define CDBUFSIZE (MAXCOMLEN+EOCDLEN)
- #define BUFSIZE 8192
- /* state of change of a file in zip archive */
- enum zip_state { ZIP_ST_UNCHANGED, ZIP_ST_DELETED, ZIP_ST_REPLACED,
- ZIP_ST_ADDED, ZIP_ST_RENAMED };
- /* constants for struct zip_file's member flags */
- #define ZIP_ZF_EOF 1 /* EOF reached */
- #define ZIP_ZF_DECOMP 2 /* decompress data */
- #define ZIP_ZF_CRC 4 /* compute and compare CRC */
- /* directory entry: general purpose bit flags */
- #define ZIP_GPBF_ENCRYPTED 0x0001 /* is encrypted */
- #define ZIP_GPBF_DATA_DESCRIPTOR 0x0008 /* crc/size after file data */
- #define ZIP_GPBF_STRONG_ENCRYPTION 0x0040 /* uses strong encryption */
- /* error information */
- struct zip_error {
- int zip_err; /* libzip error code (ZIP_ER_*) */
- int sys_err; /* copy of errno (E*) or zlib error code */
- char *str; /* string representation or NULL */
- };
- /* zip archive, part of API */
- struct zip {
- char *zn; /* file name */
- FILE *zp; /* file */
- struct zip_error error; /* error information */
- unsigned int flags; /* archive global flags */
- unsigned int ch_flags; /* changed archive global flags */
- struct zip_cdir *cdir; /* central directory */
- char *ch_comment; /* changed archive comment */
- int ch_comment_len; /* length of changed zip archive
- * comment, -1 if unchanged */
- int nentry; /* number of entries */
- int nentry_alloc; /* number of entries allocated */
- struct zip_entry *entry; /* entries */
- int nfile; /* number of opened files within archive */
- int nfile_alloc; /* number of files allocated */
- struct zip_file **file; /* opened files within archive */
- };
- /* file in zip archive, part of API */
- struct zip_file {
- struct zip *za; /* zip archive containing this file */
- struct zip_error error; /* error information */
- int flags; /* -1: eof, >0: error */
- int method; /* compression method */
- myoff_t fpos; /* position within zip file (fread/fwrite) */
- unsigned long bytes_left; /* number of bytes left to read */
- unsigned long cbytes_left; /* number of bytes of compressed data left */
- unsigned long crc; /* CRC so far */
- unsigned long crc_orig; /* CRC recorded in archive */
- char *buffer;
- z_stream *zstr;
- };
- /* zip archive directory entry (central or local) */
- struct zip_dirent {
- unsigned short version_madeby; /* (c) version of creator */
- unsigned short version_needed; /* (cl) version needed to extract */
- unsigned short bitflags; /* (cl) general purpose bit flag */
- unsigned short comp_method; /* (cl) compression method used */
- time_t last_mod; /* (cl) time of last modification */
- unsigned int crc; /* (cl) CRC-32 of uncompressed data */
- unsigned int comp_size; /* (cl) size of commpressed data */
- unsigned int uncomp_size; /* (cl) size of uncommpressed data */
- char *filename; /* (cl) file name (NUL-terminated) */
- unsigned short filename_len; /* (cl) length of filename (w/o NUL) */
- char *extrafield; /* (cl) extra field */
- unsigned short extrafield_len; /* (cl) length of extra field */
- char *comment; /* (c) file comment */
- unsigned short comment_len; /* (c) length of file comment */
- unsigned short disk_number; /* (c) disk number start */
- unsigned short int_attrib; /* (c) internal file attributes */
- unsigned int ext_attrib; /* (c) external file attributes */
- unsigned int offset; /* (c) offset of local header */
- };
- /* zip archive central directory */
- struct zip_cdir {
- struct zip_dirent *entry; /* directory entries */
- int nentry; /* number of entries */
- unsigned int size; /* size of central direcotry */
- unsigned int offset; /* offset of central directory in file */
- char *comment; /* zip archive comment */
- unsigned short comment_len; /* length of zip archive comment */
- };
- struct zip_source {
- zip_source_callback f;
- void *ud;
- };
- /* entry in zip archive directory */
- struct zip_entry {
- enum zip_state state;
- struct zip_source *source;
- char *ch_filename;
- char *ch_comment;
- int ch_comment_len;
- };
- extern const char * const _zip_err_str[];
- extern const int _zip_nerr_str;
- extern const int _zip_err_type[];
- #define ZIP_ENTRY_DATA_CHANGED(x) \
- ((x)->state == ZIP_ST_REPLACED \
- || (x)->state == ZIP_ST_ADDED)
- int _zip_cdir_compute_crc(struct zip *, uLong *);
- void _zip_cdir_free(struct zip_cdir *);
- struct zip_cdir *_zip_cdir_new(int, struct zip_error *);
- int _zip_cdir_write(struct zip_cdir *, FILE *, struct zip_error *);
- void _zip_dirent_finalize(struct zip_dirent *);
- void _zip_dirent_init(struct zip_dirent *);
- int _zip_dirent_read(struct zip_dirent *, FILE *,
- unsigned char **, unsigned int, int, struct zip_error *);
- void _zip_dirent_torrent_normalize(struct zip_dirent *);
- int _zip_dirent_write(struct zip_dirent *, FILE *, int, struct zip_error *);
- void _zip_entry_free(struct zip_entry *);
- void _zip_entry_init(struct zip *, int);
- struct zip_entry *_zip_entry_new(struct zip *);
- void _zip_error_clear(struct zip_error *);
- void _zip_error_copy(struct zip_error *, struct zip_error *);
- void _zip_error_fini(struct zip_error *);
- void _zip_error_get(struct zip_error *, int *, int *);
- void _zip_error_init(struct zip_error *);
- void _zip_error_set(struct zip_error *, int, int);
- const char *_zip_error_strerror(struct zip_error *);
- int _zip_file_fillbuf(void *, size_t, struct zip_file *);
- unsigned int _zip_file_get_offset(struct zip *, int);
- int _zip_filerange_crc(FILE *, myoff_t, myoff_t, uLong *, struct zip_error *);
- struct zip_source *_zip_source_file_or_p(struct zip *, const char *, FILE *,
- myoff_t, myoff_t);
- void _zip_free(struct zip *);
- const char *_zip_get_name(struct zip *, int, int, struct zip_error *);
- int _zip_local_header_read(struct zip *, int);
- void *_zip_memdup(const void *, size_t, struct zip_error *);
- int _zip_name_locate(struct zip *, const char *, int, struct zip_error *);
- struct zip *_zip_new(struct zip_error *);
- unsigned short _zip_read2(unsigned char **);
- unsigned int _zip_read4(unsigned char **);
- int _zip_replace(struct zip *, int, const char *, struct zip_source *);
- int _zip_set_name(struct zip *, int, const char *);
- int _zip_unchange(struct zip *, int, int);
- void _zip_unchange_data(struct zip_entry *);
- #include <errno.h>
- #include <stdio.h>
- #include <stdlib.h>
- #include <string.h>
- const char *
- _zip_error_strerror(struct zip_error *err)
- {
- const char *zs, *ss;
- char buf[128], *s;
- _zip_error_fini(err);
- if (err->zip_err < 0 || err->zip_err >= _zip_nerr_str) {
- sprintf(buf, "Unknown error %d", err->zip_err);
- zs = NULL;
- ss = buf;
- }
- else {
- zs = _zip_err_str[err->zip_err];
- switch (_zip_err_type[err->zip_err]) {
- case ZIP_ET_SYS:
- ss = strerror(err->sys_err);
- break;
- case ZIP_ET_ZLIB:
- ss = zError(err->sys_err);
- break;
- default:
- ss = NULL;
- }
- }
- if (ss == NULL)
- return zs;
- else {
- if ((s=(char *)malloc(strlen(ss)
- + (zs ? strlen(zs)+2 : 0) + 1)) == NULL)
- return _zip_err_str[ZIP_ER_MEMORY];
- sprintf(s, "%s%s%s",
- (zs ? zs : ""),
- (zs ? ": " : ""),
- ss);
- err->str = s;
- return s;
- }
- }
- #include <stdlib.h>
- void
- _zip_error_clear(struct zip_error *err)
- {
- err->zip_err = ZIP_ER_OK;
- err->sys_err = 0;
- }
- void
- _zip_error_copy(struct zip_error *dst, struct zip_error *src)
- {
- dst->zip_err = src->zip_err;
- dst->sys_err = src->sys_err;
- }
- void
- _zip_error_fini(struct zip_error *err)
- {
- free(err->str);
- err->str = NULL;
- }
- void
- _zip_error_get(struct zip_error *err, int *zep, int *sep)
- {
- if (zep)
- *zep = err->zip_err;
- if (sep) {
- if (zip_error_get_sys_type(err->zip_err) != ZIP_ET_NONE)
- *sep = err->sys_err;
- else
- *sep = 0;
- }
- }
- void
- _zip_error_init(struct zip_error *err)
- {
- err->zip_err = ZIP_ER_OK;
- err->sys_err = 0;
- err->str = NULL;
- }
- void
- _zip_error_set(struct zip_error *err, int ze, int se)
- {
- if (err) {
- err->zip_err = ze;
- err->sys_err = se;
- }
- }
- #include <sys/types.h>
- #include <sys/stat.h>
- #include <assert.h>
- #include <ctype.h>
- #include <errno.h>
- #include <fcntl.h>
- #include <stdio.h>
- #include <stdlib.h>
- #ifndef O_BINARY
- #define O_BINARY 0
- #endif
- int
- _zip_mkstemp(char *path)
- {
- int fd;
- char *start, *trv;
- struct stat sbuf;
- pid_t pid;
- /* To guarantee multiple calls generate unique names even if
- the file is not created. 676 different possibilities with 7
- or more X's, 26 with 6 or less. */
- static char xtra[2] = "aa";
- int xcnt = 0;
- pid = getpid();
- /* Move to end of path and count trailing X's. */
- for (trv = path; *trv; ++trv)
- if (*trv == 'X')
- xcnt++;
- else
- xcnt = 0;
- /* Use at least one from xtra. Use 2 if more than 6 X's. */
- if (*(trv - 1) == 'X')
- *--trv = xtra[0];
- if (xcnt > 6 && *(trv - 1) == 'X')
- *--trv = xtra[1];
- /* Set remaining X's to pid digits with 0's to the left. */
- while (*--trv == 'X') {
- *trv = (pid % 10) + '0';
- pid /= 10;
- }
- /* update xtra for next call. */
- if (xtra[0] != 'z')
- xtra[0]++;
- else {
- xtra[0] = 'a';
- if (xtra[1] != 'z')
- xtra[1]++;
- else
- xtra[1] = 'a';
- }
- /*
- * check the target directory; if you have six X's and it
- * doesn't exist this runs for a *very* long time.
- */
- for (start = trv + 1;; --trv) {
- if (trv <= path)
- break;
- if (*trv == '/') {
- *trv = '\0';
- if (stat(path, &sbuf))
- return (0);
- if (!S_ISDIR(sbuf.st_mode)) {
- errno = ENOTDIR;
- return (0);
- }
- *trv = '/';
- break;
- }
- }
- for (;;) {
- if ((fd=open(path, O_CREAT|O_EXCL|O_RDWR|O_BINARY, 0600)) >= 0)
- return (fd);
- if (errno != EEXIST)
- return (0);
- /* tricky little algorithm for backward compatibility */
- for (trv = start;;) {
- if (!*trv)
- return (0);
- if (*trv == 'z')
- *trv++ = 'a';
- else {
- if (isdigit((unsigned char)*trv))
- *trv = 'a';
- else
- ++*trv;
- break;
- }
- }
- }
- /*NOTREACHED*/
- }
- #include <stdio.h>
- #include <stdlib.h>
- #include <string.h>
- #include <errno.h>
- #include <sys/types.h>
- #include <sys/stat.h>
- static time_t _zip_d2u_time(int, int);
- static char *_zip_readfpstr(FILE *, unsigned int, int, struct zip_error *);
- static char *_zip_readstr(unsigned char **, int, int, struct zip_error *);
- static void _zip_u2d_time(time_t, unsigned short *, unsigned short *);
- static void _zip_write2(unsigned short, FILE *);
- static void _zip_write4(unsigned int, FILE *);
- void
- _zip_cdir_free(struct zip_cdir *cd)
- {
- int i;
- if (!cd)
- return;
- for (i=0; i<cd->nentry; i++)
- _zip_dirent_finalize(cd->entry+i);
- free(cd->comment);
- free(cd->entry);
- free(cd);
- }
- struct zip_cdir *
- _zip_cdir_new(int nentry, struct zip_error *error)
- {
- struct zip_cdir *cd;
- if ((cd=(struct zip_cdir *)malloc(sizeof(*cd))) == NULL) {
- _zip_error_set(error, ZIP_ER_MEMORY, 0);
- return NULL;
- }
- if ((cd->entry=(struct zip_dirent *)malloc(sizeof(*(cd->entry))*nentry))
- == NULL) {
- _zip_error_set(error, ZIP_ER_MEMORY, 0);
- free(cd);
- return NULL;
- }
- /* entries must be initialized by caller */
- cd->nentry = nentry;
- cd->size = cd->offset = 0;
- cd->comment = NULL;
- cd->comment_len = 0;
- return cd;
- }
- int
- _zip_cdir_write(struct zip_cdir *cd, FILE *fp, struct zip_error *error)
- {
- int i;
- cd->offset = ftello(fp);
- for (i=0; i<cd->nentry; i++) {
- if (_zip_dirent_write(cd->entry+i, fp, 0, error) != 0)
- return -1;
- }
- cd->size = ftello(fp) - cd->offset;
- /* clearerr(fp); */
- fwrite(EOCD_MAGIC, 1, 4, fp);
- _zip_write4(0, fp);
- _zip_write2((unsigned short)cd->nentry, fp);
- _zip_write2((unsigned short)cd->nentry, fp);
- _zip_write4(cd->size, fp);
- _zip_write4(cd->offset, fp);
- _zip_write2(cd->comment_len, fp);
- fwrite(cd->comment, 1, cd->comment_len, fp);
- if (ferror(fp)) {
- _zip_error_set(error, ZIP_ER_WRITE, errno);
- return -1;
- }
- return 0;
- }
- void
- _zip_dirent_finalize(struct zip_dirent *zde)
- {
- free(zde->filename);
- zde->filename = NULL;
- free(zde->extrafield);
- zde->extrafield = NULL;
- free(zde->comment);
- zde->comment = NULL;
- }
- void
- _zip_dirent_init(struct zip_dirent *de)
- {
- de->version_madeby = 0;
- de->version_needed = 20; /* 2.0 */
- de->bitflags = 0;
- de->comp_method = 0;
- de->last_mod = 0;
- de->crc = 0;
- de->comp_size = 0;
- de->uncomp_size = 0;
- de->filename = NULL;
- de->filename_len = 0;
- de->extrafield = NULL;
- de->extrafield_len = 0;
- de->comment = NULL;
- de->comment_len = 0;
- de->disk_number = 0;
- de->int_attrib = 0;
- de->ext_attrib = 0;
- de->offset = 0;
- }
- /* _zip_dirent_read(zde, fp, bufp, left, localp, error):
- Fills the zip directory entry zde.
- If bufp is non-NULL, data is taken from there and bufp is advanced
- by the amount of data used; no more than left bytes are used.
- Otherwise data is read from fp as needed.
- If localp != 0, it reads a local header instead of a central
- directory entry.
- Returns 0 if successful. On error, error is filled in and -1 is
- returned.
- */
- int
- _zip_dirent_read(struct zip_dirent *zde, FILE *fp,
- unsigned char **bufp, unsigned int left, int localp,
- struct zip_error *error)
- {
- unsigned char buf[CDENTRYSIZE];
- unsigned char *cur;
- unsigned short dostime, dosdate;
- unsigned int size;
- if (localp)
- size = LENTRYSIZE;
- else
- size = CDENTRYSIZE;
- if (bufp) {
- /* use data from buffer */
- cur = *bufp;
- if (left < size) {
- _zip_error_set(error, ZIP_ER_NOZIP, 0);
- return -1;
- }
- }
- else {
- /* read entry from disk */
- if ((fread(buf, 1, size, fp)<size)) {
- _zip_error_set(error, ZIP_ER_READ, errno);
- return -1;
- }
- left = size;
- cur = buf;
- }
- if (memcmp(cur, (localp ? LOCAL_MAGIC : CENTRAL_MAGIC), 4) != 0) {
- _zip_error_set(error, ZIP_ER_NOZIP, 0);
- return -1;
- }
- cur += 4;
- /* convert buffercontents to zip_dirent */
- if (!localp)
- zde->version_madeby = _zip_read2(&cur);
- else
- zde->version_madeby = 0;
- zde->version_needed = _zip_read2(&cur);
- zde->bitflags = _zip_read2(&cur);
- zde->comp_method = _zip_read2(&cur);
- /* convert to time_t */
- dostime = _zip_read2(&cur);
- dosdate = _zip_read2(&cur);
- zde->last_mod = _zip_d2u_time(dostime, dosdate);
- zde->crc = _zip_read4(&cur);
- zde->comp_size = _zip_read4(&cur);
- zde->uncomp_size = _zip_read4(&cur);
- zde->filename_len = _zip_read2(&cur);
- zde->extrafield_len = _zip_read2(&cur);
- if (localp) {
- zde->comment_len = 0;
- zde->disk_number = 0;
- zde->int_attrib = 0;
- zde->ext_attrib = 0;
- zde->offset = 0;
- } else {
- zde->comment_len = _zip_read2(&cur);
- zde->disk_number = _zip_read2(&cur);
- zde->int_attrib = _zip_read2(&cur);
- zde->ext_attrib = _zip_read4(&cur);
- zde->offset = _zip_read4(&cur);
- }
- zde->filename = NULL;
- zde->extrafield = NULL;
- zde->comment = NULL;
- if (bufp) {
- if (left < CDENTRYSIZE + (zde->filename_len+zde->extrafield_len
- +zde->comment_len)) {
- _zip_error_set(error, ZIP_ER_NOZIP, 0);
- return -1;
- }
- if (zde->filename_len) {
- zde->filename = _zip_readstr(&cur, zde->filename_len, 1, error);
- if (!zde->filename)
- return -1;
- }
- if (zde->extrafield_len) {
- zde->extrafield = _zip_readstr(&cur, zde->extrafield_len, 0,
- error);
- if (!zde->extrafield)
- return -1;
- }
- if (zde->comment_len) {
- zde->comment = _zip_readstr(&cur, zde->comment_len, 0, error);
- if (!zde->comment)
- return -1;
- }
- }
- else {
- if (zde->filename_len) {
- zde->filename = _zip_readfpstr(fp, zde->filename_len, 1, error);
- if (!zde->filename)
- return -1;
- }
- if (zde->extrafield_len) {
- zde->extrafield = _zip_readfpstr(fp, zde->extrafield_len, 0,
- error);
- if (!zde->extrafield)
- return -1;
- }
- if (zde->comment_len) {
- zde->comment = _zip_readfpstr(fp, zde->comment_len, 0, error);
- if (!zde->comment)
- return -1;
- }
- }
- if (bufp)
- *bufp = cur;
- return 0;
- }
- /* _zip_dirent_torrent_normalize(de);
- Set values suitable for torrentzip.
- */
- void
- _zip_dirent_torrent_normalize(struct zip_dirent *de)
- {
- static struct tm torrenttime;
- static time_t last_mod = 0;
- if (last_mod == 0) {
- #ifdef HAVE_STRUCT_TM_TM_ZONE
- time_t now;
- struct tm *l;
- #endif
- torrenttime.tm_sec = 0;
- torrenttime.tm_min = 32;
- torrenttime.tm_hour = 23;
- torrenttime.tm_mday = 24;
- torrenttime.tm_mon = 11;
- torrenttime.tm_year = 96;
- torrenttime.tm_wday = 0;
- torrenttime.tm_yday = 0;
- torrenttime.tm_isdst = 0;
- #ifdef HAVE_STRUCT_TM_TM_ZONE
- time(&now);
- l = localtime(&now);
- torrenttime.tm_gmtoff = l->tm_gmtoff;
- torrenttime.tm_zone = l->tm_zone;
- #endif
- last_mod = mktime(&torrenttime);
- }
- de->version_madeby = 0;
- de->version_needed = 20; /* 2.0 */
- de->bitflags = 2; /* maximum compression */
- de->comp_method = ZIP_CM_DEFLATE;
- de->last_mod = last_mod;
- de->disk_number = 0;
- de->int_attrib = 0;
- de->ext_attrib = 0;
- de->offset = 0;
- free(de->extrafield);
- de->extrafield = NULL;
- de->extrafield_len = 0;
- free(de->comment);
- de->comment = NULL;
- de->comment_len = 0;
- }
- /* _zip_dirent_write(zde, fp, localp, error):
- Writes zip directory entry zde to file fp.
- If localp != 0, it writes a local header instead of a central
- directory entry.
- Returns 0 if successful. On error, error is filled in and -1 is
- returned.
- */
- int
- _zip_dirent_write(struct zip_dirent *zde, FILE *fp, int localp,
- struct zip_error *error)
- {
- unsigned short dostime, dosdate;
- fwrite(localp ? LOCAL_MAGIC : CENTRAL_MAGIC, 1, 4, fp);
- if (!localp)
- _zip_write2(zde->version_madeby, fp);
- _zip_write2(zde->version_needed, fp);
- _zip_write2(zde->bitflags, fp);
- _zip_write2(zde->comp_method, fp);
- _zip_u2d_time(zde->last_mod, &dostime, &dosdate);
- _zip_write2(dostime, fp);
- _zip_write2(dosdate, fp);
- _zip_write4(zde->crc, fp);
- _zip_write4(zde->comp_size, fp);
- _zip_write4(zde->uncomp_size, fp);
- _zip_write2(zde->filename_len, fp);
- _zip_write2(zde->extrafield_len, fp);
- if (!localp) {
- _zip_write2(zde->comment_len, fp);
- _zip_write2(zde->disk_number, fp);
- _zip_write2(zde->int_attrib, fp);
- _zip_write4(zde->ext_attrib, fp);
- _zip_write4(zde->offset, fp);
- }
- if (zde->filename_len)
- fwrite(zde->filename, 1, zde->filename_len, fp);
- if (zde->extrafield_len)
- fwrite(zde->extrafield, 1, zde->extrafield_len, fp);
- if (!localp) {
- if (zde->comment_len)
- fwrite(zde->comment, 1, zde->comment_len, fp);
- }
- if (ferror(fp)) {
- _zip_error_set(error, ZIP_ER_WRITE, errno);
- return -1;
- }
- return 0;
- }
- static time_t
- _zip_d2u_time(int dtime, int ddate)
- {
- struct tm *tm;
- time_t now;
- now = time(NULL);
- tm = localtime(&now);
- /* let mktime decide if DST is in effect */
- tm->tm_isdst = -1;
- tm->tm_year = ((ddate>>9)&127) + 1980 - 1900;
- tm->tm_mon = ((ddate>>5)&15) - 1;
- tm->tm_mday = ddate&31;
- tm->tm_hour = (dtime>>11)&31;
- tm->tm_min = (dtime>>5)&63;
- tm->tm_sec = (dtime<<1)&62;
- return mktime(tm);
- }
- unsigned short
- _zip_read2(unsigned char **a)
- {
- unsigned short ret;
- ret = (*a)[0]+((*a)[1]<<8);
- *a += 2;
- return ret;
- }
- unsigned int
- _zip_read4(unsigned char **a)
- {
- unsigned int ret;
- ret = ((((((*a)[3]<<8)+(*a)[2])<<8)+(*a)[1])<<8)+(*a)[0];
- *a += 4;
- return ret;
- }
- static char *
- _zip_readfpstr(FILE *fp, unsigned int len, int nulp, struct zip_error *error)
- {
- char *r, *o;
- r = (char *)malloc(nulp ? len+1 : len);
- if (!r) {
- _zip_error_set(error, ZIP_ER_MEMORY, 0);
- return NULL;
- }
- if (fread(r, 1, len, fp)<len) {
- free(r);
- _zip_error_set(error, ZIP_ER_READ, errno);
- return NULL;
- }
- if (nulp) {
- /* replace any in-string NUL characters with spaces */
- r[len] = 0;
- for (o=r; o<r+len; o++)
- if (*o == '\0')
- *o = ' ';
- }
- return r;
- }
- static char *
- _zip_readstr(unsigned char **buf, int len, int nulp, struct zip_error *error)
- {
- char *r, *o;
- r = (char *)malloc(nulp ? len+1 : len);
- if (!r) {
- _zip_error_set(error, ZIP_ER_MEMORY, 0);
- return NULL;
- }
- memcpy(r, *buf, len);
- *buf += len;
- if (nulp) {
- /* replace any in-string NUL characters with spaces */
- r[len] = 0;
- for (o=r; o<r+len; o++)
- if (*o == '\0')
- *o = ' ';
- }
- return r;
- }
- static void
- _zip_write2(unsigned short i, FILE *fp)
- {
- putc(i&0xff, fp);
- putc((i>>8)&0xff, fp);
- return;
- }
- static void
- _zip_write4(unsigned int i, FILE *fp)
- {
- putc(i&0xff, fp);
- putc((i>>8)&0xff, fp);
- putc((i>>16)&0xff, fp);
- putc((i>>24)&0xff, fp);
- return;
- }
- static void
- _zip_u2d_time(time_t time, unsigned short *dtime, unsigned short *ddate)
- {
- struct tm *tm;
- tm = localtime(&time);
- *ddate = ((tm->tm_year+1900-1980)<<9) + ((tm->tm_mon+1)<<5)
- + tm->tm_mday;
- *dtime = ((tm->tm_hour)<<11) + ((tm->tm_min)<<5)
- + ((tm->tm_sec)>>1);
- return;
- }
- ZIP_EXTERN int
- zip_delete(struct zip *za, int idx)
- {
- if (idx < 0 || idx >= za->nentry) {
- _zip_error_set(&za->error, ZIP_ER_INVAL, 0);
- return -1;
- }
- /* allow duplicate file names, because the file will
- * be removed directly afterwards */
- if (_zip_unchange(za, idx, 1) != 0)
- return -1;
- za->entry[idx].state = ZIP_ST_DELETED;
- return 0;
- }
- ZIP_EXTERN void
- zip_error_clear(struct zip *za)
- {
- _zip_error_clear(&za->error);
- }
- ZIP_EXTERN int
- zip_add(struct zip *za, const char *name, struct zip_source *source)
- {
- if (name == NULL || source == NULL) {
- _zip_error_set(&za->error, ZIP_ER_INVAL, 0);
- return -1;
- }
- return _zip_replace(za, -1, name, source);
- }
- ZIP_EXTERN int
- zip_error_get_sys_type(int ze)
- {
- if (ze < 0 || ze >= _zip_nerr_str)
- return 0;
- return _zip_err_type[ze];
- }
- ZIP_EXTERN void
- zip_error_get(struct zip *za, int *zep, int *sep)
- {
- _zip_error_get(&za->error, zep, sep);
- }
- const char * const _zip_err_str[] = {
- "No error",
- "Multi-disk zip archives not supported",
- "Renaming temporary file failed",
- "Closing zip archive failed",
- "Seek error",
- "Read error",
- "Write error",
- "CRC error",
- "Containing zip archive was closed",
- "No such file",
- "File already exists",
- "Can't open file",
- "Failure to create temporary file",
- "Zlib error",
- "Malloc failure",
- "Entry has been changed",
- "Compression method not supported",
- "Premature EOF",
- "Invalid argument",
- "Not a zip archive",
- "Internal error",
- "Zip archive inconsistent",
- "Can't remove file",
- "Entry has been deleted",
- };
- const int _zip_nerr_str = sizeof(_zip_err_str)/sizeof(_zip_err_str[0]);
- #define N ZIP_ET_NONE
- #define S ZIP_ET_SYS
- #define Z ZIP_ET_ZLIB
- const int _zip_err_type[] = {
- N,
- N,
- S,
- S,
- S,
- S,
- S,
- N,
- N,
- N,
- N,
- S,
- S,
- Z,
- N,
- N,
- N,
- N,
- N,
- N,
- N,
- N,
- S,
- N,
- };
- struct zip_entry *
- _zip_entry_new(struct zip *za)
- {
- struct zip_entry *ze;
- if (!za) {
- ze = (struct zip_entry *)malloc(sizeof(struct zip_entry));
- if (!ze) {
- _zip_error_set(&za->error, ZIP_ER_MEMORY, 0);
- return NULL;
- }
- }
- else {
- if (za->nentry >= za->nentry_alloc-1) {
- za->nentry_alloc += 16;
- za->entry = (struct zip_entry *)realloc(za->entry,
- sizeof(struct zip_entry)
- * za->nentry_alloc);
- if (!za->entry) {
- _zip_error_set(&za->error, ZIP_ER_MEMORY, 0);
- return NULL;
- }
- }
- ze = za->entry+za->nentry;
- }
- ze->state = ZIP_ST_UNCHANGED;
- ze->ch_filename = NULL;
- ze->ch_comment = NULL;
- ze->ch_comment_len = -1;
- ze->source = NULL;
- if (za)
- za->nentry++;
- return ze;
- }
- void
- _zip_entry_free(struct zip_entry *ze)
- {
- free(ze->ch_filename);
- ze->ch_filename = NULL;
- free(ze->ch_comment);
- ze->ch_comment = NULL;
- ze->ch_comment_len = -1;
- _zip_unchange_data(ze);
- }
- static int add_data(struct zip *, struct zip_source *, struct zip_dirent *,
- FILE *);
- static int add_data_comp(zip_source_callback, void *, struct zip_stat *,
- FILE *, struct zip_error *);
- static int add_data_uncomp(struct zip *, zip_source_callback, void *,
- struct zip_stat *, FILE *);
- static void ch_set_error(struct zip_error *, zip_source_callback, void *);
- static int copy_data(FILE *, myoff_t, FILE *, struct zip_error *);
- static int write_cdir(struct zip *, struct zip_cdir *, FILE *);
- static int _zip_cdir_set_comment(struct zip_cdir *, struct zip *);
- static int _zip_changed(struct zip *, int *);
- static char *_zip_create_temp_output(struct zip *, FILE **);
- static int _zip_torrentzip_cmp(const void *, const void *);
- struct filelist {
- int idx;
- const char *name;
- };
- ZIP_EXTERN int
- zip_close(struct zip *za)
- {
- int survivors;
- int i, j, error;
- char *temp;
- FILE *out;
- mode_t mask;
- struct zip_cdir *cd;
- struct zip_dirent de;
- struct filelist *filelist;
- int reopen_on_error;
- int new_torrentzip;
- reopen_on_error = 0;
- if (za == NULL)
- return -1;
- if (!_zip_changed(za, &survivors)) {
- _zip_free(za);
- return 0;
- }
- /* don't create zip files with no entries */
- if (survivors == 0) {
- if (za->zn && za->zp) {
- if (remove(za->zn) != 0) {
- _zip_error_set(&za->error, ZIP_ER_REMOVE, errno);
- return -1;
- }
- }
- _zip_free(za);
- return 0;
- }
- if ((filelist=(struct filelist *)malloc(sizeof(filelist[0])*survivors))
- == NULL)
- return -1;
- if ((cd=_zip_cdir_new(survivors, &za->error)) == NULL) {
- free(filelist);
- return -1;
- }
- for (i=0; i<survivors; i++)
- _zip_dirent_init(&cd->entry[i]);
- /* archive comment is special for torrentzip */
- if (zip_get_archive_flag(za, ZIP_AFL_TORRENT, 0)) {
- cd->comment = _zip_memdup(TORRENT_SIG "XXXXXXXX",
- TORRENT_SIG_LEN + TORRENT_CRC_LEN,
- &za->error);
- if (cd->comment == NULL) {
- _zip_cdir_free(cd);
- free(filelist);
- return -1;
- }
- cd->comment_len = TORRENT_SIG_LEN + TORRENT_CRC_LEN;
- }
- else if (zip_get_archive_flag(za, ZIP_AFL_TORRENT, ZIP_FL_UNCHANGED) == 0) {
- if (_zip_cdir_set_comment(cd, za) == -1) {
- _zip_cdir_free(cd);
- free(filelist);
- return -1;
- }
- }
- if ((temp=_zip_create_temp_output(za, &out)) == NULL) {
- _zip_cdir_free(cd);
- return -1;
- }
- /* create list of files with index into original archive */
- for (i=j=0; i<za->nentry; i++) {
- if (za->entry[i].state == ZIP_ST_DELETED)
- continue;
- filelist[j].idx = i;
- filelist[j].name = zip_get_name(za, i, 0);
- j++;
- }
- if (zip_get_archive_flag(za, ZIP_AFL_TORRENT, 0))
- qsort(filelist, survivors, sizeof(filelist[0]),
- _zip_torrentzip_cmp);
- new_torrentzip = (zip_get_archive_flag(za, ZIP_AFL_TORRENT, 0) == 1
- && zip_get_archive_flag(za, ZIP_AFL_TORRENT,
- ZIP_FL_UNCHANGED) == 0);
- error = 0;
- for (j=0; j<survivors; j++) {
- i = filelist[j].idx;
- /* create new local directory entry */
- if (ZIP_ENTRY_DATA_CHANGED(za->entry+i) || new_torrentzip) {
- _zip_dirent_init(&de);
- if (zip_get_archive_flag(za, ZIP_AFL_TORRENT, 0))
- _zip_dirent_torrent_normalize(&de);
- /* use it as central directory entry */
- memcpy(cd->entry+j, &de, sizeof(cd->entry[j]));
- /* set/update file name */
- if (za->entry[i].ch_filename == NULL) {
- if (za->entry[i].state == ZIP_ST_ADDED) {
- de.filename = strdup("-");
- de.filename_len = 1;
- cd->entry[j].filename = "-";
- }
- else {
- de.filename = strdup(za->cdir->entry[i].filename);
- de.filename_len = strlen(de.filename);
- cd->entry[j].filename = za->cdir->entry[i].filename;
- cd->entry[j].filename_len = de.filename_len;
- }
- }
- }
- else {
- /* copy existing directory entries */
- if (fseeko(za->zp, za->cdir->entry[i].offset, SEEK_SET) != 0) {
- _zip_error_set(&za->error, ZIP_ER_SEEK, errno);
- error = 1;
- break;
- }
- if (_zip_dirent_read(&de, za->zp, NULL, 0, 1, &za->error) != 0) {
- error = 1;
- break;
- }
- if (de.bitflags & ZIP_GPBF_DATA_DESCRIPTOR) {
- de.crc = za->cdir->entry[i].crc;
- de.comp_size = za->cdir->entry[i].comp_size;
- de.uncomp_size = za->cdir->entry[i].uncomp_size;
- de.bitflags &= ~ZIP_GPBF_DATA_DESCRIPTOR;
- }
- memcpy(cd->entry+j, za->cdir->entry+i, sizeof(cd->entry[j]));
- }
- if (za->entry[i].ch_filename) {
- free(de.filename);
- if ((de.filename=strdup(za->entry[i].ch_filename)) == NULL) {
- error = 1;
- break;
- }
- de.filename_len = strlen(de.filename);
- cd->entry[j].filename = za->entry[i].ch_filename;
- cd->entry[j].filename_len = de.filename_len;
- }
- if (zip_get_archive_flag(za, ZIP_AFL_TORRENT, 0) == 0
- && za->entry[i].ch_comment_len != -1) {
- /* as the rest of cd entries, its malloc/free is done by za */
- cd->entry[j].comment = za->entry[i].ch_comment;
- cd->entry[j].comment_len = za->entry[i].ch_comment_len;
- }
- cd->entry[j].offset = ftello(out);
- if (ZIP_ENTRY_DATA_CHANGED(za->entry+i) || new_torrentzip) {
- struct zip_source *zs;
- zs = NULL;
- if (!ZIP_ENTRY_DATA_CHANGED(za->entry+i)) {
- if ((zs=zip_source_zip(za, za, i, ZIP_FL_RECOMPRESS, 0, -1))
- == NULL) {
- error = 1;
- break;
- }
- }
- if (add_data(za, zs ? zs : za->entry[i].source, &de, out) < 0) {
- error = 1;
- break;
- }
- cd->entry[j].last_mod = de.last_mod;
- cd->entry[j].comp_method = de.comp_method;
- cd->entry[j].comp_size = de.comp_size;
- cd->entry[j].uncomp_size = de.uncomp_size;
- cd->entry[j].crc = de.crc;
- }
- else {
- if (_zip_dirent_write(&de, out, 1, &za->error) < 0) {
- error = 1;
- break;
- }
- /* we just read the local dirent, file is at correct position */
- if (copy_data(za->zp, cd->entry[j].comp_size, out,
- &za->error) < 0) {
- error = 1;
- break;
- }
- }
- _zip_dirent_finalize(&de);
- }
- if (!error) {
- if (write_cdir(za, cd, out) < 0)
- error = 1;
- }
- /* pointers in cd entries are owned by za */
- cd->nentry = 0;
- _zip_cdir_free(cd);
- if (error) {
- _zip_dirent_finalize(&de);
- fclose(out);
- remove(temp);
- free(temp);
- return -1;
- }
- if (fclose(out) != 0) {
- _zip_error_set(&za->error, ZIP_ER_CLOSE, errno);
- remove(temp);
- free(temp);
- return -1;
- }
- if (za->zp) {
- fclose(za->zp);
- za->zp = NULL;
- reopen_on_error = 1;
- }
- if (_zip_rename(temp, za->zn) != 0) {
- _zip_error_set(&za->error, ZIP_ER_RENAME, errno);
- remove(temp);
- free(temp);
- if (reopen_on_error) {
- /* ignore errors, since we're already in an error case */
- za->zp = fopen(za->zn, "rb");
- }
- return -1;
- }
- mask = umask(0);
- umask(mask);
- chmod(za->zn, 0666&~mask);
- _zip_free(za);
- free(temp);
- return 0;
- }
- static int
- add_data(struct zip *za, struct zip_source *zs, struct zip_dirent *de, FILE *ft)
- {
- myoff_t offstart, offend;
- zip_source_callback cb;
- void *ud;
- struct zip_stat st;
- cb = zs->f;
- ud = zs->ud;
- if (cb(ud, &st, sizeof(st), ZIP_SOURCE_STAT) < (ssize_t)sizeof(st)) {
- ch_set_error(&za->error, cb, ud);
- return -1;
- }
- if (cb(ud, NULL, 0, ZIP_SOURCE_OPEN) < 0) {
- ch_set_error(&za->error, cb, ud);
- return -1;
- }
- offstart = ftello(ft);
- if (_zip_dirent_write(de, ft, 1, &za->error) < 0)
- return -1;
- if (st.comp_method != ZIP_CM_STORE) {
- if (add_data_comp(cb, ud, &st, ft, &za->error) < 0)
- return -1;
- }
- else {
- if (add_data_uncomp(za, cb, ud, &st, ft) < 0)
- return -1;
- }
- if (cb(ud, NULL, 0, ZIP_SOURCE_CLOSE) < 0) {
- ch_set_error(&za->error, cb, ud);
- return -1;
- }
- offend = ftello(ft);
- if (fseeko(ft, offstart, SEEK_SET) < 0) {
- _zip_error_set(&za->error, ZIP_ER_SEEK, errno);
- return -1;
- }
- de->last_mod = st.mtime;
- de->comp_method = st.comp_method;
- de->crc = st.crc;
- de->uncomp_size = st.size;
- de->comp_size = st.comp_size;
- if (zip_get_archive_flag(za, ZIP_AFL_TORRENT, 0))
- _zip_dirent_torrent_normalize(de);
- if (_zip_dirent_write(de, ft, 1, &za->error) < 0)
- return -1;
- if (fseeko(ft, offend, SEEK_SET) < 0) {
- _zip_error_set(&za->error, ZIP_ER_SEEK, errno);
- return -1;
- }
- return 0;
- }
- static int
- add_data_comp(zip_source_callback cb, void *ud, struct zip_stat *st,FILE *ft,
- struct zip_error *error)
- {
- char buf[BUFSIZE];
- ssize_t n;
- st->comp_size = 0;
- while ((n=cb(ud, buf, sizeof(buf), ZIP_SOURCE_READ)) > 0) {
- if (fwrite(buf, 1, n, ft) != (size_t)n) {
- _zip_error_set(error, ZIP_ER_WRITE, errno);
- return -1;
- }
- st->comp_size += n;
- }
- if (n < 0) {
- ch_set_error(error, cb, ud);
- return -1;
- }
- return 0;
- }
- static int
- add_data_uncomp(struct zip *za, zip_source_callback cb, void *ud,
- struct zip_stat *st, FILE *ft)
- {
- char b1[BUFSIZE], b2[BUFSIZE];
- int end, flush, ret;
- ssize_t n;
- size_t n2;
- z_stream zstr;
- int mem_level;
- st->comp_method = ZIP_CM_DEFLATE;
- st->comp_size = st->size = 0;
- st->crc = crc32(0, NULL, 0);
- zstr.zalloc = Z_NULL;
- zstr.zfree = Z_NULL;
- zstr.opaque = NULL;
- zstr.avail_in = 0;
- zstr.avail_out = 0;
- if (zip_get_archive_flag(za, ZIP_AFL_TORRENT, 0))
- mem_level = TORRENT_MEM_LEVEL;
- else
- mem_level = MAX_MEM_LEVEL;
- /* -MAX_WBITS: undocumented feature of zlib to _not_ write a zlib header */
- deflateInit2(&zstr, Z_BEST_COMPRESSION, Z_DEFLATED, -MAX_WBITS, mem_level,
- Z_DEFAULT_STRATEGY);
- zstr.next_out = (Bytef *)b2;
- zstr.avail_out = sizeof(b2);
- zstr.avail_in = 0;
- flush = 0;
- end = 0;
- while (!end) {
- if (zstr.avail_in == 0 && !flush) {
- if ((n=cb(ud, b1, sizeof(b1), ZIP_SOURCE_READ)) < 0) {
- ch_set_error(&za->error, cb, ud);
- deflateEnd(&zstr);
- return -1;
- }
- if (n > 0) {
- zstr.avail_in = n;
- zstr.next_in = (Bytef *)b1;
- st->size += n;
- st->crc = crc32(st->crc, (Bytef *)b1, n);
- }
- else
- flush = Z_FINISH;
- }
- ret = deflate(&zstr, flush);
- if (ret != Z_OK && ret != Z_STREAM_END) {
- _zip_error_set(&za->error, ZIP_ER_ZLIB, ret);
- return -1;
- }
- if (zstr.avail_out != sizeof(b2)) {
- n2 = sizeof(b2) - zstr.avail_out;
- if (fwrite(b2, 1, n2, ft) != n2) {
- _zip_error_set(&za->error, ZIP_ER_WRITE, errno);
- return -1;
- }
- zstr.next_out = (Bytef *)b2;
- zstr.avail_out = sizeof(b2);
- st->comp_size += n2;
- }
- if (ret == Z_STREAM_END) {
- deflateEnd(&zstr);
- end = 1;
- }
- }
- return 0;
- }
- static void
- ch_set_error(struct zip_error *error, zip_source_callback cb, void *ud)
- {
- int e[2];
- if ((cb(ud, e, sizeof(e), ZIP_SOURCE_ERROR)) < (ssize_t)sizeof(e)) {
- error->zip_err = ZIP_ER_INTERNAL;
- error->sys_err = 0;
- }
- else {
- error->zip_err = e[0];
- error->sys_err = e[1];
- }
- }
- static int
- copy_data(FILE *fs, myoff_t len, FILE *ft, struct zip_error *error)
- {
- char buf[BUFSIZE];
- int n, nn;
- if (len == 0)
- return 0;
- while (len > 0) {
- nn = len > sizeof(buf) ? sizeof(buf) : len;
- if ((n=fread(buf, 1, nn, fs)) < 0) {
- _zip_error_set(error, ZIP_ER_READ, errno);
- return -1;
- }
- else if (n == 0) {
- _zip_error_set(error, ZIP_ER_EOF, 0);
- return -1;
- }
- if (fwrite(buf, 1, n, ft) != (size_t)n) {
- _zip_error_set(error, ZIP_ER_WRITE, errno);
- return -1;
- }
- len -= n;
- }
- return 0;
- }
- static int
- write_cdir(struct zip *za, struct zip_cdir *cd, FILE *out)
- {
- myoff_t offset;
- uLong crc;
- char buf[TORRENT_CRC_LEN+1];
- if (_zip_cdir_write(cd, out, &za->error) < 0)
- return -1;
- if (zip_get_archive_flag(za, ZIP_AFL_TORRENT, 0) == 0)
- return 0;
- /* fix up torrentzip comment */
- offset = ftello(out);
- if (_zip_filerange_crc(out, cd->offset, cd->size, &crc, &za->error) < 0)
- return -1;
- snprintf(buf, sizeof(buf), "%08lX", (long)crc);
- if (fseeko(out, offset-TORRENT_CRC_LEN, SEEK_SET) < 0) {
- _zip_error_set(&za->error, ZIP_ER_SEEK, errno);
- return -1;
- }
- if (fwrite(buf, TORRENT_CRC_LEN, 1, out) != 1) {
- _zip_error_set(&za->error, ZIP_ER_WRITE, errno);
- return -1;
- }
- return 0;
- }
- static int
- _zip_cdir_set_comment(struct zip_cdir *dest, struct zip *src)
- {
- if (src->ch_comment_len != -1) {
- dest->comment = _zip_memdup(src->ch_comment,
- src->ch_comment_len, &src->error);
- if (dest->comment == NULL)
- return -1;
- dest->comment_len = src->ch_comment_len;
- } else {
- if (src->cdir && src->cdir->comment) {
- dest->comment = _zip_memdup(src->cdir->comment,
- src->cdir->comment_len, &src->error);
- if (dest->comment == NULL)
- return -1;
- dest->comment_len = src->cdir->comment_len;
- }
- }
- return 0;
- }
- static int
- _zip_changed(struct zip *za, int *survivorsp)
- {
- int changed, i, survivors;
- changed = survivors = 0;
- if (za->ch_comment_len != -1
- || za->ch_flags != za->flags)
- changed = 1;
- for (i=0; i<za->nentry; i++) {
- if ((za->entry[i].state != ZIP_ST_UNCHANGED)
- || (za->entry[i].ch_comment_len != -1))
- changed = 1;
- if (za->entry[i].state != ZIP_ST_DELETED)
- survivors++;
- }
- *survivorsp = survivors;
- return changed;
- }
- static char *
- _zip_create_temp_output(struct zip *za, FILE **outp)
- {
- char *temp;
- int tfd;
- FILE *tfp;
- if ((temp=(char *)malloc(strlen(za->zn)+8)) == NULL) {
- _zip_error_set(&za->error, ZIP_ER_MEMORY, 0);
- return NULL;
- }
- sprintf(temp, "%s.XXXXXX", za->zn);
- if ((tfd=mkstemp(temp)) == -1) {
- _zip_error_set(&za->error, ZIP_ER_TMPOPEN, errno);
- free(temp);
- return NULL;
- }
- if ((tfp=fdopen(tfd, "r+b")) == NULL) {
- _zip_error_set(&za->error, ZIP_ER_TMPOPEN, errno);
- close(tfd);
- remove(temp);
- free(temp);
- return NULL;
- }
- *outp = tfp;
- return temp;
- }
- static int
- _zip_torrentzip_cmp(const void *a, const void *b)
- {
- return strcasecmp(((const struct filelist *)a)->name,
- ((const struct filelist *)b)->name);
- }
- ZIP_EXTERN int
- zip_add_dir(struct zip *za, const char *name)
- {
- int len, ret;
- char *s;
- struct zip_source *source;
- if (name == NULL) {
- _zip_error_set(&za->error, ZIP_ER_INVAL, 0);
- return -1;
- }
- s = NULL;
- len = strlen(name);
- if (name[len-1] != '/') {
- if ((s=(char *)malloc(len+2)) == NULL) {
- _zip_error_set(&za->error, ZIP_ER_MEMORY, 0);
- return -1;
- }
- strcpy(s, name);
- s[len] = '/';
- s[len+1] = '\0';
- }
- if ((source=zip_source_buffer(za, NULL, 0, 0)) == NULL) {
- free(s);
- return -1;
- }
- ret = _zip_replace(za, -1, s ? s : name, source);
- free(s);
- if (ret < 0)
- zip_source_free(source);
- return ret;
- }
- ZIP_EXTERN int
- zip_error_to_str(char *buf, size_t len, int ze, int se)
- {
- const char *zs, *ss;
- if (ze < 0 || ze >= _zip_nerr_str)
- return snprintf(buf, len, "Unknown error %d", ze);
- zs = _zip_err_str[ze];
- switch (_zip_err_type[ze]) {
- case ZIP_ET_SYS:
- ss = strerror(se);
- break;
- case ZIP_ET_ZLIB:
- ss = zError(se);
- break;
- default:
- ss = NULL;
- }
- return snprintf(buf, len, "%s%s%s",
- zs, (ss ? ": " : ""), (ss ? ss : ""));
- }
- ZIP_EXTERN void
- zip_file_error_clear(struct zip_file *zf)
- {
- _zip_error_clear(&zf->error);
- }
- ZIP_EXTERN int
- zip_fclose(struct zip_file *zf)
- {
- int i, ret;
- if (zf->zstr)
- inflateEnd(zf->zstr);
- free(zf->buffer);
- free(zf->zstr);
- for (i=0; i<zf->za->nfile; i++) {
- if (zf->za->file[i] == zf) {
- zf->za->file[i] = zf->za->file[zf->za->nfile-1];
- zf->za->nfile--;
- break;
- }
- }
- ret = 0;
- if (zf->error.zip_err)
- ret = zf->error.zip_err;
- else if ((zf->flags & ZIP_ZF_CRC) && (zf->flags & ZIP_ZF_EOF)) {
- /* if EOF, compare CRC */
- if (zf->crc_orig != zf->crc)
- ret = ZIP_ER_CRC;
- }
- free(zf);
- return ret;
- }
- int
- _zip_filerange_crc(FILE *fp, myoff_t start, myoff_t len, uLong *crcp,
- struct zip_error *errp)
- {
- Bytef buf[BUFSIZE];
- size_t n;
- *crcp = crc32(0L, Z_NULL, 0);
- if (fseeko(fp, start, SEEK_SET) != 0) {
- _zip_error_set(errp, ZIP_ER_SEEK, errno);
- return -1;
- }
- while (len > 0) {
- n = len > BUFSIZE ? BUFSIZE : len;
- if ((n=fread(buf, 1, n, fp)) <= 0) {
- _zip_error_set(errp, ZIP_ER_READ, errno);
- return -1;
- }
- *crcp = crc32(*crcp, buf, n);
- len-= n;
- }
- return 0;
- }
- ZIP_EXTERN const char *
- zip_file_strerror(struct zip_file *zf)
- {
- return _zip_error_strerror(&zf->error);
- }
- /* _zip_file_get_offset(za, ze):
- Returns the offset of the file data for entry ze.
- On error, fills in za->error and returns 0.
- */
- unsigned int
- _zip_file_get_offset(struct zip *za, int idx)
- {
- struct zip_dirent de;
- unsigned int offset;
- offset = za->cdir->entry[idx].offset;
- if (fseeko(za->zp, offset, SEEK_SET) != 0) {
- _zip_error_set(&za->error, ZIP_ER_SEEK, errno);
- return 0;
- }
- if (_zip_dirent_read(&de, za->zp, NULL, 0, 1, &za->error) != 0)
- return 0;
- offset += LENTRYSIZE + de.filename_len + de.extrafield_len;
- _zip_dirent_finalize(&de);
- return offset;
- }
- ZIP_EXTERN void
- zip_file_error_get(struct zip_file *zf, int *zep, int *sep)
- {
- _zip_error_get(&zf->error, zep, sep);
- }
- static struct zip_file *_zip_file_new(struct zip *za);
- ZIP_EXTERN struct zip_file *
- zip_fopen_index(struct zip *za, int fileno, int flags)
- {
- int len, ret;
- int zfflags;
- struct zip_file *zf;
- if ((fileno < 0) || (fileno >= za->nentry)) {
- _zip_error_set(&za->error, ZIP_ER_INVAL, 0);
- return NULL;
- }
- if ((flags & ZIP_FL_UNCHANGED) == 0
- && ZIP_ENTRY_DATA_CHANGED(za->entry+fileno)) {
- _zip_error_set(&za->error, ZIP_ER_CHANGED, 0);
- return NULL;
- }
- if (fileno >= za->cdir->nentry) {
- _zip_error_set(&za->error, ZIP_ER_INVAL, 0);
- return NULL;
- }
- zfflags = 0;
- switch (za->cdir->entry[fileno].comp_method) {
- case ZIP_CM_STORE:
- zfflags |= ZIP_ZF_CRC;
- break;
- case ZIP_CM_DEFLATE:
- if ((flags & ZIP_FL_COMPRESSED) == 0)
- zfflags |= ZIP_ZF_CRC | ZIP_ZF_DECOMP;
- break;
- default:
- if ((flags & ZIP_FL_COMPRESSED) == 0) {
- _zip_error_set(&za->error, ZIP_ER_COMPNOTSUPP, 0);
- return NULL;
- }
- break;
- }
- zf = _zip_file_new(za);
- zf->flags = zfflags;
- /* zf->name = za->cdir->entry[fileno].filename; */
- zf->method = za->cdir->entry[fileno].comp_method;
- zf->bytes_left = za->cdir->entry[fileno].uncomp_size;
- zf->cbytes_left = za->cdir->entry[fileno].comp_size;
- zf->crc_orig = za->cdir->entry[fileno].crc;
- if ((zf->fpos=_zip_file_get_offset(za, fileno)) == 0) {
- zip_fclose(zf);
- return NULL;
- }
- if ((zf->flags & ZIP_ZF_DECOMP) == 0)
- zf->bytes_left = zf->cbytes_left;
- else {
- if ((zf->buffer=(char *)malloc(BUFSIZE)) == NULL) {
- _zip_error_set(&za->error, ZIP_ER_MEMORY, 0);
- zip_fclose(zf);
- return NULL;
- }
- len = _zip_file_fillbuf(zf->buffer, BUFSIZE, zf);
- if (len <= 0) {
- _zip_error_copy(&za->error, &zf->error);
- zip_fclose(zf);
- return NULL;
- }
- if ((zf->zstr = (z_stream *)malloc(sizeof(z_stream))) == NULL) {
- _zip_error_set(&za->error, ZIP_ER_MEMORY, 0);
- zip_fclose(zf);
- return NULL;
- }
- zf->zstr->zalloc = Z_NULL;
- zf->zstr->zfree = Z_NULL;
- zf->zstr->opaque = NULL;
- zf->zstr->next_in = (Bytef *)zf->buffer;
- zf->zstr->avail_in = len;
- /* negative value to tell zlib that there is no header */
- if ((ret=inflateInit2(zf->zstr, -MAX_WBITS)) != Z_OK) {
- _zip_error_set(&za->error, ZIP_ER_ZLIB, ret);
- zip_fclose(zf);
- return NULL;
- }
- }
- return zf;
- }
- int
- _zip_file_fillbuf(void *buf, size_t buflen, struct zip_file *zf)
- {
- int i, j;
- if (zf->error.zip_err != ZIP_ER_OK)
- return -1;
- if ((zf->flags & ZIP_ZF_EOF) || zf->cbytes_left <= 0 || buflen <= 0)
- return 0;
- if (fseeko(zf->za->zp, zf->fpos, SEEK_SET) < 0) {
- _zip_error_set(&zf->error, ZIP_ER_SEEK, errno);
- return -1;
- }
- if (buflen < zf->cbytes_left)
- i = buflen;
- else
- i = zf->cbytes_left;
- j = fread(buf, 1, i, zf->za->zp);
- if (j == 0) {
- _zip_error_set(&zf->error, ZIP_ER_EOF, 0);
- j = -1;
- }
- else if (j < 0)
- _zip_error_set(&zf->error, ZIP_ER_READ, errno);
- else {
- zf->fpos += j;
- zf->cbytes_left -= j;
- }
- return j;
- }
- static struct zip_file *
- _zip_file_new(struct zip *za)
- {
- struct zip_file *zf, **file;
- int n;
- if ((zf=(struct zip_file *)malloc(sizeof(struct zip_file))) == NULL) {
- _zip_error_set(&za->error, ZIP_ER_MEMORY, 0);
- return NULL;
- }
- if (za->nfile >= za->nfile_alloc-1) {
- n = za->nfile_alloc + 10;
- file = (struct zip_file **)realloc(za->file,
- n*sizeof(struct zip_file *));
- if (file == NULL) {
- _zip_error_set(&za->error, ZIP_ER_MEMORY, 0);
- free(zf);
- return NULL;
- }
- za->nfile_alloc = n;
- za->file = file;
- }
- za->file[za->nfile++] = zf;
- zf->za = za;
- _zip_error_init(&zf->error);
- zf->flags = 0;
- zf->crc = crc32(0L, Z_NULL, 0);
- zf->crc_orig = 0;
- zf->method = -1;
- zf->bytes_left = zf->cbytes_left = 0;
- zf->fpos = 0;
- zf->buffer = NULL;
- zf->zstr = NULL;
- return zf;
- }
- ZIP_EXTERN struct zip_file *
- zip_fopen(struct zip *za, const char *fname, int flags)
- {
- int idx;
- if ((idx=zip_name_locate(za, fname, flags)) < 0)
- return NULL;
- return zip_fopen_index(za, idx, flags);
- }
- ZIP_EXTERN int
- zip_set_file_comment(struct zip *za, int idx, const char *comment, int len)
- {
- char *tmpcom;
- if (idx < 0 || idx >= za->nentry
- || len < 0 || len > MAXCOMLEN
- || (len > 0 && comment == NULL)) {
- _zip_error_set(&za->error, ZIP_ER_INVAL, 0);
- return -1;
- }
- if (len > 0) {
- if ((tmpcom=(char *)_zip_memdup(comment, len, &za->error)) == NULL)
- return -1;
- }
- else
- tmpcom = NULL;
- free(za->entry[idx].ch_comment);
- za->entry[idx].ch_comment = tmpcom;
- za->entry[idx].ch_comment_len = len;
- return 0;
- }
- ZIP_EXTERN struct zip_source *
- zip_source_file(struct zip *za, const char *fname, myoff_t start, myoff_t len)
- {
- if (za == NULL)
- return NULL;
- if (fname == NULL || start < 0 || len < -1) {
- _zip_error_set(&za->error, ZIP_ER_INVAL, 0);
- return NULL;
- }
- return _zip_source_file_or_p(za, fname, NULL, start, len);
- }
- struct read_data {
- const char *buf, *data, *end;
- time_t mtime;
- int freep;
- };
- static ssize_t read_data(void *state, void *data, size_t len,
- enum zip_source_cmd cmd);
- ZIP_EXTERN struct zip_source *
- zip_source_buffer(struct zip *za, const void *data, myoff_t len, int freep)
- {
- struct read_data *f;
- struct zip_source *zs;
- if (za == NULL)
- return NULL;
- if (len < 0 || (data == NULL && len > 0)) {
- _zip_error_set(&za->error, ZIP_ER_INVAL, 0);
- return NULL;
- }
- if ((f=(struct read_data *)malloc(sizeof(*f))) == NULL) {
- _zip_error_set(&za->error, ZIP_ER_MEMORY, 0);
- return NULL;
- }
- f->data = (const char *)data;
- f->end = ((const char *)data)+len;
- f->freep = freep;
- f->mtime = time(NULL);
- if ((zs=zip_source_function(za, read_data, f)) == NULL) {
- free(f);
- return NULL;
- }
- return zs;
- }
- static ssize_t
- read_data(void *state, void *data, size_t len, enum zip_source_cmd cmd)
- {
- struct read_data *z;
- char *buf;
- size_t n;
- z = (struct read_data *)state;
- buf = (char *)data;
- switch (cmd) {
- case ZIP_SOURCE_OPEN:
- z->buf = z->data;
- return 0;
- case ZIP_SOURCE_READ:
- n = z->end - z->buf;
- if (n > len)
- n = len;
- if (n) {
- memcpy(buf, z->buf, n);
- z->buf += n;
- }
- return n;
- case ZIP_SOURCE_CLOSE:
- return 0;
- case ZIP_SOURCE_STAT:
- {
- struct zip_stat *st;
- if (len < sizeof(*st))
- return -1;
- st = (struct zip_stat *)data;
- zip_stat_init(st);
- st->mtime = z->mtime;
- st->size = z->end - z->data;
- return sizeof(*st);
- }
- case ZIP_SOURCE_ERROR:
- {
- int *e;
- if (len < sizeof(int)*2)
- return -1;
- e = (int *)data;
- e[0] = e[1] = 0;
- }
- return sizeof(int)*2;
- case ZIP_SOURCE_FREE:
- if (z->freep) {
- free((void *)z->data);
- z->data = NULL;
- }
- free(z);
- return 0;
- default:
- ;
- }
- return -1;
- }
- int
- _zip_set_name(struct zip *za, int idx, const char *name)
- {
- char *s;
- int i;
- if (idx < 0 || idx >= za->nentry || name == NULL) {
- _zip_error_set(&za->error, ZIP_ER_INVAL, 0);
- return -1;
- }
- if ((i=_zip_name_locate(za, name, 0, NULL)) != -1 && i != idx) {
- _zip_error_set(&za->error, ZIP_ER_EXISTS, 0);
- return -1;
- }
- /* no effective name change */
- if (i == idx)
- return 0;
- if ((s=strdup(name)) == NULL) {
- _zip_error_set(&za->error, ZIP_ER_MEMORY, 0);
- return -1;
- }
- if (za->entry[idx].state == ZIP_ST_UNCHANGED)
- za->entry[idx].state = ZIP_ST_RENAMED;
- free(za->entry[idx].ch_filename);
- za->entry[idx].ch_filename = s;
- return 0;
- }
- ZIP_EXTERN int
- zip_set_archive_flag(struct zip *za, int flag, int value)
- {
- if (value)
- za->ch_flags |= flag;
- else
- za->ch_flags &= ~flag;
- return 0;
- }
- void
- _zip_unchange_data(struct zip_entry *ze)
- {
- if (ze->source) {
- (void)ze->source->f(ze->source->ud, NULL, 0, ZIP_SOURCE_FREE);
- free(ze->source);
- ze->source = NULL;
- }
- ze->state = ze->ch_filename ? ZIP_ST_RENAMED : ZIP_ST_UNCHANGED;
- }
- ZIP_EXTERN int
- zip_unchange_archive(struct zip *za)
- {
- free(za->ch_comment);
- za->ch_comment = NULL;
- za->ch_comment_len = -1;
- za->ch_flags = za->flags;
- return 0;
- }
- ZIP_EXTERN int
- zip_unchange(struct zip *za, int idx)
- {
- return _zip_unchange(za, idx, 0);
- }
- int
- _zip_unchange(struct zip *za, int idx, int allow_duplicates)
- {
- int i;
- if (idx < 0 || idx >= za->nentry) {
- _zip_error_set(&za->error, ZIP_ER_INVAL, 0);
- return -1;
- }
- if (za->entry[idx].ch_filename) {
- if (!allow_duplicates) {
- i = _zip_name_locate(za,
- _zip_get_name(za, idx, ZIP_FL_UNCHANGED, NULL),
- 0, NULL);
- if (i != -1 && i != idx) {
- _zip_error_set(&za->error, ZIP_ER_EXISTS, 0);
- return -1;
- }
- }
- free(za->entry[idx].ch_filename);
- za->entry[idx].ch_filename = NULL;
- }
- free(za->entry[idx].ch_comment);
- za->entry[idx].ch_comment = NULL;
- za->entry[idx].ch_comment_len = -1;
- _zip_unchange_data(za->entry+idx);
- return 0;
- }
- ZIP_EXTERN int
- zip_unchange_all(struct zip *za)
- {
- int ret, i;
- ret = 0;
- for (i=0; i<za->nentry; i++)
- ret |= _zip_unchange(za, i, 1);
- ret |= zip_unchange_archive(za);
- return ret;
- }
- ZIP_EXTERN int
- zip_set_archive_comment(struct zip *za, const char *comment, int len)
- {
- char *tmpcom;
- if (len < 0 || len > MAXCOMLEN
- || (len > 0 && comment == NULL)) {
- _zip_error_set(&za->error, ZIP_ER_INVAL, 0);
- return -1;
- }
- if (len > 0) {
- if ((tmpcom=(char *)_zip_memdup(comment, len, &za->error)) == NULL)
- return -1;
- }
- else
- tmpcom = NULL;
- free(za->ch_comment);
- za->ch_comment = tmpcom;
- za->ch_comment_len = len;
- return 0;
- }
- ZIP_EXTERN int
- zip_replace(struct zip *za, int idx, struct zip_source *source)
- {
- if (idx < 0 || idx >= za->nentry || source == NULL) {
- _zip_error_set(&za->error, ZIP_ER_INVAL, 0);
- return -1;
- }
- if (_zip_replace(za, idx, NULL, source) == -1)
- return -1;
- return 0;
- }
- int
- _zip_replace(struct zip *za, int idx, const char *name,
- struct zip_source *source)
- {
- if (idx == -1) {
- if (_zip_entry_new(za) == NULL)
- return -1;
- idx = za->nentry - 1;
- }
- _zip_unchange_data(za->entry+idx);
- if (name && _zip_set_name(za, idx, name) != 0)
- return -1;
- za->entry[idx].state = ((za->cdir == NULL || idx >= za->cdir->nentry)
- ? ZIP_ST_ADDED : ZIP_ST_REPLACED);
- za->entry[idx].source = source;
- return idx;
- }
- ZIP_EXTERN int
- zip_rename(struct zip *za, int idx, const char *name)
- {
- const char *old_name;
- int old_is_dir, new_is_dir;
- if (idx >= za->nentry || idx < 0 || name[0] == '\0') {
- _zip_error_set(&za->error, ZIP_ER_INVAL, 0);
- return -1;
- }
- if ((old_name=zip_get_name(za, idx, 0)) == NULL)
- return -1;
- new_is_dir = (name[strlen(name)-1] == '/');
- old_is_dir = (old_name[strlen(old_name)-1] == '/');
- if (new_is_dir != old_is_dir) {
- _zip_error_set(&za->error, ZIP_ER_INVAL, 0);
- return -1;
- }
- return _zip_set_name(za, idx, name);
- }
- #include <sys/stat.h>
- #include <errno.h>
- #include <limits.h>
- #include <stdio.h>
- #include <stdlib.h>
- #include <string.h>
- static void set_error(int *, struct zip_error *, int);
- static struct zip *_zip_allocate_new(const char *, int *);
- static int _zip_checkcons(FILE *, struct zip_cdir *, struct zip_error *);
- static void _zip_check_torrentzip(struct zip *);
- static struct zip_cdir *_zip_find_central_dir(FILE *, int, int *, myoff_t);
- static int _zip_file_exists(const char *, int, int *);
- static int _zip_headercomp(struct zip_dirent *, int,
- struct zip_dirent *, int);
- static unsigned char *_zip_memmem(const unsigned char *, int,
- const unsigned char *, int);
- static struct zip_cdir *_zip_readcdir(FILE *, unsigned char *, unsigned char *,
- int, int, struct zip_error *);
- ZIP_EXTERN struct zip *
- zip_open(const char *fn, int flags, int *zep)
- {
- FILE *fp;
- struct zip *za;
- struct zip_cdir *cdir;
- int i;
- myoff_t len;
- switch (_zip_file_exists(fn, flags, zep)) {
- case -1:
- return NULL;
- case 0:
- return _zip_allocate_new(fn, zep);
- default:
- break;
- }
- if ((fp=fopen(fn, "rb")) == NULL) {
- set_error(zep, NULL, ZIP_ER_OPEN);
- return NULL;
- }
- fseeko(fp, 0, SEEK_END);
- len = ftello(fp);
- /* treat empty files as empty archives */
- if (len == 0) {
- if ((za=_zip_allocate_new(fn, zep)) == NULL)
- fclose(fp);
- else
- za->zp = fp;
- return za;
- }
- cdir = _zip_find_central_dir(fp, flags, zep, len);
- if (cdir == NULL) {
- fclose(fp);
- return NULL;
- }
- if ((za=_zip_allocate_new(fn, zep)) == NULL) {
- _zip_cdir_free(cdir);
- fclose(fp);
- return NULL;
- }
- za->cdir = cdir;
- za->zp = fp;
- if ((za->entry=(struct zip_entry *)malloc(sizeof(*(za->entry))
- * cdir->nentry)) == NULL) {
- set_error(zep, NULL, ZIP_ER_MEMORY);
- _zip_free(za);
- return NULL;
- }
- for (i=0; i<cdir->nentry; i++)
- _zip_entry_new(za);
- _zip_check_torrentzip(za);
- za->ch_flags = za->flags;
- return za;
- }
- static void
- set_error(int *zep, struct zip_error *err, int ze)
- {
- int se;
- if (err) {
- _zip_error_get(err, &ze, &se);
- if (zip_error_get_sys_type(ze) == ZIP_ET_SYS)
- errno = se;
- }
- if (zep)
- *zep = ze;
- }
- /* _zip_readcdir:
- tries to find a valid end-of-central-directory at the beginning of
- buf, and then the corresponding central directory entries.
- Returns a struct zip_cdir which contains the central directory
- entries, or NULL if unsuccessful. */
- static struct zip_cdir *
- _zip_readcdir(FILE *fp, unsigned char *buf, unsigned char *eocd, int buflen,
- int flags, struct zip_error *error)
- {
- struct zip_cdir *cd;
- unsigned char *cdp, **bufp;
- int i, comlen, nentry;
- comlen = buf + buflen - eocd - EOCDLEN;
- if (comlen < 0) {
- /* not enough bytes left for comment */
- _zip_error_set(error, ZIP_ER_NOZIP, 0);
- return NULL;
- }
- /* check for end-of-central-dir magic */
- if (memcmp(eocd, EOCD_MAGIC, 4) != 0) {
- _zip_error_set(error, ZIP_ER_NOZIP, 0);
- return NULL;
- }
- if (memcmp(eocd+4, "\0\0\0\0", 4) != 0) {
- _zip_error_set(error, ZIP_ER_MULTIDISK, 0);
- return NULL;
- }
- cdp = eocd + 8;
- /* number of cdir-entries on this disk */
- i = _zip_read2(&cdp);
- /* number of cdir-entries */
- nentry = _zip_read2(&cdp);
- if ((cd=_zip_cdir_new(nentry, error)) == NULL)
- return NULL;
- cd->size = _zip_read4(&cdp);
- cd->offset = _zip_read4(&cdp);
- cd->comment = NULL;
- cd->comment_len = _zip_read2(&cdp);
- if ((comlen < cd->comment_len) || (cd->nentry != i)) {
- _zip_error_set(error, ZIP_ER_NOZIP, 0);
- free(cd);
- return NULL;
- }
- if ((flags & ZIP_CHECKCONS) && comlen != cd->comment_len) {
- _zip_error_set(error, ZIP_ER_INCONS, 0);
- free(cd);
- return NULL;
- }
- if (cd->comment_len) {
- if ((cd->comment=(char *)_zip_memdup(eocd+EOCDLEN,
- cd->comment_len, error))
- == NULL) {
- free(cd);
- return NULL;
- }
- }
- cdp = eocd;
- if (cd->size < (unsigned int)(eocd-buf)) {
- /* if buffer already read in, use it */
- cdp = eocd - cd->size;
- bufp = &cdp;
- }
- else {
- /* go to start of cdir and read it entry by entry */
- bufp = NULL;
- clearerr(fp);
- fseeko(fp, cd->offset, SEEK_SET);
- /* possible consistency check: cd->offset =
- len-(cd->size+cd->comment_len+EOCDLEN) ? */
- if (ferror(fp) || ((unsigned long)ftello(fp) != cd->offset)) {
- /* seek error or offset of cdir wrong */
- if (ferror(fp))
- _zip_error_set(error, ZIP_ER_SEEK, errno);
- else
- _zip_error_set(error, ZIP_ER_NOZIP, 0);
- free(cd);
- return NULL;
- }
- }
- for (i=0; i<cd->nentry; i++) {
- if ((_zip_dirent_read(cd->entry+i, fp, bufp, eocd-cdp, 0,
- error)) < 0) {
- cd->nentry = i;
- _zip_cdir_free(cd);
- return NULL;
- }
- }
- return cd;
- }
- /* _zip_checkcons:
- Checks the consistency of the central directory by comparing central
- directory entries with local headers and checking for plausible
- file and header offsets. Returns -1 if not plausible, else the
- difference between the lowest and the highest fileposition reached */
- static int
- _zip_checkcons(FILE *fp, struct zip_cdir *cd, struct zip_error *error)
- {
- int i;
- unsigned int min, max, j;
- struct zip_dirent temp;
- if (cd->nentry) {
- max = cd->entry[0].offset;
- min = cd->entry[0].offset;
- }
- else
- min = max = 0;
- for (i=0; i<cd->nentry; i++) {
- if (cd->entry[i].offset < min)
- min = cd->entry[i].offset;
- if (min > cd->offset) {
- _zip_error_set(error, ZIP_ER_NOZIP, 0);
- return -1;
- }
- j = cd->entry[i].offset + cd->entry[i].comp_size
- + cd->entry[i].filename_len + LENTRYSIZE;
- if (j > max)
- max = j;
- if (max > cd->offset) {
- _zip_error_set(error, ZIP_ER_NOZIP, 0);
- return -1;
- }
- if (fseeko(fp, cd->entry[i].offset, SEEK_SET) != 0) {
- _zip_error_set(error, ZIP_ER_SEEK, 0);
- return -1;
- }
- if (_zip_dirent_read(&temp, fp, NULL, 0, 1, error) == -1)
- return -1;
- if (_zip_headercomp(cd->entry+i, 0, &temp, 1) != 0) {
- _zip_error_set(error, ZIP_ER_INCONS, 0);
- _zip_dirent_finalize(&temp);
- return -1;
- }
- _zip_dirent_finalize(&temp);
- }
- return max - min;
- }
- /* _zip_check_torrentzip:
- check wether ZA has a valid TORRENTZIP comment, i.e. is torrentzipped */
- static void
- _zip_check_torrentzip(struct zip *za)
- {
- uLong crc_got, crc_should;
- char buf[8+1];
- char *end;
- if (za->zp == NULL || za->cdir == NULL)
- return;
- if (za->cdir->comment_len != TORRENT_SIG_LEN+8
- || strncmp(za->cdir->comment, TORRENT_SIG, TORRENT_SIG_LEN) != 0)
- return;
- memcpy(buf, za->cdir->comment+TORRENT_SIG_LEN, 8);
- buf[8] = '\0';
- errno = 0;
- crc_should = strtoul(buf, &end, 16);
- if ((crc_should == UINT_MAX && errno != 0) || (end && *end))
- return;
- if (_zip_filerange_crc(za->zp, za->cdir->offset, za->cdir->size,
- &crc_got, NULL) < 0)
- return;
- if (crc_got == crc_should)
- za->flags |= ZIP_AFL_TORRENT;
- }
- /* _zip_headercomp:
- compares two headers h1 and h2; if they are local headers, set
- local1p or local2p respectively to 1, else 0. Return 0 if they
- are identical, -1 if not. */
- static int
- _zip_headercomp(struct zip_dirent *h1, int local1p, struct zip_dirent *h2,
- int local2p)
- {
- if ((h1->version_needed != h2->version_needed)
- #if 0
- /* some zip-files have different values in local
- and global headers for the bitflags */
- || (h1->bitflags != h2->bitflags)
- #endif
- || (h1->comp_method != h2->comp_method)
- || (h1->last_mod != h2->last_mod)
- || (h1->filename_len != h2->filename_len)
- || !h1->filename || !h2->filename
- || strcmp(h1->filename, h2->filename))
- return -1;
- /* check that CRC and sizes are zero if data descriptor is used */
- if ((h1->bitflags & ZIP_GPBF_DATA_DESCRIPTOR) && local1p
- && (h1->crc != 0
- || h1->comp_size != 0
- || h1->uncomp_size != 0))
- return -1;
- if ((h2->bitflags & ZIP_GPBF_DATA_DESCRIPTOR) && local2p
- && (h2->crc != 0
- || h2->comp_size != 0
- || h2->uncomp_size != 0))
- return -1;
- /* check that CRC and sizes are equal if no data descriptor is used */
- if (((h1->bitflags & ZIP_GPBF_DATA_DESCRIPTOR) == 0 || local1p == 0)
- && ((h2->bitflags & ZIP_GPBF_DATA_DESCRIPTOR) == 0 || local2p == 0)) {
- if ((h1->crc != h2->crc)
- || (h1->comp_size != h2->comp_size)
- || (h1->uncomp_size != h2->uncomp_size))
- return -1;
- }
- if ((local1p == local2p)
- && ((h1->extrafield_len != h2->extrafield_len)
- || (h1->extrafield_len && h2->extrafield
- && memcmp(h1->extrafield, h2->extrafield,
- h1->extrafield_len))))
- return -1;
- /* if either is local, nothing more to check */
- if (local1p || local2p)
- return 0;
- if ((h1->version_madeby != h2->version_madeby)
- || (h1->disk_number != h2->disk_number)
- || (h1->int_attrib != h2->int_attrib)
- || (h1->ext_attrib != h2->ext_attrib)
- || (h1->offset != h2->offset)
- || (h1->comment_len != h2->comment_len)
- || (h1->comment_len && h2->comment
- && memcmp(h1->comment, h2->comment, h1->comment_len)))
- return -1;
- return 0;
- }
- static struct zip *
- _zip_allocate_new(const char *fn, int *zep)
- {
- struct zip *za;
- struct zip_error error;
- if ((za=_zip_new(&error)) == NULL) {
- set_error(zep, &error, 0);
- return NULL;
- }
- za->zn = strdup(fn);
- if (!za->zn) {
- _zip_free(za);
- set_error(zep, NULL, ZIP_ER_MEMORY);
- return NULL;
- }
- return za;
- }
- static int
- _zip_file_exists(const char *fn, int flags, int *zep)
- {
- struct stat st;
- if (fn == NULL) {
- set_error(zep, NULL, ZIP_ER_INVAL);
- return -1;
- }
- if (stat(fn, &st) != 0) {
- if (flags & ZIP_CREATE)
- return 0;
- else {
- set_error(zep, NULL, ZIP_ER_OPEN);
- return -1;
- }
- }
- else if ((flags & ZIP_EXCL)) {
- set_error(zep, NULL, ZIP_ER_EXISTS);
- return -1;
- }
- /* ZIP_CREATE gets ignored if file exists and not ZIP_EXCL,
- just like open() */
- return 1;
- }
- static struct zip_cdir *
- _zip_find_central_dir(FILE *fp, int flags, int *zep, myoff_t len)
- {
- struct zip_cdir *cdir, *cdirnew;
- unsigned char *buf, *match;
- int a, best, buflen, i;
- struct zip_error zerr;
- i = fseeko(fp, -(len < CDBUFSIZE ? len : CDBUFSIZE), SEEK_END);
- if (i == -1 && errno != EFBIG) {
- /* seek before start of file on my machine */
- set_error(zep, NULL, ZIP_ER_SEEK);
- return NULL;
- }
- /* 64k is too much for stack */
- if ((buf=(unsigned char *)malloc(CDBUFSIZE)) == NULL) {
- set_error(zep, NULL, ZIP_ER_MEMORY);
- return NULL;
- }
- clearerr(fp);
- buflen = fread(buf, 1, CDBUFSIZE, fp);
- if (ferror(fp)) {
- set_error(zep, NULL, ZIP_ER_READ);
- free(buf);
- return NULL;
- }
- best = -1;
- cdir = NULL;
- match = buf;
- _zip_error_set(&zerr, ZIP_ER_NOZIP, 0);
- while ((match=_zip_memmem(match, buflen-(match-buf)-18,
- (const unsigned char *)EOCD_MAGIC, 4))!=NULL) {
- /* found match -- check, if good */
- /* to avoid finding the same match all over again */
- match++;
- if ((cdirnew=_zip_readcdir(fp, buf, match-1, buflen, flags,
- &zerr)) == NULL)
- continue;
- if (cdir) {
- if (best <= 0)
- best = _zip_checkcons(fp, cdir, &zerr);
- a = _zip_checkcons(fp, cdirnew, &zerr);
- if (best < a) {
- _zip_cdir_free(cdir);
- cdir = cdirnew;
- best = a;
- }
- else
- _zip_cdir_free(cdirnew);
- }
- else {
- cdir = cdirnew;
- if (flags & ZIP_CHECKCONS)
- best = _zip_checkcons(fp, cdir, &zerr);
- else
- best = 0;
- }
- cdirnew = NULL;
- }
- free(buf);
- if (best < 0) {
- set_error(zep, &zerr, 0);
- _zip_cdir_free(cdir);
- return NULL;
- }
- return cdir;
- }
- static unsigned char *
- _zip_memmem(const unsigned char *big, int biglen, const unsigned char *little,
- int littlelen)
- {
- const unsigned char *p;
- if ((biglen < littlelen) || (littlelen == 0))
- return NULL;
- p = big-1;
- while ((p=(const unsigned char *)
- memchr(p+1, little[0], (size_t)(big-(p+1)+biglen-littlelen+1)))
- != NULL) {
- if (memcmp(p+1, little+1, littlelen-1)==0)
- return (unsigned char *)p;
- }
- return NULL;
- }
- /* _zip_new:
- creates a new zipfile struct, and sets the contents to zero; returns
- the new struct. */
- struct zip *
- _zip_new(struct zip_error *error)
- {
- struct zip *za;
- za = (struct zip *)malloc(sizeof(struct zip));
- if (!za) {
- _zip_error_set(error, ZIP_ER_MEMORY, 0);
- return NULL;
- }
- za->zn = NULL;
- za->zp = NULL;
- _zip_error_init(&za->error);
- za->cdir = NULL;
- za->ch_comment = NULL;
- za->ch_comment_len = -1;
- za->nentry = za->nentry_alloc = 0;
- za->entry = NULL;
- za->nfile = za->nfile_alloc = 0;
- za->file = NULL;
- za->flags = za->ch_flags = 0;
- return za;
- }
- void *
- _zip_memdup(const void *mem, size_t len, struct zip_error *error)
- {
- void *ret;
- ret = malloc(len);
- if (!ret) {
- _zip_error_set(error, ZIP_ER_MEMORY, 0);
- return NULL;
- }
- memcpy(ret, mem, len);
- return ret;
- }
- ZIP_EXTERN int
- zip_get_num_files(struct zip *za)
- {
- if (za == NULL)
- return -1;
- return za->nentry;
- }
- ZIP_EXTERN const char *
- zip_get_name(struct zip *za, int idx, int flags)
- {
- return _zip_get_name(za, idx, flags, &za->error);
- }
- const char *
- _zip_get_name(struct zip *za, int idx, int flags, struct zip_error *error)
- {
- if (idx < 0 || idx >= za->nentry) {
- _zip_error_set(error, ZIP_ER_INVAL, 0);
- return NULL;
- }
- if ((flags & ZIP_FL_UNCHANGED) == 0) {
- if (za->entry[idx].state == ZIP_ST_DELETED) {
- _zip_error_set(error, ZIP_ER_DELETED, 0);
- return NULL;
- }
- if (za->entry[idx].ch_filename)
- return za->entry[idx].ch_filename;
- }
- if (za->cdir == NULL || idx >= za->cdir->nentry) {
- _zip_error_set(error, ZIP_ER_INVAL, 0);
- return NULL;
- }
- return za->cdir->entry[idx].filename;
- }
- ZIP_EXTERN const char *
- zip_get_file_comment(struct zip *za, int idx, int *lenp, int flags)
- {
- if (idx < 0 || idx >= za->nentry) {
- _zip_error_set(&za->error, ZIP_ER_INVAL, 0);
- return NULL;
- }
- if ((flags & ZIP_FL_UNCHANGED)
- || (za->entry[idx].ch_comment_len == -1)) {
- if (lenp != NULL)
- *lenp = za->cdir->entry[idx].comment_len;
- return za->cdir->entry[idx].comment;
- }
- if (lenp != NULL)
- *lenp = za->entry[idx].ch_comment_len;
- return za->entry[idx].ch_comment;
- }
- ZIP_EXTERN int
- zip_get_archive_flag(struct zip *za, int flag, int flags)
- {
- int fl;
- fl = (flags & ZIP_FL_UNCHANGED) ? za->flags : za->ch_flags;
- return (fl & flag) ? 1 : 0;
- }
- ZIP_EXTERN const char *
- zip_get_archive_comment(struct zip *za, int *lenp, int flags)
- {
- if ((flags & ZIP_FL_UNCHANGED)
- || (za->ch_comment_len == -1)) {
- if (za->cdir) {
- if (lenp != NULL)
- *lenp = za->cdir->comment_len;
- return za->cdir->comment;
- }
- else {
- if (lenp != NULL)
- *lenp = -1;
- return NULL;
- }
- }
- if (lenp != NULL)
- *lenp = za->ch_comment_len;
- return za->ch_comment;
- }
- /* _zip_free:
- frees the space allocated to a zipfile struct, and closes the
- corresponding file. */
- void
- _zip_free(struct zip *za)
- {
- int i;
- if (za == NULL)
- return;
- if (za->zn)
- free(za->zn);
- if (za->zp)
- fclose(za->zp);
- _zip_cdir_free(za->cdir);
- if (za->entry) {
- for (i=0; i<za->nentry; i++) {
- _zip_entry_free(za->entry+i);
- }
- free(za->entry);
- }
- for (i=0; i<za->nfile; i++) {
- if (za->file[i]->error.zip_err == ZIP_ER_OK) {
- _zip_error_set(&za->file[i]->error, ZIP_ER_ZIPCLOSED, 0);
- za->file[i]->za = NULL;
- }
- }
- free(za->file);
- free(za);
- return;
- }
- ZIP_EXTERN ssize_t
- zip_fread(struct zip_file *zf, void *outbuf, size_t toread)
- {
- int ret;
- size_t out_before, len;
- int i;
- if (!zf)
- return -1;
- if (zf->error.zip_err != 0)
- return -1;
- if ((zf->flags & ZIP_ZF_EOF) || (toread == 0))
- return 0;
- if (zf->bytes_left == 0) {
- zf->flags |= ZIP_ZF_EOF;
- if (zf->flags & ZIP_ZF_CRC) {
- if (zf->crc != zf->crc_orig) {
- _zip_error_set(&zf->error, ZIP_ER_CRC, 0);
- return -1;
- }
- }
- return 0;
- }
- if ((zf->flags & ZIP_ZF_DECOMP) == 0) {
- ret = _zip_file_fillbuf(outbuf, toread, zf);
- if (ret > 0) {
- if (zf->flags & ZIP_ZF_CRC)
- zf->crc = crc32(zf->crc, (Bytef *)outbuf, ret);
- zf->bytes_left -= ret;
- }
- return ret;
- }
- zf->zstr->next_out = (Bytef *)outbuf;
- zf->zstr->avail_out = toread;
- out_before = zf->zstr->total_out;
- /* endless loop until something has been accomplished */
- for (;;) {
- ret = inflate(zf->zstr, Z_SYNC_FLUSH);
- switch (ret) {
- case Z_OK:
- case Z_STREAM_END:
- /* all ok */
- /* Z_STREAM_END probably won't happen, since we didn't
- have a header */
- len = zf->zstr->total_out - out_before;
- if (len >= zf->bytes_left || len >= toread) {
- if (zf->flags & ZIP_ZF_CRC)
- zf->crc = crc32(zf->crc, (Bytef *)outbuf, len);
- zf->bytes_left -= len;
- return len;
- }
- break;
- case Z_BUF_ERROR:
- if (zf->zstr->avail_in == 0) {
- i = _zip_file_fillbuf(zf->buffer, BUFSIZE, zf);
- if (i == 0) {
- _zip_error_set(&zf->error, ZIP_ER_INCONS, 0);
- return -1;
- }
- else if (i < 0)
- return -1;
- zf->zstr->next_in = (Bytef *)zf->buffer;
- zf->zstr->avail_in = i;
- continue;
- }
- /* fallthrough */
- case Z_NEED_DICT:
- case Z_DATA_ERROR:
- case Z_STREAM_ERROR:
- case Z_MEM_ERROR:
- _zip_error_set(&zf->error, ZIP_ER_ZLIB, ret);
- return -1;
- }
- }
- }
- ZIP_EXTERN const char *
- zip_strerror(struct zip *za)
- {
- return _zip_error_strerror(&za->error);
- }
- ZIP_EXTERN void
- zip_stat_init(struct zip_stat *st)
- {
- st->name = NULL;
- st->index = -1;
- st->crc = 0;
- st->mtime = (time_t)-1;
- st->size = -1;
- st->comp_size = -1;
- st->comp_method = ZIP_CM_STORE;
- st->encryption_method = ZIP_EM_NONE;
- }
- ZIP_EXTERN int
- zip_stat_index(struct zip *za, int index, int flags, struct zip_stat *st)
- {
- const char *name;
- if (index < 0 || index >= za->nentry) {
- _zip_error_set(&za->error, ZIP_ER_INVAL, 0);
- return -1;
- }
- if ((name=zip_get_name(za, index, flags)) == NULL)
- return -1;
- if ((flags & ZIP_FL_UNCHANGED) == 0
- && ZIP_ENTRY_DATA_CHANGED(za->entry+index)) {
- if (za->entry[index].source->f(za->entry[index].source->ud,
- st, sizeof(*st), ZIP_SOURCE_STAT) < 0) {
- _zip_error_set(&za->error, ZIP_ER_CHANGED, 0);
- return -1;
- }
- }
- else {
- if (za->cdir == NULL || index >= za->cdir->nentry) {
- _zip_error_set(&za->error, ZIP_ER_INVAL, 0);
- return -1;
- }
- st->crc = za->cdir->entry[index].crc;
- st->size = za->cdir->entry[index].uncomp_size;
- st->mtime = za->cdir->entry[index].last_mod;
- st->comp_size = za->cdir->entry[index].comp_size;
- st->comp_method = za->cdir->entry[index].comp_method;
- if (za->cdir->entry[index].bitflags & ZIP_GPBF_ENCRYPTED) {
- if (za->cdir->entry[index].bitflags & ZIP_GPBF_STRONG_ENCRYPTION) {
- /* XXX */
- st->encryption_method = ZIP_EM_UNKNOWN;
- }
- else
- st->encryption_method = ZIP_EM_TRAD_PKWARE;
- }
- else
- st->encryption_method = ZIP_EM_NONE;
- /* st->bitflags = za->cdir->entry[index].bitflags; */
- }
- st->index = index;
- st->name = name;
- return 0;
- }
- ZIP_EXTERN int
- zip_stat(struct zip *za, const char *fname, int flags, struct zip_stat *st)
- {
- int idx;
- if ((idx=zip_name_locate(za, fname, flags)) < 0)
- return -1;
- return zip_stat_index(za, idx, flags, st);
- }
- struct read_zip {
- struct zip_file *zf;
- struct zip_stat st;
- myoff_t off, len;
- };
- static ssize_t read_zip(void *st, void *data, size_t len,
- enum zip_source_cmd cmd);
- ZIP_EXTERN struct zip_source *
- zip_source_zip(struct zip *za, struct zip *srcza, int srcidx, int flags,
- myoff_t start, myoff_t len)
- {
- struct zip_error error;
- struct zip_source *zs;
- struct read_zip *p;
- /* XXX: ZIP_FL_RECOMPRESS */
- if (za == NULL)
- return NULL;
- if (srcza == NULL || start < 0 || len < -1 || srcidx < 0 || srcidx >= srcza->nentry) {
- _zip_error_set(&za->error, ZIP_ER_INVAL, 0);
- return NULL;
- }
- if ((flags & ZIP_FL_UNCHANGED) == 0
- && ZIP_ENTRY_DATA_CHANGED(srcza->entry+srcidx)) {
- _zip_error_set(&za->error, ZIP_ER_CHANGED, 0);
- return NULL;
- }
- if (len == 0)
- len = -1;
- if (start == 0 && len == -1 && (flags & ZIP_FL_RECOMPRESS) == 0)
- flags |= ZIP_FL_COMPRESSED;
- else
- flags &= ~ZIP_FL_COMPRESSED;
- if ((p=(struct read_zip *)malloc(sizeof(*p))) == NULL) {
- _zip_error_set(&za->error, ZIP_ER_MEMORY, 0);
- return NULL;
- }
- _zip_error_copy(&error, &srcza->error);
- if (zip_stat_index(srcza, srcidx, flags, &p->st) < 0
- || (p->zf=zip_fopen_index(srcza, srcidx, flags)) == NULL) {
- free(p);
- _zip_error_copy(&za->error, &srcza->error);
- _zip_error_copy(&srcza->error, &error);
- return NULL;
- }
- p->off = start;
- p->len = len;
- if ((flags & ZIP_FL_COMPRESSED) == 0) {
- p->st.size = p->st.comp_size = len;
- p->st.comp_method = ZIP_CM_STORE;
- p->st.crc = 0;
- }
- if ((zs=zip_source_function(za, read_zip, p)) == NULL) {
- free(p);
- return NULL;
- }
- return zs;
- }
- static ssize_t
- read_zip(void *state, void *data, size_t len, enum zip_source_cmd cmd)
- {
- struct read_zip *z;
- char b[8192], *buf;
- int i, n;
- z = (struct read_zip *)state;
- buf = (char *)data;
- switch (cmd) {
- case ZIP_SOURCE_OPEN:
- for (n=0; n<z->off; n+= i) {
- i = (z->off-n > sizeof(b) ? sizeof(b) : z->off-n);
- if ((i=zip_fread(z->zf, b, i)) < 0) {
- zip_fclose(z->zf);
- z->zf = NULL;
- return -1;
- }
- }
- return 0;
- case ZIP_SOURCE_READ:
- if (z->len != -1)
- n = len > z->len ? z->len : len;
- else
- n = len;
- if ((i=zip_fread(z->zf, buf, n)) < 0)
- return -1;
- if (z->len != -1)
- z->len -= i;
- return i;
- case ZIP_SOURCE_CLOSE:
- return 0;
- case ZIP_SOURCE_STAT:
- if (len < sizeof(z->st))
- return -1;
- len = sizeof(z->st);
- memcpy(data, &z->st, len);
- return len;
- case ZIP_SOURCE_ERROR:
- {
- int *e;
- if (len < sizeof(int)*2)
- return -1;
- e = (int *)data;
- zip_file_error_get(z->zf, e, e+1);
- }
- return sizeof(int)*2;
- case ZIP_SOURCE_FREE:
- zip_fclose(z->zf);
- free(z);
- return 0;
- default:
- ;
- }
- return -1;
- }
- ZIP_EXTERN struct zip_source *
- zip_source_function(struct zip *za, zip_source_callback zcb, void *ud)
- {
- struct zip_source *zs;
- if (za == NULL)
- return NULL;
- if ((zs=(struct zip_source *)malloc(sizeof(*zs))) == NULL) {
- _zip_error_set(&za->error, ZIP_ER_MEMORY, 0);
- return NULL;
- }
- zs->f = zcb;
- zs->ud = ud;
- return zs;
- }
- ZIP_EXTERN void
- zip_source_free(struct zip_source *source)
- {
- if (source == NULL)
- return;
- (void)source->f(source->ud, NULL, 0, ZIP_SOURCE_FREE);
- free(source);
- }
- struct read_file {
- char *fname; /* name of file to copy from */
- FILE *f; /* file to copy from */
- myoff_t off; /* start offset of */
- myoff_t len; /* lengt of data to copy */
- myoff_t remain; /* bytes remaining to be copied */
- int e[2]; /* error codes */
- };
- static ssize_t read_file(void *state, void *data, size_t len,
- enum zip_source_cmd cmd);
- ZIP_EXTERN struct zip_source *
- zip_source_filep(struct zip *za, FILE *file, myoff_t start, myoff_t len)
- {
- if (za == NULL)
- return NULL;
- if (file == NULL || start < 0 || len < -1) {
- _zip_error_set(&za->error, ZIP_ER_INVAL, 0);
- return NULL;
- }
- return _zip_source_file_or_p(za, NULL, file, start, len);
- }
- struct zip_source *
- _zip_source_file_or_p(struct zip *za, const char *fname, FILE *file,
- myoff_t start, myoff_t len)
- {
- struct read_file *f;
- struct zip_source *zs;
- if (file == NULL && fname == NULL) {
- _zip_error_set(&za->error, ZIP_ER_INVAL, 0);
- return NULL;
- }
- if ((f=(struct read_file *)malloc(sizeof(struct read_file))) == NULL) {
- _zip_error_set(&za->error, ZIP_ER_MEMORY, 0);
- return NULL;
- }
- f->fname = NULL;
- if (fname) {
- if ((f->fname=strdup(fname)) == NULL) {
- _zip_error_set(&za->error, ZIP_ER_MEMORY, 0);
- free(f);
- return NULL;
- }
- }
- f->f = file;
- f->off = start;
- f->len = (len ? len : -1);
- if ((zs=zip_source_function(za, read_file, f)) == NULL) {
- free(f);
- return NULL;
- }
- return zs;
- }
- static ssize_t
- read_file(void *state, void *data, size_t len, enum zip_source_cmd cmd)
- {
- struct read_file *z;
- char *buf;
- int i, n;
- z = (struct read_file *)state;
- buf = (char *)data;
- switch (cmd) {
- case ZIP_SOURCE_OPEN:
- if (z->fname) {
- if ((z->f=fopen(z->fname, "rb")) == NULL) {
- z->e[0] = ZIP_ER_OPEN;
- z->e[1] = errno;
- return -1;
- }
- }
- if (fseeko(z->f, z->off, SEEK_SET) < 0) {
- z->e[0] = ZIP_ER_SEEK;
- z->e[1] = errno;
- return -1;
- }
- z->remain = z->len;
- return 0;
- case ZIP_SOURCE_READ:
- if (z->remain != -1)
- n = len > z->remain ? z->remain : len;
- else
- n = len;
- if ((i=fread(buf, 1, n, z->f)) < 0) {
- z->e[0] = ZIP_ER_READ;
- z->e[1] = errno;
- return -1;
- }
- if (z->remain != -1)
- z->remain -= i;
- return i;
- case ZIP_SOURCE_CLOSE:
- if (z->fname) {
- fclose(z->f);
- z->f = NULL;
- }
- return 0;
- case ZIP_SOURCE_STAT:
- {
- struct zip_stat *st;
- struct stat fst;
- int err;
- if (len < sizeof(*st))
- return -1;
- if (z->f)
- err = fstat(fileno(z->f), &fst);
- else
- err = stat(z->fname, &fst);
- if (err != 0) {
- z->e[0] = ZIP_ER_READ; /* best match */
- z->e[1] = errno;
- return -1;
- }
- st = (struct zip_stat *)data;
- zip_stat_init(st);
- st->mtime = fst.st_mtime;
- if (z->len != -1)
- st->size = z->len;
- else if ((fst.st_mode&S_IFMT) == S_IFREG)
- st->size = fst.st_size;
- return sizeof(*st);
- }
- case ZIP_SOURCE_ERROR:
- if (len < sizeof(int)*2)
- return -1;
- memcpy(data, z->e, sizeof(int)*2);
- return sizeof(int)*2;
- case ZIP_SOURCE_FREE:
- free(z->fname);
- if (z->f)
- fclose(z->f);
- free(z);
- return 0;
- default:
- ;
- }
- return -1;
- }
- ZIP_EXTERN int
- zip_name_locate(struct zip *za, const char *fname, int flags)
- {
- return _zip_name_locate(za, fname, flags, &za->error);
- }
- int
- _zip_name_locate(struct zip *za, const char *fname, int flags,
- struct zip_error *error)
- {
- int (*cmp)(const char *, const char *);
- const char *fn, *p;
- int i, n;
- if (fname == NULL) {
- _zip_error_set(error, ZIP_ER_INVAL, 0);
- return -1;
- }
- cmp = (flags & ZIP_FL_NOCASE) ? strcasecmp : strcmp;
- n = (flags & ZIP_FL_UNCHANGED) ? za->cdir->nentry : za->nentry;
- for (i=0; i<n; i++) {
- if (flags & ZIP_FL_UNCHANGED)
- fn = za->cdir->entry[i].filename;
- else
- fn = _zip_get_name(za, i, flags, error);
- /* newly added (partially filled) entry */
- if (fn == NULL)
- continue;
- if (flags & ZIP_FL_NODIR) {
- p = strrchr(fn, '/');
- if (p)
- fn = p+1;
- }
- if (cmp(fname, fn) == 0)
- return i;
- }
- _zip_error_set(error, ZIP_ER_NOENT, 0);
- return -1;
- }
|