annotate src/zlib-1.2.8/contrib/minizip/zip.c @ 128:5b4145a0d408

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