annotate src/zlib-1.2.8/contrib/minizip/zip.c @ 43:5ea0608b923f

Current zlib source
author Chris Cannam
date Tue, 18 Oct 2016 14:33:52 +0100
parents
children
rev   line source
Chris@43 1 /* zip.c -- IO on .zip files using zlib
Chris@43 2 Version 1.1, February 14h, 2010
Chris@43 3 part of the MiniZip project - ( http://www.winimage.com/zLibDll/minizip.html )
Chris@43 4
Chris@43 5 Copyright (C) 1998-2010 Gilles Vollant (minizip) ( http://www.winimage.com/zLibDll/minizip.html )
Chris@43 6
Chris@43 7 Modifications for Zip64 support
Chris@43 8 Copyright (C) 2009-2010 Mathias Svensson ( http://result42.com )
Chris@43 9
Chris@43 10 For more info read MiniZip_info.txt
Chris@43 11
Chris@43 12 Changes
Chris@43 13 Oct-2009 - Mathias Svensson - Remove old C style function prototypes
Chris@43 14 Oct-2009 - Mathias Svensson - Added Zip64 Support when creating new file archives
Chris@43 15 Oct-2009 - Mathias Svensson - Did some code cleanup and refactoring to get better overview of some functions.
Chris@43 16 Oct-2009 - Mathias Svensson - Added zipRemoveExtraInfoBlock to strip extra field data from its ZIP64 data
Chris@43 17 It is used when recreting zip archive with RAW when deleting items from a zip.
Chris@43 18 ZIP64 data is automaticly added to items that needs it, and existing ZIP64 data need to be removed.
Chris@43 19 Oct-2009 - Mathias Svensson - Added support for BZIP2 as compression mode (bzip2 lib is required)
Chris@43 20 Jan-2010 - back to unzip and minizip 1.0 name scheme, with compatibility layer
Chris@43 21
Chris@43 22 */
Chris@43 23
Chris@43 24
Chris@43 25 #include <stdio.h>
Chris@43 26 #include <stdlib.h>
Chris@43 27 #include <string.h>
Chris@43 28 #include <time.h>
Chris@43 29 #include "zlib.h"
Chris@43 30 #include "zip.h"
Chris@43 31
Chris@43 32 #ifdef STDC
Chris@43 33 # include <stddef.h>
Chris@43 34 # include <string.h>
Chris@43 35 # include <stdlib.h>
Chris@43 36 #endif
Chris@43 37 #ifdef NO_ERRNO_H
Chris@43 38 extern int errno;
Chris@43 39 #else
Chris@43 40 # include <errno.h>
Chris@43 41 #endif
Chris@43 42
Chris@43 43
Chris@43 44 #ifndef local
Chris@43 45 # define local static
Chris@43 46 #endif
Chris@43 47 /* compile with -Dlocal if your debugger can't find static symbols */
Chris@43 48
Chris@43 49 #ifndef VERSIONMADEBY
Chris@43 50 # define VERSIONMADEBY (0x0) /* platform depedent */
Chris@43 51 #endif
Chris@43 52
Chris@43 53 #ifndef Z_BUFSIZE
Chris@43 54 #define Z_BUFSIZE (64*1024) //(16384)
Chris@43 55 #endif
Chris@43 56
Chris@43 57 #ifndef Z_MAXFILENAMEINZIP
Chris@43 58 #define Z_MAXFILENAMEINZIP (256)
Chris@43 59 #endif
Chris@43 60
Chris@43 61 #ifndef ALLOC
Chris@43 62 # define ALLOC(size) (malloc(size))
Chris@43 63 #endif
Chris@43 64 #ifndef TRYFREE
Chris@43 65 # define TRYFREE(p) {if (p) free(p);}
Chris@43 66 #endif
Chris@43 67
Chris@43 68 /*
Chris@43 69 #define SIZECENTRALDIRITEM (0x2e)
Chris@43 70 #define SIZEZIPLOCALHEADER (0x1e)
Chris@43 71 */
Chris@43 72
Chris@43 73 /* I've found an old Unix (a SunOS 4.1.3_U1) without all SEEK_* defined.... */
Chris@43 74
Chris@43 75
Chris@43 76 // NOT sure that this work on ALL platform
Chris@43 77 #define MAKEULONG64(a, b) ((ZPOS64_T)(((unsigned long)(a)) | ((ZPOS64_T)((unsigned long)(b))) << 32))
Chris@43 78
Chris@43 79 #ifndef SEEK_CUR
Chris@43 80 #define SEEK_CUR 1
Chris@43 81 #endif
Chris@43 82
Chris@43 83 #ifndef SEEK_END
Chris@43 84 #define SEEK_END 2
Chris@43 85 #endif
Chris@43 86
Chris@43 87 #ifndef SEEK_SET
Chris@43 88 #define SEEK_SET 0
Chris@43 89 #endif
Chris@43 90
Chris@43 91 #ifndef DEF_MEM_LEVEL
Chris@43 92 #if MAX_MEM_LEVEL >= 8
Chris@43 93 # define DEF_MEM_LEVEL 8
Chris@43 94 #else
Chris@43 95 # define DEF_MEM_LEVEL MAX_MEM_LEVEL
Chris@43 96 #endif
Chris@43 97 #endif
Chris@43 98 const char zip_copyright[] =" zip 1.01 Copyright 1998-2004 Gilles Vollant - http://www.winimage.com/zLibDll";
Chris@43 99
Chris@43 100
Chris@43 101 #define SIZEDATA_INDATABLOCK (4096-(4*4))
Chris@43 102
Chris@43 103 #define LOCALHEADERMAGIC (0x04034b50)
Chris@43 104 #define CENTRALHEADERMAGIC (0x02014b50)
Chris@43 105 #define ENDHEADERMAGIC (0x06054b50)
Chris@43 106 #define ZIP64ENDHEADERMAGIC (0x6064b50)
Chris@43 107 #define ZIP64ENDLOCHEADERMAGIC (0x7064b50)
Chris@43 108
Chris@43 109 #define FLAG_LOCALHEADER_OFFSET (0x06)
Chris@43 110 #define CRC_LOCALHEADER_OFFSET (0x0e)
Chris@43 111
Chris@43 112 #define SIZECENTRALHEADER (0x2e) /* 46 */
Chris@43 113
Chris@43 114 typedef struct linkedlist_datablock_internal_s
Chris@43 115 {
Chris@43 116 struct linkedlist_datablock_internal_s* next_datablock;
Chris@43 117 uLong avail_in_this_block;
Chris@43 118 uLong filled_in_this_block;
Chris@43 119 uLong unused; /* for future use and alignement */
Chris@43 120 unsigned char data[SIZEDATA_INDATABLOCK];
Chris@43 121 } linkedlist_datablock_internal;
Chris@43 122
Chris@43 123 typedef struct linkedlist_data_s
Chris@43 124 {
Chris@43 125 linkedlist_datablock_internal* first_block;
Chris@43 126 linkedlist_datablock_internal* last_block;
Chris@43 127 } linkedlist_data;
Chris@43 128
Chris@43 129
Chris@43 130 typedef struct
Chris@43 131 {
Chris@43 132 z_stream stream; /* zLib stream structure for inflate */
Chris@43 133 #ifdef HAVE_BZIP2
Chris@43 134 bz_stream bstream; /* bzLib stream structure for bziped */
Chris@43 135 #endif
Chris@43 136
Chris@43 137 int stream_initialised; /* 1 is stream is initialised */
Chris@43 138 uInt pos_in_buffered_data; /* last written byte in buffered_data */
Chris@43 139
Chris@43 140 ZPOS64_T pos_local_header; /* offset of the local header of the file
Chris@43 141 currenty writing */
Chris@43 142 char* central_header; /* central header data for the current file */
Chris@43 143 uLong size_centralExtra;
Chris@43 144 uLong size_centralheader; /* size of the central header for cur file */
Chris@43 145 uLong size_centralExtraFree; /* Extra bytes allocated to the centralheader but that are not used */
Chris@43 146 uLong flag; /* flag of the file currently writing */
Chris@43 147
Chris@43 148 int method; /* compression method of file currenty wr.*/
Chris@43 149 int raw; /* 1 for directly writing raw data */
Chris@43 150 Byte buffered_data[Z_BUFSIZE];/* buffer contain compressed data to be writ*/
Chris@43 151 uLong dosDate;
Chris@43 152 uLong crc32;
Chris@43 153 int encrypt;
Chris@43 154 int zip64; /* Add ZIP64 extened information in the extra field */
Chris@43 155 ZPOS64_T pos_zip64extrainfo;
Chris@43 156 ZPOS64_T totalCompressedData;
Chris@43 157 ZPOS64_T totalUncompressedData;
Chris@43 158 #ifndef NOCRYPT
Chris@43 159 unsigned long keys[3]; /* keys defining the pseudo-random sequence */
Chris@43 160 const z_crc_t* pcrc_32_tab;
Chris@43 161 int crypt_header_size;
Chris@43 162 #endif
Chris@43 163 } curfile64_info;
Chris@43 164
Chris@43 165 typedef struct
Chris@43 166 {
Chris@43 167 zlib_filefunc64_32_def z_filefunc;
Chris@43 168 voidpf filestream; /* io structore of the zipfile */
Chris@43 169 linkedlist_data central_dir;/* datablock with central dir in construction*/
Chris@43 170 int in_opened_file_inzip; /* 1 if a file in the zip is currently writ.*/
Chris@43 171 curfile64_info ci; /* info on the file curretly writing */
Chris@43 172
Chris@43 173 ZPOS64_T begin_pos; /* position of the beginning of the zipfile */
Chris@43 174 ZPOS64_T add_position_when_writting_offset;
Chris@43 175 ZPOS64_T number_entry;
Chris@43 176
Chris@43 177 #ifndef NO_ADDFILEINEXISTINGZIP
Chris@43 178 char *globalcomment;
Chris@43 179 #endif
Chris@43 180
Chris@43 181 } zip64_internal;
Chris@43 182
Chris@43 183
Chris@43 184 #ifndef NOCRYPT
Chris@43 185 #define INCLUDECRYPTINGCODE_IFCRYPTALLOWED
Chris@43 186 #include "crypt.h"
Chris@43 187 #endif
Chris@43 188
Chris@43 189 local linkedlist_datablock_internal* allocate_new_datablock()
Chris@43 190 {
Chris@43 191 linkedlist_datablock_internal* ldi;
Chris@43 192 ldi = (linkedlist_datablock_internal*)
Chris@43 193 ALLOC(sizeof(linkedlist_datablock_internal));
Chris@43 194 if (ldi!=NULL)
Chris@43 195 {
Chris@43 196 ldi->next_datablock = NULL ;
Chris@43 197 ldi->filled_in_this_block = 0 ;
Chris@43 198 ldi->avail_in_this_block = SIZEDATA_INDATABLOCK ;
Chris@43 199 }
Chris@43 200 return ldi;
Chris@43 201 }
Chris@43 202
Chris@43 203 local void free_datablock(linkedlist_datablock_internal* ldi)
Chris@43 204 {
Chris@43 205 while (ldi!=NULL)
Chris@43 206 {
Chris@43 207 linkedlist_datablock_internal* ldinext = ldi->next_datablock;
Chris@43 208 TRYFREE(ldi);
Chris@43 209 ldi = ldinext;
Chris@43 210 }
Chris@43 211 }
Chris@43 212
Chris@43 213 local void init_linkedlist(linkedlist_data* ll)
Chris@43 214 {
Chris@43 215 ll->first_block = ll->last_block = NULL;
Chris@43 216 }
Chris@43 217
Chris@43 218 local void free_linkedlist(linkedlist_data* ll)
Chris@43 219 {
Chris@43 220 free_datablock(ll->first_block);
Chris@43 221 ll->first_block = ll->last_block = NULL;
Chris@43 222 }
Chris@43 223
Chris@43 224
Chris@43 225 local int add_data_in_datablock(linkedlist_data* ll, const void* buf, uLong len)
Chris@43 226 {
Chris@43 227 linkedlist_datablock_internal* ldi;
Chris@43 228 const unsigned char* from_copy;
Chris@43 229
Chris@43 230 if (ll==NULL)
Chris@43 231 return ZIP_INTERNALERROR;
Chris@43 232
Chris@43 233 if (ll->last_block == NULL)
Chris@43 234 {
Chris@43 235 ll->first_block = ll->last_block = allocate_new_datablock();
Chris@43 236 if (ll->first_block == NULL)
Chris@43 237 return ZIP_INTERNALERROR;
Chris@43 238 }
Chris@43 239
Chris@43 240 ldi = ll->last_block;
Chris@43 241 from_copy = (unsigned char*)buf;
Chris@43 242
Chris@43 243 while (len>0)
Chris@43 244 {
Chris@43 245 uInt copy_this;
Chris@43 246 uInt i;
Chris@43 247 unsigned char* to_copy;
Chris@43 248
Chris@43 249 if (ldi->avail_in_this_block==0)
Chris@43 250 {
Chris@43 251 ldi->next_datablock = allocate_new_datablock();
Chris@43 252 if (ldi->next_datablock == NULL)
Chris@43 253 return ZIP_INTERNALERROR;
Chris@43 254 ldi = ldi->next_datablock ;
Chris@43 255 ll->last_block = ldi;
Chris@43 256 }
Chris@43 257
Chris@43 258 if (ldi->avail_in_this_block < len)
Chris@43 259 copy_this = (uInt)ldi->avail_in_this_block;
Chris@43 260 else
Chris@43 261 copy_this = (uInt)len;
Chris@43 262
Chris@43 263 to_copy = &(ldi->data[ldi->filled_in_this_block]);
Chris@43 264
Chris@43 265 for (i=0;i<copy_this;i++)
Chris@43 266 *(to_copy+i)=*(from_copy+i);
Chris@43 267
Chris@43 268 ldi->filled_in_this_block += copy_this;
Chris@43 269 ldi->avail_in_this_block -= copy_this;
Chris@43 270 from_copy += copy_this ;
Chris@43 271 len -= copy_this;
Chris@43 272 }
Chris@43 273 return ZIP_OK;
Chris@43 274 }
Chris@43 275
Chris@43 276
Chris@43 277
Chris@43 278 /****************************************************************************/
Chris@43 279
Chris@43 280 #ifndef NO_ADDFILEINEXISTINGZIP
Chris@43 281 /* ===========================================================================
Chris@43 282 Inputs a long in LSB order to the given file
Chris@43 283 nbByte == 1, 2 ,4 or 8 (byte, short or long, ZPOS64_T)
Chris@43 284 */
Chris@43 285
Chris@43 286 local int zip64local_putValue OF((const zlib_filefunc64_32_def* pzlib_filefunc_def, voidpf filestream, ZPOS64_T x, int nbByte));
Chris@43 287 local int zip64local_putValue (const zlib_filefunc64_32_def* pzlib_filefunc_def, voidpf filestream, ZPOS64_T x, int nbByte)
Chris@43 288 {
Chris@43 289 unsigned char buf[8];
Chris@43 290 int n;
Chris@43 291 for (n = 0; n < nbByte; n++)
Chris@43 292 {
Chris@43 293 buf[n] = (unsigned char)(x & 0xff);
Chris@43 294 x >>= 8;
Chris@43 295 }
Chris@43 296 if (x != 0)
Chris@43 297 { /* data overflow - hack for ZIP64 (X Roche) */
Chris@43 298 for (n = 0; n < nbByte; n++)
Chris@43 299 {
Chris@43 300 buf[n] = 0xff;
Chris@43 301 }
Chris@43 302 }
Chris@43 303
Chris@43 304 if (ZWRITE64(*pzlib_filefunc_def,filestream,buf,nbByte)!=(uLong)nbByte)
Chris@43 305 return ZIP_ERRNO;
Chris@43 306 else
Chris@43 307 return ZIP_OK;
Chris@43 308 }
Chris@43 309
Chris@43 310 local void zip64local_putValue_inmemory OF((void* dest, ZPOS64_T x, int nbByte));
Chris@43 311 local void zip64local_putValue_inmemory (void* dest, ZPOS64_T x, int nbByte)
Chris@43 312 {
Chris@43 313 unsigned char* buf=(unsigned char*)dest;
Chris@43 314 int n;
Chris@43 315 for (n = 0; n < nbByte; n++) {
Chris@43 316 buf[n] = (unsigned char)(x & 0xff);
Chris@43 317 x >>= 8;
Chris@43 318 }
Chris@43 319
Chris@43 320 if (x != 0)
Chris@43 321 { /* data overflow - hack for ZIP64 */
Chris@43 322 for (n = 0; n < nbByte; n++)
Chris@43 323 {
Chris@43 324 buf[n] = 0xff;
Chris@43 325 }
Chris@43 326 }
Chris@43 327 }
Chris@43 328
Chris@43 329 /****************************************************************************/
Chris@43 330
Chris@43 331
Chris@43 332 local uLong zip64local_TmzDateToDosDate(const tm_zip* ptm)
Chris@43 333 {
Chris@43 334 uLong year = (uLong)ptm->tm_year;
Chris@43 335 if (year>=1980)
Chris@43 336 year-=1980;
Chris@43 337 else if (year>=80)
Chris@43 338 year-=80;
Chris@43 339 return
Chris@43 340 (uLong) (((ptm->tm_mday) + (32 * (ptm->tm_mon+1)) + (512 * year)) << 16) |
Chris@43 341 ((ptm->tm_sec/2) + (32* ptm->tm_min) + (2048 * (uLong)ptm->tm_hour));
Chris@43 342 }
Chris@43 343
Chris@43 344
Chris@43 345 /****************************************************************************/
Chris@43 346
Chris@43 347 local int zip64local_getByte OF((const zlib_filefunc64_32_def* pzlib_filefunc_def, voidpf filestream, int *pi));
Chris@43 348
Chris@43 349 local int zip64local_getByte(const zlib_filefunc64_32_def* pzlib_filefunc_def,voidpf filestream,int* pi)
Chris@43 350 {
Chris@43 351 unsigned char c;
Chris@43 352 int err = (int)ZREAD64(*pzlib_filefunc_def,filestream,&c,1);
Chris@43 353 if (err==1)
Chris@43 354 {
Chris@43 355 *pi = (int)c;
Chris@43 356 return ZIP_OK;
Chris@43 357 }
Chris@43 358 else
Chris@43 359 {
Chris@43 360 if (ZERROR64(*pzlib_filefunc_def,filestream))
Chris@43 361 return ZIP_ERRNO;
Chris@43 362 else
Chris@43 363 return ZIP_EOF;
Chris@43 364 }
Chris@43 365 }
Chris@43 366
Chris@43 367
Chris@43 368 /* ===========================================================================
Chris@43 369 Reads a long in LSB order from the given gz_stream. Sets
Chris@43 370 */
Chris@43 371 local int zip64local_getShort OF((const zlib_filefunc64_32_def* pzlib_filefunc_def, voidpf filestream, uLong *pX));
Chris@43 372
Chris@43 373 local int zip64local_getShort (const zlib_filefunc64_32_def* pzlib_filefunc_def, voidpf filestream, uLong* pX)
Chris@43 374 {
Chris@43 375 uLong x ;
Chris@43 376 int i = 0;
Chris@43 377 int err;
Chris@43 378
Chris@43 379 err = zip64local_getByte(pzlib_filefunc_def,filestream,&i);
Chris@43 380 x = (uLong)i;
Chris@43 381
Chris@43 382 if (err==ZIP_OK)
Chris@43 383 err = zip64local_getByte(pzlib_filefunc_def,filestream,&i);
Chris@43 384 x += ((uLong)i)<<8;
Chris@43 385
Chris@43 386 if (err==ZIP_OK)
Chris@43 387 *pX = x;
Chris@43 388 else
Chris@43 389 *pX = 0;
Chris@43 390 return err;
Chris@43 391 }
Chris@43 392
Chris@43 393 local int zip64local_getLong OF((const zlib_filefunc64_32_def* pzlib_filefunc_def, voidpf filestream, uLong *pX));
Chris@43 394
Chris@43 395 local int zip64local_getLong (const zlib_filefunc64_32_def* pzlib_filefunc_def, voidpf filestream, uLong* pX)
Chris@43 396 {
Chris@43 397 uLong x ;
Chris@43 398 int i = 0;
Chris@43 399 int err;
Chris@43 400
Chris@43 401 err = zip64local_getByte(pzlib_filefunc_def,filestream,&i);
Chris@43 402 x = (uLong)i;
Chris@43 403
Chris@43 404 if (err==ZIP_OK)
Chris@43 405 err = zip64local_getByte(pzlib_filefunc_def,filestream,&i);
Chris@43 406 x += ((uLong)i)<<8;
Chris@43 407
Chris@43 408 if (err==ZIP_OK)
Chris@43 409 err = zip64local_getByte(pzlib_filefunc_def,filestream,&i);
Chris@43 410 x += ((uLong)i)<<16;
Chris@43 411
Chris@43 412 if (err==ZIP_OK)
Chris@43 413 err = zip64local_getByte(pzlib_filefunc_def,filestream,&i);
Chris@43 414 x += ((uLong)i)<<24;
Chris@43 415
Chris@43 416 if (err==ZIP_OK)
Chris@43 417 *pX = x;
Chris@43 418 else
Chris@43 419 *pX = 0;
Chris@43 420 return err;
Chris@43 421 }
Chris@43 422
Chris@43 423 local int zip64local_getLong64 OF((const zlib_filefunc64_32_def* pzlib_filefunc_def, voidpf filestream, ZPOS64_T *pX));
Chris@43 424
Chris@43 425
Chris@43 426 local int zip64local_getLong64 (const zlib_filefunc64_32_def* pzlib_filefunc_def, voidpf filestream, ZPOS64_T *pX)
Chris@43 427 {
Chris@43 428 ZPOS64_T x;
Chris@43 429 int i = 0;
Chris@43 430 int err;
Chris@43 431
Chris@43 432 err = zip64local_getByte(pzlib_filefunc_def,filestream,&i);
Chris@43 433 x = (ZPOS64_T)i;
Chris@43 434
Chris@43 435 if (err==ZIP_OK)
Chris@43 436 err = zip64local_getByte(pzlib_filefunc_def,filestream,&i);
Chris@43 437 x += ((ZPOS64_T)i)<<8;
Chris@43 438
Chris@43 439 if (err==ZIP_OK)
Chris@43 440 err = zip64local_getByte(pzlib_filefunc_def,filestream,&i);
Chris@43 441 x += ((ZPOS64_T)i)<<16;
Chris@43 442
Chris@43 443 if (err==ZIP_OK)
Chris@43 444 err = zip64local_getByte(pzlib_filefunc_def,filestream,&i);
Chris@43 445 x += ((ZPOS64_T)i)<<24;
Chris@43 446
Chris@43 447 if (err==ZIP_OK)
Chris@43 448 err = zip64local_getByte(pzlib_filefunc_def,filestream,&i);
Chris@43 449 x += ((ZPOS64_T)i)<<32;
Chris@43 450
Chris@43 451 if (err==ZIP_OK)
Chris@43 452 err = zip64local_getByte(pzlib_filefunc_def,filestream,&i);
Chris@43 453 x += ((ZPOS64_T)i)<<40;
Chris@43 454
Chris@43 455 if (err==ZIP_OK)
Chris@43 456 err = zip64local_getByte(pzlib_filefunc_def,filestream,&i);
Chris@43 457 x += ((ZPOS64_T)i)<<48;
Chris@43 458
Chris@43 459 if (err==ZIP_OK)
Chris@43 460 err = zip64local_getByte(pzlib_filefunc_def,filestream,&i);
Chris@43 461 x += ((ZPOS64_T)i)<<56;
Chris@43 462
Chris@43 463 if (err==ZIP_OK)
Chris@43 464 *pX = x;
Chris@43 465 else
Chris@43 466 *pX = 0;
Chris@43 467
Chris@43 468 return err;
Chris@43 469 }
Chris@43 470
Chris@43 471 #ifndef BUFREADCOMMENT
Chris@43 472 #define BUFREADCOMMENT (0x400)
Chris@43 473 #endif
Chris@43 474 /*
Chris@43 475 Locate the Central directory of a zipfile (at the end, just before
Chris@43 476 the global comment)
Chris@43 477 */
Chris@43 478 local ZPOS64_T zip64local_SearchCentralDir OF((const zlib_filefunc64_32_def* pzlib_filefunc_def, voidpf filestream));
Chris@43 479
Chris@43 480 local ZPOS64_T zip64local_SearchCentralDir(const zlib_filefunc64_32_def* pzlib_filefunc_def, voidpf filestream)
Chris@43 481 {
Chris@43 482 unsigned char* buf;
Chris@43 483 ZPOS64_T uSizeFile;
Chris@43 484 ZPOS64_T uBackRead;
Chris@43 485 ZPOS64_T uMaxBack=0xffff; /* maximum size of global comment */
Chris@43 486 ZPOS64_T uPosFound=0;
Chris@43 487
Chris@43 488 if (ZSEEK64(*pzlib_filefunc_def,filestream,0,ZLIB_FILEFUNC_SEEK_END) != 0)
Chris@43 489 return 0;
Chris@43 490
Chris@43 491
Chris@43 492 uSizeFile = ZTELL64(*pzlib_filefunc_def,filestream);
Chris@43 493
Chris@43 494 if (uMaxBack>uSizeFile)
Chris@43 495 uMaxBack = uSizeFile;
Chris@43 496
Chris@43 497 buf = (unsigned char*)ALLOC(BUFREADCOMMENT+4);
Chris@43 498 if (buf==NULL)
Chris@43 499 return 0;
Chris@43 500
Chris@43 501 uBackRead = 4;
Chris@43 502 while (uBackRead<uMaxBack)
Chris@43 503 {
Chris@43 504 uLong uReadSize;
Chris@43 505 ZPOS64_T uReadPos ;
Chris@43 506 int i;
Chris@43 507 if (uBackRead+BUFREADCOMMENT>uMaxBack)
Chris@43 508 uBackRead = uMaxBack;
Chris@43 509 else
Chris@43 510 uBackRead+=BUFREADCOMMENT;
Chris@43 511 uReadPos = uSizeFile-uBackRead ;
Chris@43 512
Chris@43 513 uReadSize = ((BUFREADCOMMENT+4) < (uSizeFile-uReadPos)) ?
Chris@43 514 (BUFREADCOMMENT+4) : (uLong)(uSizeFile-uReadPos);
Chris@43 515 if (ZSEEK64(*pzlib_filefunc_def,filestream,uReadPos,ZLIB_FILEFUNC_SEEK_SET)!=0)
Chris@43 516 break;
Chris@43 517
Chris@43 518 if (ZREAD64(*pzlib_filefunc_def,filestream,buf,uReadSize)!=uReadSize)
Chris@43 519 break;
Chris@43 520
Chris@43 521 for (i=(int)uReadSize-3; (i--)>0;)
Chris@43 522 if (((*(buf+i))==0x50) && ((*(buf+i+1))==0x4b) &&
Chris@43 523 ((*(buf+i+2))==0x05) && ((*(buf+i+3))==0x06))
Chris@43 524 {
Chris@43 525 uPosFound = uReadPos+i;
Chris@43 526 break;
Chris@43 527 }
Chris@43 528
Chris@43 529 if (uPosFound!=0)
Chris@43 530 break;
Chris@43 531 }
Chris@43 532 TRYFREE(buf);
Chris@43 533 return uPosFound;
Chris@43 534 }
Chris@43 535
Chris@43 536 /*
Chris@43 537 Locate the End of Zip64 Central directory locator and from there find the CD of a zipfile (at the end, just before
Chris@43 538 the global comment)
Chris@43 539 */
Chris@43 540 local ZPOS64_T zip64local_SearchCentralDir64 OF((const zlib_filefunc64_32_def* pzlib_filefunc_def, voidpf filestream));
Chris@43 541
Chris@43 542 local ZPOS64_T zip64local_SearchCentralDir64(const zlib_filefunc64_32_def* pzlib_filefunc_def, voidpf filestream)
Chris@43 543 {
Chris@43 544 unsigned char* buf;
Chris@43 545 ZPOS64_T uSizeFile;
Chris@43 546 ZPOS64_T uBackRead;
Chris@43 547 ZPOS64_T uMaxBack=0xffff; /* maximum size of global comment */
Chris@43 548 ZPOS64_T uPosFound=0;
Chris@43 549 uLong uL;
Chris@43 550 ZPOS64_T relativeOffset;
Chris@43 551
Chris@43 552 if (ZSEEK64(*pzlib_filefunc_def,filestream,0,ZLIB_FILEFUNC_SEEK_END) != 0)
Chris@43 553 return 0;
Chris@43 554
Chris@43 555 uSizeFile = ZTELL64(*pzlib_filefunc_def,filestream);
Chris@43 556
Chris@43 557 if (uMaxBack>uSizeFile)
Chris@43 558 uMaxBack = uSizeFile;
Chris@43 559
Chris@43 560 buf = (unsigned char*)ALLOC(BUFREADCOMMENT+4);
Chris@43 561 if (buf==NULL)
Chris@43 562 return 0;
Chris@43 563
Chris@43 564 uBackRead = 4;
Chris@43 565 while (uBackRead<uMaxBack)
Chris@43 566 {
Chris@43 567 uLong uReadSize;
Chris@43 568 ZPOS64_T uReadPos;
Chris@43 569 int i;
Chris@43 570 if (uBackRead+BUFREADCOMMENT>uMaxBack)
Chris@43 571 uBackRead = uMaxBack;
Chris@43 572 else
Chris@43 573 uBackRead+=BUFREADCOMMENT;
Chris@43 574 uReadPos = uSizeFile-uBackRead ;
Chris@43 575
Chris@43 576 uReadSize = ((BUFREADCOMMENT+4) < (uSizeFile-uReadPos)) ?
Chris@43 577 (BUFREADCOMMENT+4) : (uLong)(uSizeFile-uReadPos);
Chris@43 578 if (ZSEEK64(*pzlib_filefunc_def,filestream,uReadPos,ZLIB_FILEFUNC_SEEK_SET)!=0)
Chris@43 579 break;
Chris@43 580
Chris@43 581 if (ZREAD64(*pzlib_filefunc_def,filestream,buf,uReadSize)!=uReadSize)
Chris@43 582 break;
Chris@43 583
Chris@43 584 for (i=(int)uReadSize-3; (i--)>0;)
Chris@43 585 {
Chris@43 586 // Signature "0x07064b50" Zip64 end of central directory locater
Chris@43 587 if (((*(buf+i))==0x50) && ((*(buf+i+1))==0x4b) && ((*(buf+i+2))==0x06) && ((*(buf+i+3))==0x07))
Chris@43 588 {
Chris@43 589 uPosFound = uReadPos+i;
Chris@43 590 break;
Chris@43 591 }
Chris@43 592 }
Chris@43 593
Chris@43 594 if (uPosFound!=0)
Chris@43 595 break;
Chris@43 596 }
Chris@43 597
Chris@43 598 TRYFREE(buf);
Chris@43 599 if (uPosFound == 0)
Chris@43 600 return 0;
Chris@43 601
Chris@43 602 /* Zip64 end of central directory locator */
Chris@43 603 if (ZSEEK64(*pzlib_filefunc_def,filestream, uPosFound,ZLIB_FILEFUNC_SEEK_SET)!=0)
Chris@43 604 return 0;
Chris@43 605
Chris@43 606 /* the signature, already checked */
Chris@43 607 if (zip64local_getLong(pzlib_filefunc_def,filestream,&uL)!=ZIP_OK)
Chris@43 608 return 0;
Chris@43 609
Chris@43 610 /* number of the disk with the start of the zip64 end of central directory */
Chris@43 611 if (zip64local_getLong(pzlib_filefunc_def,filestream,&uL)!=ZIP_OK)
Chris@43 612 return 0;
Chris@43 613 if (uL != 0)
Chris@43 614 return 0;
Chris@43 615
Chris@43 616 /* relative offset of the zip64 end of central directory record */
Chris@43 617 if (zip64local_getLong64(pzlib_filefunc_def,filestream,&relativeOffset)!=ZIP_OK)
Chris@43 618 return 0;
Chris@43 619
Chris@43 620 /* total number of disks */
Chris@43 621 if (zip64local_getLong(pzlib_filefunc_def,filestream,&uL)!=ZIP_OK)
Chris@43 622 return 0;
Chris@43 623 if (uL != 1)
Chris@43 624 return 0;
Chris@43 625
Chris@43 626 /* Goto Zip64 end of central directory record */
Chris@43 627 if (ZSEEK64(*pzlib_filefunc_def,filestream, relativeOffset,ZLIB_FILEFUNC_SEEK_SET)!=0)
Chris@43 628 return 0;
Chris@43 629
Chris@43 630 /* the signature */
Chris@43 631 if (zip64local_getLong(pzlib_filefunc_def,filestream,&uL)!=ZIP_OK)
Chris@43 632 return 0;
Chris@43 633
Chris@43 634 if (uL != 0x06064b50) // signature of 'Zip64 end of central directory'
Chris@43 635 return 0;
Chris@43 636
Chris@43 637 return relativeOffset;
Chris@43 638 }
Chris@43 639
Chris@43 640 int LoadCentralDirectoryRecord(zip64_internal* pziinit)
Chris@43 641 {
Chris@43 642 int err=ZIP_OK;
Chris@43 643 ZPOS64_T byte_before_the_zipfile;/* byte before the zipfile, (>0 for sfx)*/
Chris@43 644
Chris@43 645 ZPOS64_T size_central_dir; /* size of the central directory */
Chris@43 646 ZPOS64_T offset_central_dir; /* offset of start of central directory */
Chris@43 647 ZPOS64_T central_pos;
Chris@43 648 uLong uL;
Chris@43 649
Chris@43 650 uLong number_disk; /* number of the current dist, used for
Chris@43 651 spaning ZIP, unsupported, always 0*/
Chris@43 652 uLong number_disk_with_CD; /* number the the disk with central dir, used
Chris@43 653 for spaning ZIP, unsupported, always 0*/
Chris@43 654 ZPOS64_T number_entry;
Chris@43 655 ZPOS64_T number_entry_CD; /* total number of entries in
Chris@43 656 the central dir
Chris@43 657 (same than number_entry on nospan) */
Chris@43 658 uLong VersionMadeBy;
Chris@43 659 uLong VersionNeeded;
Chris@43 660 uLong size_comment;
Chris@43 661
Chris@43 662 int hasZIP64Record = 0;
Chris@43 663
Chris@43 664 // check first if we find a ZIP64 record
Chris@43 665 central_pos = zip64local_SearchCentralDir64(&pziinit->z_filefunc,pziinit->filestream);
Chris@43 666 if(central_pos > 0)
Chris@43 667 {
Chris@43 668 hasZIP64Record = 1;
Chris@43 669 }
Chris@43 670 else if(central_pos == 0)
Chris@43 671 {
Chris@43 672 central_pos = zip64local_SearchCentralDir(&pziinit->z_filefunc,pziinit->filestream);
Chris@43 673 }
Chris@43 674
Chris@43 675 /* disable to allow appending to empty ZIP archive
Chris@43 676 if (central_pos==0)
Chris@43 677 err=ZIP_ERRNO;
Chris@43 678 */
Chris@43 679
Chris@43 680 if(hasZIP64Record)
Chris@43 681 {
Chris@43 682 ZPOS64_T sizeEndOfCentralDirectory;
Chris@43 683 if (ZSEEK64(pziinit->z_filefunc, pziinit->filestream, central_pos, ZLIB_FILEFUNC_SEEK_SET) != 0)
Chris@43 684 err=ZIP_ERRNO;
Chris@43 685
Chris@43 686 /* the signature, already checked */
Chris@43 687 if (zip64local_getLong(&pziinit->z_filefunc, pziinit->filestream,&uL)!=ZIP_OK)
Chris@43 688 err=ZIP_ERRNO;
Chris@43 689
Chris@43 690 /* size of zip64 end of central directory record */
Chris@43 691 if (zip64local_getLong64(&pziinit->z_filefunc, pziinit->filestream, &sizeEndOfCentralDirectory)!=ZIP_OK)
Chris@43 692 err=ZIP_ERRNO;
Chris@43 693
Chris@43 694 /* version made by */
Chris@43 695 if (zip64local_getShort(&pziinit->z_filefunc, pziinit->filestream, &VersionMadeBy)!=ZIP_OK)
Chris@43 696 err=ZIP_ERRNO;
Chris@43 697
Chris@43 698 /* version needed to extract */
Chris@43 699 if (zip64local_getShort(&pziinit->z_filefunc, pziinit->filestream, &VersionNeeded)!=ZIP_OK)
Chris@43 700 err=ZIP_ERRNO;
Chris@43 701
Chris@43 702 /* number of this disk */
Chris@43 703 if (zip64local_getLong(&pziinit->z_filefunc, pziinit->filestream,&number_disk)!=ZIP_OK)
Chris@43 704 err=ZIP_ERRNO;
Chris@43 705
Chris@43 706 /* number of the disk with the start of the central directory */
Chris@43 707 if (zip64local_getLong(&pziinit->z_filefunc, pziinit->filestream,&number_disk_with_CD)!=ZIP_OK)
Chris@43 708 err=ZIP_ERRNO;
Chris@43 709
Chris@43 710 /* total number of entries in the central directory on this disk */
Chris@43 711 if (zip64local_getLong64(&pziinit->z_filefunc, pziinit->filestream, &number_entry)!=ZIP_OK)
Chris@43 712 err=ZIP_ERRNO;
Chris@43 713
Chris@43 714 /* total number of entries in the central directory */
Chris@43 715 if (zip64local_getLong64(&pziinit->z_filefunc, pziinit->filestream,&number_entry_CD)!=ZIP_OK)
Chris@43 716 err=ZIP_ERRNO;
Chris@43 717
Chris@43 718 if ((number_entry_CD!=number_entry) || (number_disk_with_CD!=0) || (number_disk!=0))
Chris@43 719 err=ZIP_BADZIPFILE;
Chris@43 720
Chris@43 721 /* size of the central directory */
Chris@43 722 if (zip64local_getLong64(&pziinit->z_filefunc, pziinit->filestream,&size_central_dir)!=ZIP_OK)
Chris@43 723 err=ZIP_ERRNO;
Chris@43 724
Chris@43 725 /* offset of start of central directory with respect to the
Chris@43 726 starting disk number */
Chris@43 727 if (zip64local_getLong64(&pziinit->z_filefunc, pziinit->filestream,&offset_central_dir)!=ZIP_OK)
Chris@43 728 err=ZIP_ERRNO;
Chris@43 729
Chris@43 730 // TODO..
Chris@43 731 // read the comment from the standard central header.
Chris@43 732 size_comment = 0;
Chris@43 733 }
Chris@43 734 else
Chris@43 735 {
Chris@43 736 // Read End of central Directory info
Chris@43 737 if (ZSEEK64(pziinit->z_filefunc, pziinit->filestream, central_pos,ZLIB_FILEFUNC_SEEK_SET)!=0)
Chris@43 738 err=ZIP_ERRNO;
Chris@43 739
Chris@43 740 /* the signature, already checked */
Chris@43 741 if (zip64local_getLong(&pziinit->z_filefunc, pziinit->filestream,&uL)!=ZIP_OK)
Chris@43 742 err=ZIP_ERRNO;
Chris@43 743
Chris@43 744 /* number of this disk */
Chris@43 745 if (zip64local_getShort(&pziinit->z_filefunc, pziinit->filestream,&number_disk)!=ZIP_OK)
Chris@43 746 err=ZIP_ERRNO;
Chris@43 747
Chris@43 748 /* number of the disk with the start of the central directory */
Chris@43 749 if (zip64local_getShort(&pziinit->z_filefunc, pziinit->filestream,&number_disk_with_CD)!=ZIP_OK)
Chris@43 750 err=ZIP_ERRNO;
Chris@43 751
Chris@43 752 /* total number of entries in the central dir on this disk */
Chris@43 753 number_entry = 0;
Chris@43 754 if (zip64local_getShort(&pziinit->z_filefunc, pziinit->filestream, &uL)!=ZIP_OK)
Chris@43 755 err=ZIP_ERRNO;
Chris@43 756 else
Chris@43 757 number_entry = uL;
Chris@43 758
Chris@43 759 /* total number of entries in the central dir */
Chris@43 760 number_entry_CD = 0;
Chris@43 761 if (zip64local_getShort(&pziinit->z_filefunc, pziinit->filestream, &uL)!=ZIP_OK)
Chris@43 762 err=ZIP_ERRNO;
Chris@43 763 else
Chris@43 764 number_entry_CD = uL;
Chris@43 765
Chris@43 766 if ((number_entry_CD!=number_entry) || (number_disk_with_CD!=0) || (number_disk!=0))
Chris@43 767 err=ZIP_BADZIPFILE;
Chris@43 768
Chris@43 769 /* size of the central directory */
Chris@43 770 size_central_dir = 0;
Chris@43 771 if (zip64local_getLong(&pziinit->z_filefunc, pziinit->filestream, &uL)!=ZIP_OK)
Chris@43 772 err=ZIP_ERRNO;
Chris@43 773 else
Chris@43 774 size_central_dir = uL;
Chris@43 775
Chris@43 776 /* offset of start of central directory with respect to the starting disk number */
Chris@43 777 offset_central_dir = 0;
Chris@43 778 if (zip64local_getLong(&pziinit->z_filefunc, pziinit->filestream, &uL)!=ZIP_OK)
Chris@43 779 err=ZIP_ERRNO;
Chris@43 780 else
Chris@43 781 offset_central_dir = uL;
Chris@43 782
Chris@43 783
Chris@43 784 /* zipfile global comment length */
Chris@43 785 if (zip64local_getShort(&pziinit->z_filefunc, pziinit->filestream, &size_comment)!=ZIP_OK)
Chris@43 786 err=ZIP_ERRNO;
Chris@43 787 }
Chris@43 788
Chris@43 789 if ((central_pos<offset_central_dir+size_central_dir) &&
Chris@43 790 (err==ZIP_OK))
Chris@43 791 err=ZIP_BADZIPFILE;
Chris@43 792
Chris@43 793 if (err!=ZIP_OK)
Chris@43 794 {
Chris@43 795 ZCLOSE64(pziinit->z_filefunc, pziinit->filestream);
Chris@43 796 return ZIP_ERRNO;
Chris@43 797 }
Chris@43 798
Chris@43 799 if (size_comment>0)
Chris@43 800 {
Chris@43 801 pziinit->globalcomment = (char*)ALLOC(size_comment+1);
Chris@43 802 if (pziinit->globalcomment)
Chris@43 803 {
Chris@43 804 size_comment = ZREAD64(pziinit->z_filefunc, pziinit->filestream, pziinit->globalcomment,size_comment);
Chris@43 805 pziinit->globalcomment[size_comment]=0;
Chris@43 806 }
Chris@43 807 }
Chris@43 808
Chris@43 809 byte_before_the_zipfile = central_pos - (offset_central_dir+size_central_dir);
Chris@43 810 pziinit->add_position_when_writting_offset = byte_before_the_zipfile;
Chris@43 811
Chris@43 812 {
Chris@43 813 ZPOS64_T size_central_dir_to_read = size_central_dir;
Chris@43 814 size_t buf_size = SIZEDATA_INDATABLOCK;
Chris@43 815 void* buf_read = (void*)ALLOC(buf_size);
Chris@43 816 if (ZSEEK64(pziinit->z_filefunc, pziinit->filestream, offset_central_dir + byte_before_the_zipfile, ZLIB_FILEFUNC_SEEK_SET) != 0)
Chris@43 817 err=ZIP_ERRNO;
Chris@43 818
Chris@43 819 while ((size_central_dir_to_read>0) && (err==ZIP_OK))
Chris@43 820 {
Chris@43 821 ZPOS64_T read_this = SIZEDATA_INDATABLOCK;
Chris@43 822 if (read_this > size_central_dir_to_read)
Chris@43 823 read_this = size_central_dir_to_read;
Chris@43 824
Chris@43 825 if (ZREAD64(pziinit->z_filefunc, pziinit->filestream,buf_read,(uLong)read_this) != read_this)
Chris@43 826 err=ZIP_ERRNO;
Chris@43 827
Chris@43 828 if (err==ZIP_OK)
Chris@43 829 err = add_data_in_datablock(&pziinit->central_dir,buf_read, (uLong)read_this);
Chris@43 830
Chris@43 831 size_central_dir_to_read-=read_this;
Chris@43 832 }
Chris@43 833 TRYFREE(buf_read);
Chris@43 834 }
Chris@43 835 pziinit->begin_pos = byte_before_the_zipfile;
Chris@43 836 pziinit->number_entry = number_entry_CD;
Chris@43 837
Chris@43 838 if (ZSEEK64(pziinit->z_filefunc, pziinit->filestream, offset_central_dir+byte_before_the_zipfile,ZLIB_FILEFUNC_SEEK_SET) != 0)
Chris@43 839 err=ZIP_ERRNO;
Chris@43 840
Chris@43 841 return err;
Chris@43 842 }
Chris@43 843
Chris@43 844
Chris@43 845 #endif /* !NO_ADDFILEINEXISTINGZIP*/
Chris@43 846
Chris@43 847
Chris@43 848 /************************************************************/
Chris@43 849 extern zipFile ZEXPORT zipOpen3 (const void *pathname, int append, zipcharpc* globalcomment, zlib_filefunc64_32_def* pzlib_filefunc64_32_def)
Chris@43 850 {
Chris@43 851 zip64_internal ziinit;
Chris@43 852 zip64_internal* zi;
Chris@43 853 int err=ZIP_OK;
Chris@43 854
Chris@43 855 ziinit.z_filefunc.zseek32_file = NULL;
Chris@43 856 ziinit.z_filefunc.ztell32_file = NULL;
Chris@43 857 if (pzlib_filefunc64_32_def==NULL)
Chris@43 858 fill_fopen64_filefunc(&ziinit.z_filefunc.zfile_func64);
Chris@43 859 else
Chris@43 860 ziinit.z_filefunc = *pzlib_filefunc64_32_def;
Chris@43 861
Chris@43 862 ziinit.filestream = ZOPEN64(ziinit.z_filefunc,
Chris@43 863 pathname,
Chris@43 864 (append == APPEND_STATUS_CREATE) ?
Chris@43 865 (ZLIB_FILEFUNC_MODE_READ | ZLIB_FILEFUNC_MODE_WRITE | ZLIB_FILEFUNC_MODE_CREATE) :
Chris@43 866 (ZLIB_FILEFUNC_MODE_READ | ZLIB_FILEFUNC_MODE_WRITE | ZLIB_FILEFUNC_MODE_EXISTING));
Chris@43 867
Chris@43 868 if (ziinit.filestream == NULL)
Chris@43 869 return NULL;
Chris@43 870
Chris@43 871 if (append == APPEND_STATUS_CREATEAFTER)
Chris@43 872 ZSEEK64(ziinit.z_filefunc,ziinit.filestream,0,SEEK_END);
Chris@43 873
Chris@43 874 ziinit.begin_pos = ZTELL64(ziinit.z_filefunc,ziinit.filestream);
Chris@43 875 ziinit.in_opened_file_inzip = 0;
Chris@43 876 ziinit.ci.stream_initialised = 0;
Chris@43 877 ziinit.number_entry = 0;
Chris@43 878 ziinit.add_position_when_writting_offset = 0;
Chris@43 879 init_linkedlist(&(ziinit.central_dir));
Chris@43 880
Chris@43 881
Chris@43 882
Chris@43 883 zi = (zip64_internal*)ALLOC(sizeof(zip64_internal));
Chris@43 884 if (zi==NULL)
Chris@43 885 {
Chris@43 886 ZCLOSE64(ziinit.z_filefunc,ziinit.filestream);
Chris@43 887 return NULL;
Chris@43 888 }
Chris@43 889
Chris@43 890 /* now we add file in a zipfile */
Chris@43 891 # ifndef NO_ADDFILEINEXISTINGZIP
Chris@43 892 ziinit.globalcomment = NULL;
Chris@43 893 if (append == APPEND_STATUS_ADDINZIP)
Chris@43 894 {
Chris@43 895 // Read and Cache Central Directory Records
Chris@43 896 err = LoadCentralDirectoryRecord(&ziinit);
Chris@43 897 }
Chris@43 898
Chris@43 899 if (globalcomment)
Chris@43 900 {
Chris@43 901 *globalcomment = ziinit.globalcomment;
Chris@43 902 }
Chris@43 903 # endif /* !NO_ADDFILEINEXISTINGZIP*/
Chris@43 904
Chris@43 905 if (err != ZIP_OK)
Chris@43 906 {
Chris@43 907 # ifndef NO_ADDFILEINEXISTINGZIP
Chris@43 908 TRYFREE(ziinit.globalcomment);
Chris@43 909 # endif /* !NO_ADDFILEINEXISTINGZIP*/
Chris@43 910 TRYFREE(zi);
Chris@43 911 return NULL;
Chris@43 912 }
Chris@43 913 else
Chris@43 914 {
Chris@43 915 *zi = ziinit;
Chris@43 916 return (zipFile)zi;
Chris@43 917 }
Chris@43 918 }
Chris@43 919
Chris@43 920 extern zipFile ZEXPORT zipOpen2 (const char *pathname, int append, zipcharpc* globalcomment, zlib_filefunc_def* pzlib_filefunc32_def)
Chris@43 921 {
Chris@43 922 if (pzlib_filefunc32_def != NULL)
Chris@43 923 {
Chris@43 924 zlib_filefunc64_32_def zlib_filefunc64_32_def_fill;
Chris@43 925 fill_zlib_filefunc64_32_def_from_filefunc32(&zlib_filefunc64_32_def_fill,pzlib_filefunc32_def);
Chris@43 926 return zipOpen3(pathname, append, globalcomment, &zlib_filefunc64_32_def_fill);
Chris@43 927 }
Chris@43 928 else
Chris@43 929 return zipOpen3(pathname, append, globalcomment, NULL);
Chris@43 930 }
Chris@43 931
Chris@43 932 extern zipFile ZEXPORT zipOpen2_64 (const void *pathname, int append, zipcharpc* globalcomment, zlib_filefunc64_def* pzlib_filefunc_def)
Chris@43 933 {
Chris@43 934 if (pzlib_filefunc_def != NULL)
Chris@43 935 {
Chris@43 936 zlib_filefunc64_32_def zlib_filefunc64_32_def_fill;
Chris@43 937 zlib_filefunc64_32_def_fill.zfile_func64 = *pzlib_filefunc_def;
Chris@43 938 zlib_filefunc64_32_def_fill.ztell32_file = NULL;
Chris@43 939 zlib_filefunc64_32_def_fill.zseek32_file = NULL;
Chris@43 940 return zipOpen3(pathname, append, globalcomment, &zlib_filefunc64_32_def_fill);
Chris@43 941 }
Chris@43 942 else
Chris@43 943 return zipOpen3(pathname, append, globalcomment, NULL);
Chris@43 944 }
Chris@43 945
Chris@43 946
Chris@43 947
Chris@43 948 extern zipFile ZEXPORT zipOpen (const char* pathname, int append)
Chris@43 949 {
Chris@43 950 return zipOpen3((const void*)pathname,append,NULL,NULL);
Chris@43 951 }
Chris@43 952
Chris@43 953 extern zipFile ZEXPORT zipOpen64 (const void* pathname, int append)
Chris@43 954 {
Chris@43 955 return zipOpen3(pathname,append,NULL,NULL);
Chris@43 956 }
Chris@43 957
Chris@43 958 int Write_LocalFileHeader(zip64_internal* zi, const char* filename, uInt size_extrafield_local, const void* extrafield_local)
Chris@43 959 {
Chris@43 960 /* write the local header */
Chris@43 961 int err;
Chris@43 962 uInt size_filename = (uInt)strlen(filename);
Chris@43 963 uInt size_extrafield = size_extrafield_local;
Chris@43 964
Chris@43 965 err = zip64local_putValue(&zi->z_filefunc,zi->filestream,(uLong)LOCALHEADERMAGIC, 4);
Chris@43 966
Chris@43 967 if (err==ZIP_OK)
Chris@43 968 {
Chris@43 969 if(zi->ci.zip64)
Chris@43 970 err = zip64local_putValue(&zi->z_filefunc,zi->filestream,(uLong)45,2);/* version needed to extract */
Chris@43 971 else
Chris@43 972 err = zip64local_putValue(&zi->z_filefunc,zi->filestream,(uLong)20,2);/* version needed to extract */
Chris@43 973 }
Chris@43 974
Chris@43 975 if (err==ZIP_OK)
Chris@43 976 err = zip64local_putValue(&zi->z_filefunc,zi->filestream,(uLong)zi->ci.flag,2);
Chris@43 977
Chris@43 978 if (err==ZIP_OK)
Chris@43 979 err = zip64local_putValue(&zi->z_filefunc,zi->filestream,(uLong)zi->ci.method,2);
Chris@43 980
Chris@43 981 if (err==ZIP_OK)
Chris@43 982 err = zip64local_putValue(&zi->z_filefunc,zi->filestream,(uLong)zi->ci.dosDate,4);
Chris@43 983
Chris@43 984 // CRC / Compressed size / Uncompressed size will be filled in later and rewritten later
Chris@43 985 if (err==ZIP_OK)
Chris@43 986 err = zip64local_putValue(&zi->z_filefunc,zi->filestream,(uLong)0,4); /* crc 32, unknown */
Chris@43 987 if (err==ZIP_OK)
Chris@43 988 {
Chris@43 989 if(zi->ci.zip64)
Chris@43 990 err = zip64local_putValue(&zi->z_filefunc,zi->filestream,(uLong)0xFFFFFFFF,4); /* compressed size, unknown */
Chris@43 991 else
Chris@43 992 err = zip64local_putValue(&zi->z_filefunc,zi->filestream,(uLong)0,4); /* compressed size, unknown */
Chris@43 993 }
Chris@43 994 if (err==ZIP_OK)
Chris@43 995 {
Chris@43 996 if(zi->ci.zip64)
Chris@43 997 err = zip64local_putValue(&zi->z_filefunc,zi->filestream,(uLong)0xFFFFFFFF,4); /* uncompressed size, unknown */
Chris@43 998 else
Chris@43 999 err = zip64local_putValue(&zi->z_filefunc,zi->filestream,(uLong)0,4); /* uncompressed size, unknown */
Chris@43 1000 }
Chris@43 1001
Chris@43 1002 if (err==ZIP_OK)
Chris@43 1003 err = zip64local_putValue(&zi->z_filefunc,zi->filestream,(uLong)size_filename,2);
Chris@43 1004
Chris@43 1005 if(zi->ci.zip64)
Chris@43 1006 {
Chris@43 1007 size_extrafield += 20;
Chris@43 1008 }
Chris@43 1009
Chris@43 1010 if (err==ZIP_OK)
Chris@43 1011 err = zip64local_putValue(&zi->z_filefunc,zi->filestream,(uLong)size_extrafield,2);
Chris@43 1012
Chris@43 1013 if ((err==ZIP_OK) && (size_filename > 0))
Chris@43 1014 {
Chris@43 1015 if (ZWRITE64(zi->z_filefunc,zi->filestream,filename,size_filename)!=size_filename)
Chris@43 1016 err = ZIP_ERRNO;
Chris@43 1017 }
Chris@43 1018
Chris@43 1019 if ((err==ZIP_OK) && (size_extrafield_local > 0))
Chris@43 1020 {
Chris@43 1021 if (ZWRITE64(zi->z_filefunc, zi->filestream, extrafield_local, size_extrafield_local) != size_extrafield_local)
Chris@43 1022 err = ZIP_ERRNO;
Chris@43 1023 }
Chris@43 1024
Chris@43 1025
Chris@43 1026 if ((err==ZIP_OK) && (zi->ci.zip64))
Chris@43 1027 {
Chris@43 1028 // write the Zip64 extended info
Chris@43 1029 short HeaderID = 1;
Chris@43 1030 short DataSize = 16;
Chris@43 1031 ZPOS64_T CompressedSize = 0;
Chris@43 1032 ZPOS64_T UncompressedSize = 0;
Chris@43 1033
Chris@43 1034 // Remember position of Zip64 extended info for the local file header. (needed when we update size after done with file)
Chris@43 1035 zi->ci.pos_zip64extrainfo = ZTELL64(zi->z_filefunc,zi->filestream);
Chris@43 1036
Chris@43 1037 err = zip64local_putValue(&zi->z_filefunc, zi->filestream, (short)HeaderID,2);
Chris@43 1038 err = zip64local_putValue(&zi->z_filefunc, zi->filestream, (short)DataSize,2);
Chris@43 1039
Chris@43 1040 err = zip64local_putValue(&zi->z_filefunc, zi->filestream, (ZPOS64_T)UncompressedSize,8);
Chris@43 1041 err = zip64local_putValue(&zi->z_filefunc, zi->filestream, (ZPOS64_T)CompressedSize,8);
Chris@43 1042 }
Chris@43 1043
Chris@43 1044 return err;
Chris@43 1045 }
Chris@43 1046
Chris@43 1047 /*
Chris@43 1048 NOTE.
Chris@43 1049 When writing RAW the ZIP64 extended information in extrafield_local and extrafield_global needs to be stripped
Chris@43 1050 before calling this function it can be done with zipRemoveExtraInfoBlock
Chris@43 1051
Chris@43 1052 It is not done here because then we need to realloc a new buffer since parameters are 'const' and I want to minimize
Chris@43 1053 unnecessary allocations.
Chris@43 1054 */
Chris@43 1055 extern int ZEXPORT zipOpenNewFileInZip4_64 (zipFile file, const char* filename, const zip_fileinfo* zipfi,
Chris@43 1056 const void* extrafield_local, uInt size_extrafield_local,
Chris@43 1057 const void* extrafield_global, uInt size_extrafield_global,
Chris@43 1058 const char* comment, int method, int level, int raw,
Chris@43 1059 int windowBits,int memLevel, int strategy,
Chris@43 1060 const char* password, uLong crcForCrypting,
Chris@43 1061 uLong versionMadeBy, uLong flagBase, int zip64)
Chris@43 1062 {
Chris@43 1063 zip64_internal* zi;
Chris@43 1064 uInt size_filename;
Chris@43 1065 uInt size_comment;
Chris@43 1066 uInt i;
Chris@43 1067 int err = ZIP_OK;
Chris@43 1068
Chris@43 1069 # ifdef NOCRYPT
Chris@43 1070 (crcForCrypting);
Chris@43 1071 if (password != NULL)
Chris@43 1072 return ZIP_PARAMERROR;
Chris@43 1073 # endif
Chris@43 1074
Chris@43 1075 if (file == NULL)
Chris@43 1076 return ZIP_PARAMERROR;
Chris@43 1077
Chris@43 1078 #ifdef HAVE_BZIP2
Chris@43 1079 if ((method!=0) && (method!=Z_DEFLATED) && (method!=Z_BZIP2ED))
Chris@43 1080 return ZIP_PARAMERROR;
Chris@43 1081 #else
Chris@43 1082 if ((method!=0) && (method!=Z_DEFLATED))
Chris@43 1083 return ZIP_PARAMERROR;
Chris@43 1084 #endif
Chris@43 1085
Chris@43 1086 zi = (zip64_internal*)file;
Chris@43 1087
Chris@43 1088 if (zi->in_opened_file_inzip == 1)
Chris@43 1089 {
Chris@43 1090 err = zipCloseFileInZip (file);
Chris@43 1091 if (err != ZIP_OK)
Chris@43 1092 return err;
Chris@43 1093 }
Chris@43 1094
Chris@43 1095 if (filename==NULL)
Chris@43 1096 filename="-";
Chris@43 1097
Chris@43 1098 if (comment==NULL)
Chris@43 1099 size_comment = 0;
Chris@43 1100 else
Chris@43 1101 size_comment = (uInt)strlen(comment);
Chris@43 1102
Chris@43 1103 size_filename = (uInt)strlen(filename);
Chris@43 1104
Chris@43 1105 if (zipfi == NULL)
Chris@43 1106 zi->ci.dosDate = 0;
Chris@43 1107 else
Chris@43 1108 {
Chris@43 1109 if (zipfi->dosDate != 0)
Chris@43 1110 zi->ci.dosDate = zipfi->dosDate;
Chris@43 1111 else
Chris@43 1112 zi->ci.dosDate = zip64local_TmzDateToDosDate(&zipfi->tmz_date);
Chris@43 1113 }
Chris@43 1114
Chris@43 1115 zi->ci.flag = flagBase;
Chris@43 1116 if ((level==8) || (level==9))
Chris@43 1117 zi->ci.flag |= 2;
Chris@43 1118 if (level==2)
Chris@43 1119 zi->ci.flag |= 4;
Chris@43 1120 if (level==1)
Chris@43 1121 zi->ci.flag |= 6;
Chris@43 1122 if (password != NULL)
Chris@43 1123 zi->ci.flag |= 1;
Chris@43 1124
Chris@43 1125 zi->ci.crc32 = 0;
Chris@43 1126 zi->ci.method = method;
Chris@43 1127 zi->ci.encrypt = 0;
Chris@43 1128 zi->ci.stream_initialised = 0;
Chris@43 1129 zi->ci.pos_in_buffered_data = 0;
Chris@43 1130 zi->ci.raw = raw;
Chris@43 1131 zi->ci.pos_local_header = ZTELL64(zi->z_filefunc,zi->filestream);
Chris@43 1132
Chris@43 1133 zi->ci.size_centralheader = SIZECENTRALHEADER + size_filename + size_extrafield_global + size_comment;
Chris@43 1134 zi->ci.size_centralExtraFree = 32; // Extra space we have reserved in case we need to add ZIP64 extra info data
Chris@43 1135
Chris@43 1136 zi->ci.central_header = (char*)ALLOC((uInt)zi->ci.size_centralheader + zi->ci.size_centralExtraFree);
Chris@43 1137
Chris@43 1138 zi->ci.size_centralExtra = size_extrafield_global;
Chris@43 1139 zip64local_putValue_inmemory(zi->ci.central_header,(uLong)CENTRALHEADERMAGIC,4);
Chris@43 1140 /* version info */
Chris@43 1141 zip64local_putValue_inmemory(zi->ci.central_header+4,(uLong)versionMadeBy,2);
Chris@43 1142 zip64local_putValue_inmemory(zi->ci.central_header+6,(uLong)20,2);
Chris@43 1143 zip64local_putValue_inmemory(zi->ci.central_header+8,(uLong)zi->ci.flag,2);
Chris@43 1144 zip64local_putValue_inmemory(zi->ci.central_header+10,(uLong)zi->ci.method,2);
Chris@43 1145 zip64local_putValue_inmemory(zi->ci.central_header+12,(uLong)zi->ci.dosDate,4);
Chris@43 1146 zip64local_putValue_inmemory(zi->ci.central_header+16,(uLong)0,4); /*crc*/
Chris@43 1147 zip64local_putValue_inmemory(zi->ci.central_header+20,(uLong)0,4); /*compr size*/
Chris@43 1148 zip64local_putValue_inmemory(zi->ci.central_header+24,(uLong)0,4); /*uncompr size*/
Chris@43 1149 zip64local_putValue_inmemory(zi->ci.central_header+28,(uLong)size_filename,2);
Chris@43 1150 zip64local_putValue_inmemory(zi->ci.central_header+30,(uLong)size_extrafield_global,2);
Chris@43 1151 zip64local_putValue_inmemory(zi->ci.central_header+32,(uLong)size_comment,2);
Chris@43 1152 zip64local_putValue_inmemory(zi->ci.central_header+34,(uLong)0,2); /*disk nm start*/
Chris@43 1153
Chris@43 1154 if (zipfi==NULL)
Chris@43 1155 zip64local_putValue_inmemory(zi->ci.central_header+36,(uLong)0,2);
Chris@43 1156 else
Chris@43 1157 zip64local_putValue_inmemory(zi->ci.central_header+36,(uLong)zipfi->internal_fa,2);
Chris@43 1158
Chris@43 1159 if (zipfi==NULL)
Chris@43 1160 zip64local_putValue_inmemory(zi->ci.central_header+38,(uLong)0,4);
Chris@43 1161 else
Chris@43 1162 zip64local_putValue_inmemory(zi->ci.central_header+38,(uLong)zipfi->external_fa,4);
Chris@43 1163
Chris@43 1164 if(zi->ci.pos_local_header >= 0xffffffff)
Chris@43 1165 zip64local_putValue_inmemory(zi->ci.central_header+42,(uLong)0xffffffff,4);
Chris@43 1166 else
Chris@43 1167 zip64local_putValue_inmemory(zi->ci.central_header+42,(uLong)zi->ci.pos_local_header - zi->add_position_when_writting_offset,4);
Chris@43 1168
Chris@43 1169 for (i=0;i<size_filename;i++)
Chris@43 1170 *(zi->ci.central_header+SIZECENTRALHEADER+i) = *(filename+i);
Chris@43 1171
Chris@43 1172 for (i=0;i<size_extrafield_global;i++)
Chris@43 1173 *(zi->ci.central_header+SIZECENTRALHEADER+size_filename+i) =
Chris@43 1174 *(((const char*)extrafield_global)+i);
Chris@43 1175
Chris@43 1176 for (i=0;i<size_comment;i++)
Chris@43 1177 *(zi->ci.central_header+SIZECENTRALHEADER+size_filename+
Chris@43 1178 size_extrafield_global+i) = *(comment+i);
Chris@43 1179 if (zi->ci.central_header == NULL)
Chris@43 1180 return ZIP_INTERNALERROR;
Chris@43 1181
Chris@43 1182 zi->ci.zip64 = zip64;
Chris@43 1183 zi->ci.totalCompressedData = 0;
Chris@43 1184 zi->ci.totalUncompressedData = 0;
Chris@43 1185 zi->ci.pos_zip64extrainfo = 0;
Chris@43 1186
Chris@43 1187 err = Write_LocalFileHeader(zi, filename, size_extrafield_local, extrafield_local);
Chris@43 1188
Chris@43 1189 #ifdef HAVE_BZIP2
Chris@43 1190 zi->ci.bstream.avail_in = (uInt)0;
Chris@43 1191 zi->ci.bstream.avail_out = (uInt)Z_BUFSIZE;
Chris@43 1192 zi->ci.bstream.next_out = (char*)zi->ci.buffered_data;
Chris@43 1193 zi->ci.bstream.total_in_hi32 = 0;
Chris@43 1194 zi->ci.bstream.total_in_lo32 = 0;
Chris@43 1195 zi->ci.bstream.total_out_hi32 = 0;
Chris@43 1196 zi->ci.bstream.total_out_lo32 = 0;
Chris@43 1197 #endif
Chris@43 1198
Chris@43 1199 zi->ci.stream.avail_in = (uInt)0;
Chris@43 1200 zi->ci.stream.avail_out = (uInt)Z_BUFSIZE;
Chris@43 1201 zi->ci.stream.next_out = zi->ci.buffered_data;
Chris@43 1202 zi->ci.stream.total_in = 0;
Chris@43 1203 zi->ci.stream.total_out = 0;
Chris@43 1204 zi->ci.stream.data_type = Z_BINARY;
Chris@43 1205
Chris@43 1206 #ifdef HAVE_BZIP2
Chris@43 1207 if ((err==ZIP_OK) && (zi->ci.method == Z_DEFLATED || zi->ci.method == Z_BZIP2ED) && (!zi->ci.raw))
Chris@43 1208 #else
Chris@43 1209 if ((err==ZIP_OK) && (zi->ci.method == Z_DEFLATED) && (!zi->ci.raw))
Chris@43 1210 #endif
Chris@43 1211 {
Chris@43 1212 if(zi->ci.method == Z_DEFLATED)
Chris@43 1213 {
Chris@43 1214 zi->ci.stream.zalloc = (alloc_func)0;
Chris@43 1215 zi->ci.stream.zfree = (free_func)0;
Chris@43 1216 zi->ci.stream.opaque = (voidpf)0;
Chris@43 1217
Chris@43 1218 if (windowBits>0)
Chris@43 1219 windowBits = -windowBits;
Chris@43 1220
Chris@43 1221 err = deflateInit2(&zi->ci.stream, level, Z_DEFLATED, windowBits, memLevel, strategy);
Chris@43 1222
Chris@43 1223 if (err==Z_OK)
Chris@43 1224 zi->ci.stream_initialised = Z_DEFLATED;
Chris@43 1225 }
Chris@43 1226 else if(zi->ci.method == Z_BZIP2ED)
Chris@43 1227 {
Chris@43 1228 #ifdef HAVE_BZIP2
Chris@43 1229 // Init BZip stuff here
Chris@43 1230 zi->ci.bstream.bzalloc = 0;
Chris@43 1231 zi->ci.bstream.bzfree = 0;
Chris@43 1232 zi->ci.bstream.opaque = (voidpf)0;
Chris@43 1233
Chris@43 1234 err = BZ2_bzCompressInit(&zi->ci.bstream, level, 0,35);
Chris@43 1235 if(err == BZ_OK)
Chris@43 1236 zi->ci.stream_initialised = Z_BZIP2ED;
Chris@43 1237 #endif
Chris@43 1238 }
Chris@43 1239
Chris@43 1240 }
Chris@43 1241
Chris@43 1242 # ifndef NOCRYPT
Chris@43 1243 zi->ci.crypt_header_size = 0;
Chris@43 1244 if ((err==Z_OK) && (password != NULL))
Chris@43 1245 {
Chris@43 1246 unsigned char bufHead[RAND_HEAD_LEN];
Chris@43 1247 unsigned int sizeHead;
Chris@43 1248 zi->ci.encrypt = 1;
Chris@43 1249 zi->ci.pcrc_32_tab = get_crc_table();
Chris@43 1250 /*init_keys(password,zi->ci.keys,zi->ci.pcrc_32_tab);*/
Chris@43 1251
Chris@43 1252 sizeHead=crypthead(password,bufHead,RAND_HEAD_LEN,zi->ci.keys,zi->ci.pcrc_32_tab,crcForCrypting);
Chris@43 1253 zi->ci.crypt_header_size = sizeHead;
Chris@43 1254
Chris@43 1255 if (ZWRITE64(zi->z_filefunc,zi->filestream,bufHead,sizeHead) != sizeHead)
Chris@43 1256 err = ZIP_ERRNO;
Chris@43 1257 }
Chris@43 1258 # endif
Chris@43 1259
Chris@43 1260 if (err==Z_OK)
Chris@43 1261 zi->in_opened_file_inzip = 1;
Chris@43 1262 return err;
Chris@43 1263 }
Chris@43 1264
Chris@43 1265 extern int ZEXPORT zipOpenNewFileInZip4 (zipFile file, const char* filename, const zip_fileinfo* zipfi,
Chris@43 1266 const void* extrafield_local, uInt size_extrafield_local,
Chris@43 1267 const void* extrafield_global, uInt size_extrafield_global,
Chris@43 1268 const char* comment, int method, int level, int raw,
Chris@43 1269 int windowBits,int memLevel, int strategy,
Chris@43 1270 const char* password, uLong crcForCrypting,
Chris@43 1271 uLong versionMadeBy, uLong flagBase)
Chris@43 1272 {
Chris@43 1273 return zipOpenNewFileInZip4_64 (file, filename, zipfi,
Chris@43 1274 extrafield_local, size_extrafield_local,
Chris@43 1275 extrafield_global, size_extrafield_global,
Chris@43 1276 comment, method, level, raw,
Chris@43 1277 windowBits, memLevel, strategy,
Chris@43 1278 password, crcForCrypting, versionMadeBy, flagBase, 0);
Chris@43 1279 }
Chris@43 1280
Chris@43 1281 extern int ZEXPORT zipOpenNewFileInZip3 (zipFile file, const char* filename, const zip_fileinfo* zipfi,
Chris@43 1282 const void* extrafield_local, uInt size_extrafield_local,
Chris@43 1283 const void* extrafield_global, uInt size_extrafield_global,
Chris@43 1284 const char* comment, int method, int level, int raw,
Chris@43 1285 int windowBits,int memLevel, int strategy,
Chris@43 1286 const char* password, uLong crcForCrypting)
Chris@43 1287 {
Chris@43 1288 return zipOpenNewFileInZip4_64 (file, filename, zipfi,
Chris@43 1289 extrafield_local, size_extrafield_local,
Chris@43 1290 extrafield_global, size_extrafield_global,
Chris@43 1291 comment, method, level, raw,
Chris@43 1292 windowBits, memLevel, strategy,
Chris@43 1293 password, crcForCrypting, VERSIONMADEBY, 0, 0);
Chris@43 1294 }
Chris@43 1295
Chris@43 1296 extern int ZEXPORT zipOpenNewFileInZip3_64(zipFile file, const char* filename, const zip_fileinfo* zipfi,
Chris@43 1297 const void* extrafield_local, uInt size_extrafield_local,
Chris@43 1298 const void* extrafield_global, uInt size_extrafield_global,
Chris@43 1299 const char* comment, int method, int level, int raw,
Chris@43 1300 int windowBits,int memLevel, int strategy,
Chris@43 1301 const char* password, uLong crcForCrypting, int zip64)
Chris@43 1302 {
Chris@43 1303 return zipOpenNewFileInZip4_64 (file, filename, zipfi,
Chris@43 1304 extrafield_local, size_extrafield_local,
Chris@43 1305 extrafield_global, size_extrafield_global,
Chris@43 1306 comment, method, level, raw,
Chris@43 1307 windowBits, memLevel, strategy,
Chris@43 1308 password, crcForCrypting, VERSIONMADEBY, 0, zip64);
Chris@43 1309 }
Chris@43 1310
Chris@43 1311 extern int ZEXPORT zipOpenNewFileInZip2(zipFile file, const char* filename, const zip_fileinfo* zipfi,
Chris@43 1312 const void* extrafield_local, uInt size_extrafield_local,
Chris@43 1313 const void* extrafield_global, uInt size_extrafield_global,
Chris@43 1314 const char* comment, int method, int level, int raw)
Chris@43 1315 {
Chris@43 1316 return zipOpenNewFileInZip4_64 (file, filename, zipfi,
Chris@43 1317 extrafield_local, size_extrafield_local,
Chris@43 1318 extrafield_global, size_extrafield_global,
Chris@43 1319 comment, method, level, raw,
Chris@43 1320 -MAX_WBITS, DEF_MEM_LEVEL, Z_DEFAULT_STRATEGY,
Chris@43 1321 NULL, 0, VERSIONMADEBY, 0, 0);
Chris@43 1322 }
Chris@43 1323
Chris@43 1324 extern int ZEXPORT zipOpenNewFileInZip2_64(zipFile file, const char* filename, const zip_fileinfo* zipfi,
Chris@43 1325 const void* extrafield_local, uInt size_extrafield_local,
Chris@43 1326 const void* extrafield_global, uInt size_extrafield_global,
Chris@43 1327 const char* comment, int method, int level, int raw, int zip64)
Chris@43 1328 {
Chris@43 1329 return zipOpenNewFileInZip4_64 (file, filename, zipfi,
Chris@43 1330 extrafield_local, size_extrafield_local,
Chris@43 1331 extrafield_global, size_extrafield_global,
Chris@43 1332 comment, method, level, raw,
Chris@43 1333 -MAX_WBITS, DEF_MEM_LEVEL, Z_DEFAULT_STRATEGY,
Chris@43 1334 NULL, 0, VERSIONMADEBY, 0, zip64);
Chris@43 1335 }
Chris@43 1336
Chris@43 1337 extern int ZEXPORT zipOpenNewFileInZip64 (zipFile file, const char* filename, const zip_fileinfo* zipfi,
Chris@43 1338 const void* extrafield_local, uInt size_extrafield_local,
Chris@43 1339 const void*extrafield_global, uInt size_extrafield_global,
Chris@43 1340 const char* comment, int method, int level, int zip64)
Chris@43 1341 {
Chris@43 1342 return zipOpenNewFileInZip4_64 (file, filename, zipfi,
Chris@43 1343 extrafield_local, size_extrafield_local,
Chris@43 1344 extrafield_global, size_extrafield_global,
Chris@43 1345 comment, method, level, 0,
Chris@43 1346 -MAX_WBITS, DEF_MEM_LEVEL, Z_DEFAULT_STRATEGY,
Chris@43 1347 NULL, 0, VERSIONMADEBY, 0, zip64);
Chris@43 1348 }
Chris@43 1349
Chris@43 1350 extern int ZEXPORT zipOpenNewFileInZip (zipFile file, const char* filename, const zip_fileinfo* zipfi,
Chris@43 1351 const void* extrafield_local, uInt size_extrafield_local,
Chris@43 1352 const void*extrafield_global, uInt size_extrafield_global,
Chris@43 1353 const char* comment, int method, int level)
Chris@43 1354 {
Chris@43 1355 return zipOpenNewFileInZip4_64 (file, filename, zipfi,
Chris@43 1356 extrafield_local, size_extrafield_local,
Chris@43 1357 extrafield_global, size_extrafield_global,
Chris@43 1358 comment, method, level, 0,
Chris@43 1359 -MAX_WBITS, DEF_MEM_LEVEL, Z_DEFAULT_STRATEGY,
Chris@43 1360 NULL, 0, VERSIONMADEBY, 0, 0);
Chris@43 1361 }
Chris@43 1362
Chris@43 1363 local int zip64FlushWriteBuffer(zip64_internal* zi)
Chris@43 1364 {
Chris@43 1365 int err=ZIP_OK;
Chris@43 1366
Chris@43 1367 if (zi->ci.encrypt != 0)
Chris@43 1368 {
Chris@43 1369 #ifndef NOCRYPT
Chris@43 1370 uInt i;
Chris@43 1371 int t;
Chris@43 1372 for (i=0;i<zi->ci.pos_in_buffered_data;i++)
Chris@43 1373 zi->ci.buffered_data[i] = zencode(zi->ci.keys, zi->ci.pcrc_32_tab, zi->ci.buffered_data[i],t);
Chris@43 1374 #endif
Chris@43 1375 }
Chris@43 1376
Chris@43 1377 if (ZWRITE64(zi->z_filefunc,zi->filestream,zi->ci.buffered_data,zi->ci.pos_in_buffered_data) != zi->ci.pos_in_buffered_data)
Chris@43 1378 err = ZIP_ERRNO;
Chris@43 1379
Chris@43 1380 zi->ci.totalCompressedData += zi->ci.pos_in_buffered_data;
Chris@43 1381
Chris@43 1382 #ifdef HAVE_BZIP2
Chris@43 1383 if(zi->ci.method == Z_BZIP2ED)
Chris@43 1384 {
Chris@43 1385 zi->ci.totalUncompressedData += zi->ci.bstream.total_in_lo32;
Chris@43 1386 zi->ci.bstream.total_in_lo32 = 0;
Chris@43 1387 zi->ci.bstream.total_in_hi32 = 0;
Chris@43 1388 }
Chris@43 1389 else
Chris@43 1390 #endif
Chris@43 1391 {
Chris@43 1392 zi->ci.totalUncompressedData += zi->ci.stream.total_in;
Chris@43 1393 zi->ci.stream.total_in = 0;
Chris@43 1394 }
Chris@43 1395
Chris@43 1396
Chris@43 1397 zi->ci.pos_in_buffered_data = 0;
Chris@43 1398
Chris@43 1399 return err;
Chris@43 1400 }
Chris@43 1401
Chris@43 1402 extern int ZEXPORT zipWriteInFileInZip (zipFile file,const void* buf,unsigned int len)
Chris@43 1403 {
Chris@43 1404 zip64_internal* zi;
Chris@43 1405 int err=ZIP_OK;
Chris@43 1406
Chris@43 1407 if (file == NULL)
Chris@43 1408 return ZIP_PARAMERROR;
Chris@43 1409 zi = (zip64_internal*)file;
Chris@43 1410
Chris@43 1411 if (zi->in_opened_file_inzip == 0)
Chris@43 1412 return ZIP_PARAMERROR;
Chris@43 1413
Chris@43 1414 zi->ci.crc32 = crc32(zi->ci.crc32,buf,(uInt)len);
Chris@43 1415
Chris@43 1416 #ifdef HAVE_BZIP2
Chris@43 1417 if(zi->ci.method == Z_BZIP2ED && (!zi->ci.raw))
Chris@43 1418 {
Chris@43 1419 zi->ci.bstream.next_in = (void*)buf;
Chris@43 1420 zi->ci.bstream.avail_in = len;
Chris@43 1421 err = BZ_RUN_OK;
Chris@43 1422
Chris@43 1423 while ((err==BZ_RUN_OK) && (zi->ci.bstream.avail_in>0))
Chris@43 1424 {
Chris@43 1425 if (zi->ci.bstream.avail_out == 0)
Chris@43 1426 {
Chris@43 1427 if (zip64FlushWriteBuffer(zi) == ZIP_ERRNO)
Chris@43 1428 err = ZIP_ERRNO;
Chris@43 1429 zi->ci.bstream.avail_out = (uInt)Z_BUFSIZE;
Chris@43 1430 zi->ci.bstream.next_out = (char*)zi->ci.buffered_data;
Chris@43 1431 }
Chris@43 1432
Chris@43 1433
Chris@43 1434 if(err != BZ_RUN_OK)
Chris@43 1435 break;
Chris@43 1436
Chris@43 1437 if ((zi->ci.method == Z_BZIP2ED) && (!zi->ci.raw))
Chris@43 1438 {
Chris@43 1439 uLong uTotalOutBefore_lo = zi->ci.bstream.total_out_lo32;
Chris@43 1440 // uLong uTotalOutBefore_hi = zi->ci.bstream.total_out_hi32;
Chris@43 1441 err=BZ2_bzCompress(&zi->ci.bstream, BZ_RUN);
Chris@43 1442
Chris@43 1443 zi->ci.pos_in_buffered_data += (uInt)(zi->ci.bstream.total_out_lo32 - uTotalOutBefore_lo) ;
Chris@43 1444 }
Chris@43 1445 }
Chris@43 1446
Chris@43 1447 if(err == BZ_RUN_OK)
Chris@43 1448 err = ZIP_OK;
Chris@43 1449 }
Chris@43 1450 else
Chris@43 1451 #endif
Chris@43 1452 {
Chris@43 1453 zi->ci.stream.next_in = (Bytef*)buf;
Chris@43 1454 zi->ci.stream.avail_in = len;
Chris@43 1455
Chris@43 1456 while ((err==ZIP_OK) && (zi->ci.stream.avail_in>0))
Chris@43 1457 {
Chris@43 1458 if (zi->ci.stream.avail_out == 0)
Chris@43 1459 {
Chris@43 1460 if (zip64FlushWriteBuffer(zi) == ZIP_ERRNO)
Chris@43 1461 err = ZIP_ERRNO;
Chris@43 1462 zi->ci.stream.avail_out = (uInt)Z_BUFSIZE;
Chris@43 1463 zi->ci.stream.next_out = zi->ci.buffered_data;
Chris@43 1464 }
Chris@43 1465
Chris@43 1466
Chris@43 1467 if(err != ZIP_OK)
Chris@43 1468 break;
Chris@43 1469
Chris@43 1470 if ((zi->ci.method == Z_DEFLATED) && (!zi->ci.raw))
Chris@43 1471 {
Chris@43 1472 uLong uTotalOutBefore = zi->ci.stream.total_out;
Chris@43 1473 err=deflate(&zi->ci.stream, Z_NO_FLUSH);
Chris@43 1474 if(uTotalOutBefore > zi->ci.stream.total_out)
Chris@43 1475 {
Chris@43 1476 int bBreak = 0;
Chris@43 1477 bBreak++;
Chris@43 1478 }
Chris@43 1479
Chris@43 1480 zi->ci.pos_in_buffered_data += (uInt)(zi->ci.stream.total_out - uTotalOutBefore) ;
Chris@43 1481 }
Chris@43 1482 else
Chris@43 1483 {
Chris@43 1484 uInt copy_this,i;
Chris@43 1485 if (zi->ci.stream.avail_in < zi->ci.stream.avail_out)
Chris@43 1486 copy_this = zi->ci.stream.avail_in;
Chris@43 1487 else
Chris@43 1488 copy_this = zi->ci.stream.avail_out;
Chris@43 1489
Chris@43 1490 for (i = 0; i < copy_this; i++)
Chris@43 1491 *(((char*)zi->ci.stream.next_out)+i) =
Chris@43 1492 *(((const char*)zi->ci.stream.next_in)+i);
Chris@43 1493 {
Chris@43 1494 zi->ci.stream.avail_in -= copy_this;
Chris@43 1495 zi->ci.stream.avail_out-= copy_this;
Chris@43 1496 zi->ci.stream.next_in+= copy_this;
Chris@43 1497 zi->ci.stream.next_out+= copy_this;
Chris@43 1498 zi->ci.stream.total_in+= copy_this;
Chris@43 1499 zi->ci.stream.total_out+= copy_this;
Chris@43 1500 zi->ci.pos_in_buffered_data += copy_this;
Chris@43 1501 }
Chris@43 1502 }
Chris@43 1503 }// while(...)
Chris@43 1504 }
Chris@43 1505
Chris@43 1506 return err;
Chris@43 1507 }
Chris@43 1508
Chris@43 1509 extern int ZEXPORT zipCloseFileInZipRaw (zipFile file, uLong uncompressed_size, uLong crc32)
Chris@43 1510 {
Chris@43 1511 return zipCloseFileInZipRaw64 (file, uncompressed_size, crc32);
Chris@43 1512 }
Chris@43 1513
Chris@43 1514 extern int ZEXPORT zipCloseFileInZipRaw64 (zipFile file, ZPOS64_T uncompressed_size, uLong crc32)
Chris@43 1515 {
Chris@43 1516 zip64_internal* zi;
Chris@43 1517 ZPOS64_T compressed_size;
Chris@43 1518 uLong invalidValue = 0xffffffff;
Chris@43 1519 short datasize = 0;
Chris@43 1520 int err=ZIP_OK;
Chris@43 1521
Chris@43 1522 if (file == NULL)
Chris@43 1523 return ZIP_PARAMERROR;
Chris@43 1524 zi = (zip64_internal*)file;
Chris@43 1525
Chris@43 1526 if (zi->in_opened_file_inzip == 0)
Chris@43 1527 return ZIP_PARAMERROR;
Chris@43 1528 zi->ci.stream.avail_in = 0;
Chris@43 1529
Chris@43 1530 if ((zi->ci.method == Z_DEFLATED) && (!zi->ci.raw))
Chris@43 1531 {
Chris@43 1532 while (err==ZIP_OK)
Chris@43 1533 {
Chris@43 1534 uLong uTotalOutBefore;
Chris@43 1535 if (zi->ci.stream.avail_out == 0)
Chris@43 1536 {
Chris@43 1537 if (zip64FlushWriteBuffer(zi) == ZIP_ERRNO)
Chris@43 1538 err = ZIP_ERRNO;
Chris@43 1539 zi->ci.stream.avail_out = (uInt)Z_BUFSIZE;
Chris@43 1540 zi->ci.stream.next_out = zi->ci.buffered_data;
Chris@43 1541 }
Chris@43 1542 uTotalOutBefore = zi->ci.stream.total_out;
Chris@43 1543 err=deflate(&zi->ci.stream, Z_FINISH);
Chris@43 1544 zi->ci.pos_in_buffered_data += (uInt)(zi->ci.stream.total_out - uTotalOutBefore) ;
Chris@43 1545 }
Chris@43 1546 }
Chris@43 1547 else if ((zi->ci.method == Z_BZIP2ED) && (!zi->ci.raw))
Chris@43 1548 {
Chris@43 1549 #ifdef HAVE_BZIP2
Chris@43 1550 err = BZ_FINISH_OK;
Chris@43 1551 while (err==BZ_FINISH_OK)
Chris@43 1552 {
Chris@43 1553 uLong uTotalOutBefore;
Chris@43 1554 if (zi->ci.bstream.avail_out == 0)
Chris@43 1555 {
Chris@43 1556 if (zip64FlushWriteBuffer(zi) == ZIP_ERRNO)
Chris@43 1557 err = ZIP_ERRNO;
Chris@43 1558 zi->ci.bstream.avail_out = (uInt)Z_BUFSIZE;
Chris@43 1559 zi->ci.bstream.next_out = (char*)zi->ci.buffered_data;
Chris@43 1560 }
Chris@43 1561 uTotalOutBefore = zi->ci.bstream.total_out_lo32;
Chris@43 1562 err=BZ2_bzCompress(&zi->ci.bstream, BZ_FINISH);
Chris@43 1563 if(err == BZ_STREAM_END)
Chris@43 1564 err = Z_STREAM_END;
Chris@43 1565
Chris@43 1566 zi->ci.pos_in_buffered_data += (uInt)(zi->ci.bstream.total_out_lo32 - uTotalOutBefore);
Chris@43 1567 }
Chris@43 1568
Chris@43 1569 if(err == BZ_FINISH_OK)
Chris@43 1570 err = ZIP_OK;
Chris@43 1571 #endif
Chris@43 1572 }
Chris@43 1573
Chris@43 1574 if (err==Z_STREAM_END)
Chris@43 1575 err=ZIP_OK; /* this is normal */
Chris@43 1576
Chris@43 1577 if ((zi->ci.pos_in_buffered_data>0) && (err==ZIP_OK))
Chris@43 1578 {
Chris@43 1579 if (zip64FlushWriteBuffer(zi)==ZIP_ERRNO)
Chris@43 1580 err = ZIP_ERRNO;
Chris@43 1581 }
Chris@43 1582
Chris@43 1583 if ((zi->ci.method == Z_DEFLATED) && (!zi->ci.raw))
Chris@43 1584 {
Chris@43 1585 int tmp_err = deflateEnd(&zi->ci.stream);
Chris@43 1586 if (err == ZIP_OK)
Chris@43 1587 err = tmp_err;
Chris@43 1588 zi->ci.stream_initialised = 0;
Chris@43 1589 }
Chris@43 1590 #ifdef HAVE_BZIP2
Chris@43 1591 else if((zi->ci.method == Z_BZIP2ED) && (!zi->ci.raw))
Chris@43 1592 {
Chris@43 1593 int tmperr = BZ2_bzCompressEnd(&zi->ci.bstream);
Chris@43 1594 if (err==ZIP_OK)
Chris@43 1595 err = tmperr;
Chris@43 1596 zi->ci.stream_initialised = 0;
Chris@43 1597 }
Chris@43 1598 #endif
Chris@43 1599
Chris@43 1600 if (!zi->ci.raw)
Chris@43 1601 {
Chris@43 1602 crc32 = (uLong)zi->ci.crc32;
Chris@43 1603 uncompressed_size = zi->ci.totalUncompressedData;
Chris@43 1604 }
Chris@43 1605 compressed_size = zi->ci.totalCompressedData;
Chris@43 1606
Chris@43 1607 # ifndef NOCRYPT
Chris@43 1608 compressed_size += zi->ci.crypt_header_size;
Chris@43 1609 # endif
Chris@43 1610
Chris@43 1611 // update Current Item crc and sizes,
Chris@43 1612 if(compressed_size >= 0xffffffff || uncompressed_size >= 0xffffffff || zi->ci.pos_local_header >= 0xffffffff)
Chris@43 1613 {
Chris@43 1614 /*version Made by*/
Chris@43 1615 zip64local_putValue_inmemory(zi->ci.central_header+4,(uLong)45,2);
Chris@43 1616 /*version needed*/
Chris@43 1617 zip64local_putValue_inmemory(zi->ci.central_header+6,(uLong)45,2);
Chris@43 1618
Chris@43 1619 }
Chris@43 1620
Chris@43 1621 zip64local_putValue_inmemory(zi->ci.central_header+16,crc32,4); /*crc*/
Chris@43 1622
Chris@43 1623
Chris@43 1624 if(compressed_size >= 0xffffffff)
Chris@43 1625 zip64local_putValue_inmemory(zi->ci.central_header+20, invalidValue,4); /*compr size*/
Chris@43 1626 else
Chris@43 1627 zip64local_putValue_inmemory(zi->ci.central_header+20, compressed_size,4); /*compr size*/
Chris@43 1628
Chris@43 1629 /// set internal file attributes field
Chris@43 1630 if (zi->ci.stream.data_type == Z_ASCII)
Chris@43 1631 zip64local_putValue_inmemory(zi->ci.central_header+36,(uLong)Z_ASCII,2);
Chris@43 1632
Chris@43 1633 if(uncompressed_size >= 0xffffffff)
Chris@43 1634 zip64local_putValue_inmemory(zi->ci.central_header+24, invalidValue,4); /*uncompr size*/
Chris@43 1635 else
Chris@43 1636 zip64local_putValue_inmemory(zi->ci.central_header+24, uncompressed_size,4); /*uncompr size*/
Chris@43 1637
Chris@43 1638 // Add ZIP64 extra info field for uncompressed size
Chris@43 1639 if(uncompressed_size >= 0xffffffff)
Chris@43 1640 datasize += 8;
Chris@43 1641
Chris@43 1642 // Add ZIP64 extra info field for compressed size
Chris@43 1643 if(compressed_size >= 0xffffffff)
Chris@43 1644 datasize += 8;
Chris@43 1645
Chris@43 1646 // Add ZIP64 extra info field for relative offset to local file header of current file
Chris@43 1647 if(zi->ci.pos_local_header >= 0xffffffff)
Chris@43 1648 datasize += 8;
Chris@43 1649
Chris@43 1650 if(datasize > 0)
Chris@43 1651 {
Chris@43 1652 char* p = NULL;
Chris@43 1653
Chris@43 1654 if((uLong)(datasize + 4) > zi->ci.size_centralExtraFree)
Chris@43 1655 {
Chris@43 1656 // we can not write more data to the buffer that we have room for.
Chris@43 1657 return ZIP_BADZIPFILE;
Chris@43 1658 }
Chris@43 1659
Chris@43 1660 p = zi->ci.central_header + zi->ci.size_centralheader;
Chris@43 1661
Chris@43 1662 // Add Extra Information Header for 'ZIP64 information'
Chris@43 1663 zip64local_putValue_inmemory(p, 0x0001, 2); // HeaderID
Chris@43 1664 p += 2;
Chris@43 1665 zip64local_putValue_inmemory(p, datasize, 2); // DataSize
Chris@43 1666 p += 2;
Chris@43 1667
Chris@43 1668 if(uncompressed_size >= 0xffffffff)
Chris@43 1669 {
Chris@43 1670 zip64local_putValue_inmemory(p, uncompressed_size, 8);
Chris@43 1671 p += 8;
Chris@43 1672 }
Chris@43 1673
Chris@43 1674 if(compressed_size >= 0xffffffff)
Chris@43 1675 {
Chris@43 1676 zip64local_putValue_inmemory(p, compressed_size, 8);
Chris@43 1677 p += 8;
Chris@43 1678 }
Chris@43 1679
Chris@43 1680 if(zi->ci.pos_local_header >= 0xffffffff)
Chris@43 1681 {
Chris@43 1682 zip64local_putValue_inmemory(p, zi->ci.pos_local_header, 8);
Chris@43 1683 p += 8;
Chris@43 1684 }
Chris@43 1685
Chris@43 1686 // Update how much extra free space we got in the memory buffer
Chris@43 1687 // and increase the centralheader size so the new ZIP64 fields are included
Chris@43 1688 // ( 4 below is the size of HeaderID and DataSize field )
Chris@43 1689 zi->ci.size_centralExtraFree -= datasize + 4;
Chris@43 1690 zi->ci.size_centralheader += datasize + 4;
Chris@43 1691
Chris@43 1692 // Update the extra info size field
Chris@43 1693 zi->ci.size_centralExtra += datasize + 4;
Chris@43 1694 zip64local_putValue_inmemory(zi->ci.central_header+30,(uLong)zi->ci.size_centralExtra,2);
Chris@43 1695 }
Chris@43 1696
Chris@43 1697 if (err==ZIP_OK)
Chris@43 1698 err = add_data_in_datablock(&zi->central_dir, zi->ci.central_header, (uLong)zi->ci.size_centralheader);
Chris@43 1699
Chris@43 1700 free(zi->ci.central_header);
Chris@43 1701
Chris@43 1702 if (err==ZIP_OK)
Chris@43 1703 {
Chris@43 1704 // Update the LocalFileHeader with the new values.
Chris@43 1705
Chris@43 1706 ZPOS64_T cur_pos_inzip = ZTELL64(zi->z_filefunc,zi->filestream);
Chris@43 1707
Chris@43 1708 if (ZSEEK64(zi->z_filefunc,zi->filestream, zi->ci.pos_local_header + 14,ZLIB_FILEFUNC_SEEK_SET)!=0)
Chris@43 1709 err = ZIP_ERRNO;
Chris@43 1710
Chris@43 1711 if (err==ZIP_OK)
Chris@43 1712 err = zip64local_putValue(&zi->z_filefunc,zi->filestream,crc32,4); /* crc 32, unknown */
Chris@43 1713
Chris@43 1714 if(uncompressed_size >= 0xffffffff || compressed_size >= 0xffffffff )
Chris@43 1715 {
Chris@43 1716 if(zi->ci.pos_zip64extrainfo > 0)
Chris@43 1717 {
Chris@43 1718 // Update the size in the ZIP64 extended field.
Chris@43 1719 if (ZSEEK64(zi->z_filefunc,zi->filestream, zi->ci.pos_zip64extrainfo + 4,ZLIB_FILEFUNC_SEEK_SET)!=0)
Chris@43 1720 err = ZIP_ERRNO;
Chris@43 1721
Chris@43 1722 if (err==ZIP_OK) /* compressed size, unknown */
Chris@43 1723 err = zip64local_putValue(&zi->z_filefunc, zi->filestream, uncompressed_size, 8);
Chris@43 1724
Chris@43 1725 if (err==ZIP_OK) /* uncompressed size, unknown */
Chris@43 1726 err = zip64local_putValue(&zi->z_filefunc, zi->filestream, compressed_size, 8);
Chris@43 1727 }
Chris@43 1728 else
Chris@43 1729 err = ZIP_BADZIPFILE; // Caller passed zip64 = 0, so no room for zip64 info -> fatal
Chris@43 1730 }
Chris@43 1731 else
Chris@43 1732 {
Chris@43 1733 if (err==ZIP_OK) /* compressed size, unknown */
Chris@43 1734 err = zip64local_putValue(&zi->z_filefunc,zi->filestream,compressed_size,4);
Chris@43 1735
Chris@43 1736 if (err==ZIP_OK) /* uncompressed size, unknown */
Chris@43 1737 err = zip64local_putValue(&zi->z_filefunc,zi->filestream,uncompressed_size,4);
Chris@43 1738 }
Chris@43 1739
Chris@43 1740 if (ZSEEK64(zi->z_filefunc,zi->filestream, cur_pos_inzip,ZLIB_FILEFUNC_SEEK_SET)!=0)
Chris@43 1741 err = ZIP_ERRNO;
Chris@43 1742 }
Chris@43 1743
Chris@43 1744 zi->number_entry ++;
Chris@43 1745 zi->in_opened_file_inzip = 0;
Chris@43 1746
Chris@43 1747 return err;
Chris@43 1748 }
Chris@43 1749
Chris@43 1750 extern int ZEXPORT zipCloseFileInZip (zipFile file)
Chris@43 1751 {
Chris@43 1752 return zipCloseFileInZipRaw (file,0,0);
Chris@43 1753 }
Chris@43 1754
Chris@43 1755 int Write_Zip64EndOfCentralDirectoryLocator(zip64_internal* zi, ZPOS64_T zip64eocd_pos_inzip)
Chris@43 1756 {
Chris@43 1757 int err = ZIP_OK;
Chris@43 1758 ZPOS64_T pos = zip64eocd_pos_inzip - zi->add_position_when_writting_offset;
Chris@43 1759
Chris@43 1760 err = zip64local_putValue(&zi->z_filefunc,zi->filestream,(uLong)ZIP64ENDLOCHEADERMAGIC,4);
Chris@43 1761
Chris@43 1762 /*num disks*/
Chris@43 1763 if (err==ZIP_OK) /* number of the disk with the start of the central directory */
Chris@43 1764 err = zip64local_putValue(&zi->z_filefunc,zi->filestream,(uLong)0,4);
Chris@43 1765
Chris@43 1766 /*relative offset*/
Chris@43 1767 if (err==ZIP_OK) /* Relative offset to the Zip64EndOfCentralDirectory */
Chris@43 1768 err = zip64local_putValue(&zi->z_filefunc,zi->filestream, pos,8);
Chris@43 1769
Chris@43 1770 /*total disks*/ /* Do not support spawning of disk so always say 1 here*/
Chris@43 1771 if (err==ZIP_OK) /* number of the disk with the start of the central directory */
Chris@43 1772 err = zip64local_putValue(&zi->z_filefunc,zi->filestream,(uLong)1,4);
Chris@43 1773
Chris@43 1774 return err;
Chris@43 1775 }
Chris@43 1776
Chris@43 1777 int Write_Zip64EndOfCentralDirectoryRecord(zip64_internal* zi, uLong size_centraldir, ZPOS64_T centraldir_pos_inzip)
Chris@43 1778 {
Chris@43 1779 int err = ZIP_OK;
Chris@43 1780
Chris@43 1781 uLong Zip64DataSize = 44;
Chris@43 1782
Chris@43 1783 err = zip64local_putValue(&zi->z_filefunc,zi->filestream,(uLong)ZIP64ENDHEADERMAGIC,4);
Chris@43 1784
Chris@43 1785 if (err==ZIP_OK) /* size of this 'zip64 end of central directory' */
Chris@43 1786 err = zip64local_putValue(&zi->z_filefunc,zi->filestream,(ZPOS64_T)Zip64DataSize,8); // why ZPOS64_T of this ?
Chris@43 1787
Chris@43 1788 if (err==ZIP_OK) /* version made by */
Chris@43 1789 err = zip64local_putValue(&zi->z_filefunc,zi->filestream,(uLong)45,2);
Chris@43 1790
Chris@43 1791 if (err==ZIP_OK) /* version needed */
Chris@43 1792 err = zip64local_putValue(&zi->z_filefunc,zi->filestream,(uLong)45,2);
Chris@43 1793
Chris@43 1794 if (err==ZIP_OK) /* number of this disk */
Chris@43 1795 err = zip64local_putValue(&zi->z_filefunc,zi->filestream,(uLong)0,4);
Chris@43 1796
Chris@43 1797 if (err==ZIP_OK) /* number of the disk with the start of the central directory */
Chris@43 1798 err = zip64local_putValue(&zi->z_filefunc,zi->filestream,(uLong)0,4);
Chris@43 1799
Chris@43 1800 if (err==ZIP_OK) /* total number of entries in the central dir on this disk */
Chris@43 1801 err = zip64local_putValue(&zi->z_filefunc, zi->filestream, zi->number_entry, 8);
Chris@43 1802
Chris@43 1803 if (err==ZIP_OK) /* total number of entries in the central dir */
Chris@43 1804 err = zip64local_putValue(&zi->z_filefunc, zi->filestream, zi->number_entry, 8);
Chris@43 1805
Chris@43 1806 if (err==ZIP_OK) /* size of the central directory */
Chris@43 1807 err = zip64local_putValue(&zi->z_filefunc,zi->filestream,(ZPOS64_T)size_centraldir,8);
Chris@43 1808
Chris@43 1809 if (err==ZIP_OK) /* offset of start of central directory with respect to the starting disk number */
Chris@43 1810 {
Chris@43 1811 ZPOS64_T pos = centraldir_pos_inzip - zi->add_position_when_writting_offset;
Chris@43 1812 err = zip64local_putValue(&zi->z_filefunc,zi->filestream, (ZPOS64_T)pos,8);
Chris@43 1813 }
Chris@43 1814 return err;
Chris@43 1815 }
Chris@43 1816 int Write_EndOfCentralDirectoryRecord(zip64_internal* zi, uLong size_centraldir, ZPOS64_T centraldir_pos_inzip)
Chris@43 1817 {
Chris@43 1818 int err = ZIP_OK;
Chris@43 1819
Chris@43 1820 /*signature*/
Chris@43 1821 err = zip64local_putValue(&zi->z_filefunc,zi->filestream,(uLong)ENDHEADERMAGIC,4);
Chris@43 1822
Chris@43 1823 if (err==ZIP_OK) /* number of this disk */
Chris@43 1824 err = zip64local_putValue(&zi->z_filefunc,zi->filestream,(uLong)0,2);
Chris@43 1825
Chris@43 1826 if (err==ZIP_OK) /* number of the disk with the start of the central directory */
Chris@43 1827 err = zip64local_putValue(&zi->z_filefunc,zi->filestream,(uLong)0,2);
Chris@43 1828
Chris@43 1829 if (err==ZIP_OK) /* total number of entries in the central dir on this disk */
Chris@43 1830 {
Chris@43 1831 {
Chris@43 1832 if(zi->number_entry >= 0xFFFF)
Chris@43 1833 err = zip64local_putValue(&zi->z_filefunc,zi->filestream,(uLong)0xffff,2); // use value in ZIP64 record
Chris@43 1834 else
Chris@43 1835 err = zip64local_putValue(&zi->z_filefunc,zi->filestream,(uLong)zi->number_entry,2);
Chris@43 1836 }
Chris@43 1837 }
Chris@43 1838
Chris@43 1839 if (err==ZIP_OK) /* total number of entries in the central dir */
Chris@43 1840 {
Chris@43 1841 if(zi->number_entry >= 0xFFFF)
Chris@43 1842 err = zip64local_putValue(&zi->z_filefunc,zi->filestream,(uLong)0xffff,2); // use value in ZIP64 record
Chris@43 1843 else
Chris@43 1844 err = zip64local_putValue(&zi->z_filefunc,zi->filestream,(uLong)zi->number_entry,2);
Chris@43 1845 }
Chris@43 1846
Chris@43 1847 if (err==ZIP_OK) /* size of the central directory */
Chris@43 1848 err = zip64local_putValue(&zi->z_filefunc,zi->filestream,(uLong)size_centraldir,4);
Chris@43 1849
Chris@43 1850 if (err==ZIP_OK) /* offset of start of central directory with respect to the starting disk number */
Chris@43 1851 {
Chris@43 1852 ZPOS64_T pos = centraldir_pos_inzip - zi->add_position_when_writting_offset;
Chris@43 1853 if(pos >= 0xffffffff)
Chris@43 1854 {
Chris@43 1855 err = zip64local_putValue(&zi->z_filefunc,zi->filestream, (uLong)0xffffffff,4);
Chris@43 1856 }
Chris@43 1857 else
Chris@43 1858 err = zip64local_putValue(&zi->z_filefunc,zi->filestream, (uLong)(centraldir_pos_inzip - zi->add_position_when_writting_offset),4);
Chris@43 1859 }
Chris@43 1860
Chris@43 1861 return err;
Chris@43 1862 }
Chris@43 1863
Chris@43 1864 int Write_GlobalComment(zip64_internal* zi, const char* global_comment)
Chris@43 1865 {
Chris@43 1866 int err = ZIP_OK;
Chris@43 1867 uInt size_global_comment = 0;
Chris@43 1868
Chris@43 1869 if(global_comment != NULL)
Chris@43 1870 size_global_comment = (uInt)strlen(global_comment);
Chris@43 1871
Chris@43 1872 err = zip64local_putValue(&zi->z_filefunc,zi->filestream,(uLong)size_global_comment,2);
Chris@43 1873
Chris@43 1874 if (err == ZIP_OK && size_global_comment > 0)
Chris@43 1875 {
Chris@43 1876 if (ZWRITE64(zi->z_filefunc,zi->filestream, global_comment, size_global_comment) != size_global_comment)
Chris@43 1877 err = ZIP_ERRNO;
Chris@43 1878 }
Chris@43 1879 return err;
Chris@43 1880 }
Chris@43 1881
Chris@43 1882 extern int ZEXPORT zipClose (zipFile file, const char* global_comment)
Chris@43 1883 {
Chris@43 1884 zip64_internal* zi;
Chris@43 1885 int err = 0;
Chris@43 1886 uLong size_centraldir = 0;
Chris@43 1887 ZPOS64_T centraldir_pos_inzip;
Chris@43 1888 ZPOS64_T pos;
Chris@43 1889
Chris@43 1890 if (file == NULL)
Chris@43 1891 return ZIP_PARAMERROR;
Chris@43 1892
Chris@43 1893 zi = (zip64_internal*)file;
Chris@43 1894
Chris@43 1895 if (zi->in_opened_file_inzip == 1)
Chris@43 1896 {
Chris@43 1897 err = zipCloseFileInZip (file);
Chris@43 1898 }
Chris@43 1899
Chris@43 1900 #ifndef NO_ADDFILEINEXISTINGZIP
Chris@43 1901 if (global_comment==NULL)
Chris@43 1902 global_comment = zi->globalcomment;
Chris@43 1903 #endif
Chris@43 1904
Chris@43 1905 centraldir_pos_inzip = ZTELL64(zi->z_filefunc,zi->filestream);
Chris@43 1906
Chris@43 1907 if (err==ZIP_OK)
Chris@43 1908 {
Chris@43 1909 linkedlist_datablock_internal* ldi = zi->central_dir.first_block;
Chris@43 1910 while (ldi!=NULL)
Chris@43 1911 {
Chris@43 1912 if ((err==ZIP_OK) && (ldi->filled_in_this_block>0))
Chris@43 1913 {
Chris@43 1914 if (ZWRITE64(zi->z_filefunc,zi->filestream, ldi->data, ldi->filled_in_this_block) != ldi->filled_in_this_block)
Chris@43 1915 err = ZIP_ERRNO;
Chris@43 1916 }
Chris@43 1917
Chris@43 1918 size_centraldir += ldi->filled_in_this_block;
Chris@43 1919 ldi = ldi->next_datablock;
Chris@43 1920 }
Chris@43 1921 }
Chris@43 1922 free_linkedlist(&(zi->central_dir));
Chris@43 1923
Chris@43 1924 pos = centraldir_pos_inzip - zi->add_position_when_writting_offset;
Chris@43 1925 if(pos >= 0xffffffff || zi->number_entry > 0xFFFF)
Chris@43 1926 {
Chris@43 1927 ZPOS64_T Zip64EOCDpos = ZTELL64(zi->z_filefunc,zi->filestream);
Chris@43 1928 Write_Zip64EndOfCentralDirectoryRecord(zi, size_centraldir, centraldir_pos_inzip);
Chris@43 1929
Chris@43 1930 Write_Zip64EndOfCentralDirectoryLocator(zi, Zip64EOCDpos);
Chris@43 1931 }
Chris@43 1932
Chris@43 1933 if (err==ZIP_OK)
Chris@43 1934 err = Write_EndOfCentralDirectoryRecord(zi, size_centraldir, centraldir_pos_inzip);
Chris@43 1935
Chris@43 1936 if(err == ZIP_OK)
Chris@43 1937 err = Write_GlobalComment(zi, global_comment);
Chris@43 1938
Chris@43 1939 if (ZCLOSE64(zi->z_filefunc,zi->filestream) != 0)
Chris@43 1940 if (err == ZIP_OK)
Chris@43 1941 err = ZIP_ERRNO;
Chris@43 1942
Chris@43 1943 #ifndef NO_ADDFILEINEXISTINGZIP
Chris@43 1944 TRYFREE(zi->globalcomment);
Chris@43 1945 #endif
Chris@43 1946 TRYFREE(zi);
Chris@43 1947
Chris@43 1948 return err;
Chris@43 1949 }
Chris@43 1950
Chris@43 1951 extern int ZEXPORT zipRemoveExtraInfoBlock (char* pData, int* dataLen, short sHeader)
Chris@43 1952 {
Chris@43 1953 char* p = pData;
Chris@43 1954 int size = 0;
Chris@43 1955 char* pNewHeader;
Chris@43 1956 char* pTmp;
Chris@43 1957 short header;
Chris@43 1958 short dataSize;
Chris@43 1959
Chris@43 1960 int retVal = ZIP_OK;
Chris@43 1961
Chris@43 1962 if(pData == NULL || *dataLen < 4)
Chris@43 1963 return ZIP_PARAMERROR;
Chris@43 1964
Chris@43 1965 pNewHeader = (char*)ALLOC(*dataLen);
Chris@43 1966 pTmp = pNewHeader;
Chris@43 1967
Chris@43 1968 while(p < (pData + *dataLen))
Chris@43 1969 {
Chris@43 1970 header = *(short*)p;
Chris@43 1971 dataSize = *(((short*)p)+1);
Chris@43 1972
Chris@43 1973 if( header == sHeader ) // Header found.
Chris@43 1974 {
Chris@43 1975 p += dataSize + 4; // skip it. do not copy to temp buffer
Chris@43 1976 }
Chris@43 1977 else
Chris@43 1978 {
Chris@43 1979 // Extra Info block should not be removed, So copy it to the temp buffer.
Chris@43 1980 memcpy(pTmp, p, dataSize + 4);
Chris@43 1981 p += dataSize + 4;
Chris@43 1982 size += dataSize + 4;
Chris@43 1983 }
Chris@43 1984
Chris@43 1985 }
Chris@43 1986
Chris@43 1987 if(size < *dataLen)
Chris@43 1988 {
Chris@43 1989 // clean old extra info block.
Chris@43 1990 memset(pData,0, *dataLen);
Chris@43 1991
Chris@43 1992 // copy the new extra info block over the old
Chris@43 1993 if(size > 0)
Chris@43 1994 memcpy(pData, pNewHeader, size);
Chris@43 1995
Chris@43 1996 // set the new extra info size
Chris@43 1997 *dataLen = size;
Chris@43 1998
Chris@43 1999 retVal = ZIP_OK;
Chris@43 2000 }
Chris@43 2001 else
Chris@43 2002 retVal = ZIP_ERRNO;
Chris@43 2003
Chris@43 2004 TRYFREE(pNewHeader);
Chris@43 2005
Chris@43 2006 return retVal;
Chris@43 2007 }