Fix remaining bugs, and ensure correct calling sequences for source functions.

All tests (and all ckmame tests) pass.

--HG--
branch : HEAD
diff --git a/lib/zip.h b/lib/zip.h
index 4863e22..dbc719c 100644
--- a/lib/zip.h
+++ b/lib/zip.h
@@ -255,7 +255,7 @@
 				    const char *, int);
 ZIP_EXTERN struct zip_source *zip_source_buffer(struct zip *, const void *,
 						zip_uint64_t, int);
-ZIP_EXTERN int zip_source_close(struct zip_source *);
+ZIP_EXTERN void zip_source_close(struct zip_source *);
 ZIP_EXTERN struct zip_source *zip_source_crc(struct zip *, struct zip_source *,
 					     int);
 ZIP_EXTERN struct zip_source *zip_source_deflate(struct zip *,
diff --git a/lib/zip_close.c b/lib/zip_close.c
index f6474b9..7e9d187 100644
--- a/lib/zip_close.c
+++ b/lib/zip_close.c
@@ -45,8 +45,7 @@
 static int add_data(struct zip *, struct zip_source *, struct zip_dirent *,
 		    FILE *);
 static int copy_data(FILE *, off_t, FILE *, struct zip_error *);
-static int copy_source(struct zip *, struct zip_source *, struct zip_stat *,
-		       FILE *);
+static int copy_source(struct zip *, struct zip_source *, FILE *);
 static int write_cdir(struct zip *, struct zip_cdir *, FILE *);
 static int _zip_cdir_set_comment(struct zip_cdir *, struct zip *);
 static char *_zip_create_temp_output(struct zip *, FILE **);
@@ -318,7 +317,7 @@
 add_data(struct zip *za, struct zip_source *src, struct zip_dirent *de,
 	 FILE *ft)
 {
-    off_t offstart, offend;
+    off_t offstart, offdata, offend;
     struct zip_stat st;
     struct zip_source *s2;
     zip_compression_implementation comp_impl;
@@ -334,6 +333,11 @@
     if (_zip_dirent_write(de, ft, 1, &za->error) < 0)
 	return -1;
 
+    if ((s2=zip_source_crc(za, src, 0)) == NULL) {
+	zip_source_pop(s2);
+	return -1;
+    }
+    
     /* XXX: deflate 0-byte files for torrentzip? */
     if (((st.valid & ZIP_STAT_COMP_METHOD) == 0
 	 || st.comp_method == ZIP_CM_STORE)
@@ -342,20 +346,26 @@
 	if ((comp_impl=zip_get_compression_implementation(ZIP_CM_DEFLATE))
 	    == NULL) {
 	    _zip_error_set(&za->error, ZIP_ER_COMPNOTSUPP, 0);
+	    zip_source_pop(s2);
 	    return -1;
 	}
-	if ((s2=comp_impl(za, src, ZIP_CM_DEFLATE, ZIP_CODEC_ENCODE))
+	if ((s2=comp_impl(za, s2, ZIP_CM_DEFLATE, ZIP_CODEC_ENCODE))
 	    == NULL) {
 	    /* XXX: set error? */
-	    zip_source_free(src);
+	    zip_source_pop(s2);
 	    return -1;
 	}
     }
     else
 	s2 = src;
 
-    ret = copy_source(za, s2, &st, ft);
-
+    offdata = ftello(ft);
+	
+    ret = copy_source(za, s2, ft);
+	
+    if (zip_source_stat(s2, &st) < 0)
+	ret = -1;
+    
     while (s2 != src) {
 	if ((s2=zip_source_pop(s2)) == NULL) {
 	    /* XXX: set erorr */
@@ -367,7 +377,6 @@
     if (ret < 0)
 	return -1;
 
-
     offend = ftello(ft);
 
     if (fseeko(ft, offstart, SEEK_SET) < 0) {
@@ -375,12 +384,11 @@
 	return -1;
     }
 
-    
     de->last_mod = st.mtime;
     de->comp_method = st.comp_method;
     de->crc = st.crc;
     de->uncomp_size = st.size;
-    de->comp_size = st.comp_size;
+    de->comp_size = offend - offdata;
 
     if (zip_get_archive_flag(za, ZIP_AFL_TORRENT, 0))
 	_zip_dirent_torrent_normalize(de);
@@ -432,8 +440,7 @@
 
 
 static int
-copy_source(struct zip *za, struct zip_source *src, struct zip_stat *st,
-	    FILE *ft)
+copy_source(struct zip *za, struct zip_source *src, FILE *ft)
 {
     char buf[BUFSIZE];
     zip_int64_t n;
@@ -444,7 +451,6 @@
 	return -1;
     }
 
-    st->comp_size = 0;
     ret = 0;
     while ((n=zip_source_read(src, buf, sizeof(buf))) > 0) {
 	if (fwrite(buf, 1, n, ft) != (size_t)n) {
@@ -452,8 +458,6 @@
 	    ret = -1;
 	    break;
 	}
-	
-	st->comp_size += n;
     }
     
     if (n < 0) {
@@ -462,11 +466,7 @@
 	ret = -1;
     }	
 
-    if (zip_source_close(src) < 0) {
-	if (ret == 0)
-	    _zip_error_set_from_source(&za->error, src);
-	ret = -1;
-    }
+    zip_source_close(src);
     
     return ret;
 }
diff --git a/lib/zip_source_close.c b/lib/zip_source_close.c
index a889b97..882a199 100644
--- a/lib/zip_source_close.c
+++ b/lib/zip_source_close.c
@@ -37,28 +37,18 @@
 
 
 
-ZIP_EXTERN int
+ZIP_EXTERN void
 zip_source_close(struct zip_source *src)
 {
-    int ret;
-
-    /* XXX: Should we even allow ZIP_SOURCE_CLOSE to return an error? */
+    if (!src->is_open)
+	return;
 
     if (src->src == NULL)
-	return src->cb.f(src->ud, NULL, 0, ZIP_SOURCE_CLOSE);
-
-    ret =  src->cb.l(src->src, src->ud, NULL, 0, ZIP_SOURCE_CLOSE);
-    if (ret < 0) {
-	if (ret == ZIP_SOURCE_ERR_LOWER)
-	    src->error_source = ZIP_LES_LOWER;
-	else
-	    src->error_source = ZIP_LES_UPPER;
-    }
-
-    if (zip_source_close(src->src) < 0) {
-	src->error_source = ZIP_LES_LOWER;
-	return -1;
+	(void)src->cb.f(src->ud, NULL, 0, ZIP_SOURCE_CLOSE);
+    else {
+	(void)src->cb.l(src->src, src->ud, NULL, 0, ZIP_SOURCE_CLOSE);
+	zip_source_close(src->src);
     }
     
-    return ret < 0 ? -1 : 0;
+    src->is_open = 0;
 }
diff --git a/lib/zip_source_deflate.c b/lib/zip_source_deflate.c
index e349470..0da1940 100644
--- a/lib/zip_source_deflate.c
+++ b/lib/zip_source_deflate.c
@@ -105,7 +105,6 @@
 {
     int end, ret;
     zip_int64_t n;
-    zip_uint64_t out_before;
 
     if (ctx->e[0] != 0)
 	return -1;
@@ -115,7 +114,6 @@
 	
     ctx->zstr.next_out = (Bytef *)data;
     ctx->zstr.avail_out = len;
-    out_before = ctx->zstr.total_out;
 
     end = 0;
     while (!end) {
@@ -126,7 +124,8 @@
 	case Z_STREAM_END:
 	    /* all ok */
 
-	    if (ctx->zstr.total_out - out_before >= len)
+	    if (ctx->zstr.avail_out == 0
+		|| (ctx->eof && ctx->zstr.avail_in == 0))
 		end = 1;
 	    break;
 
@@ -167,8 +166,8 @@
 	}
     }
 
-    if (ctx->zstr.total_out > out_before)
-	return ctx->zstr.total_out - out_before;
+    if (ctx->zstr.avail_out < len)
+	return len - ctx->zstr.avail_out;
 
     return (ctx->e[0] == 0) ? 0 : -1;
 }
@@ -321,6 +320,9 @@
 
     switch (cmd) {
     case ZIP_SOURCE_OPEN:
+	if ((n=zip_source_read(src, ctx->buffer, sizeof(ctx->buffer))) < 0)
+	    return ZIP_SOURCE_ERR_LOWER;
+
 	ctx->zstr.zalloc = Z_NULL;
 	ctx->zstr.zfree = Z_NULL;
 	ctx->zstr.opaque = NULL;
diff --git a/lib/zip_source_free.c b/lib/zip_source_free.c
index 1e6ffe1..b1e7840 100644
--- a/lib/zip_source_free.c
+++ b/lib/zip_source_free.c
@@ -45,12 +45,15 @@
     if (src == NULL)
 	return;
 
-    if (src->src) {
+    if (src->is_open)
+	zip_source_close(src);
+
+    if (src->src == NULL)
+	(void)src->cb.f(src->ud, NULL, 0, ZIP_SOURCE_FREE);
+    else {
 	(void)src->cb.l(src->src, src->ud, NULL, 0, ZIP_SOURCE_FREE);
 	zip_source_free(src->src);
     }
-    else
-	(void)src->cb.f(src->ud, NULL, 0, ZIP_SOURCE_FREE);
 
     free(src);
 }
diff --git a/lib/zip_source_function.c b/lib/zip_source_function.c
index 0e75686..cb92e33 100644
--- a/lib/zip_source_function.c
+++ b/lib/zip_source_function.c
@@ -47,14 +47,32 @@
     if (za == NULL)
 	return NULL;
 
-    if ((zs=(struct zip_source *)malloc(sizeof(*zs))) == NULL) {
-	_zip_error_set(&za->error, ZIP_ER_MEMORY, 0);
+    if ((zs=_zip_source_new(za)) == NULL)
 	return NULL;
-    }
 
-    zs->src = NULL;
     zs->cb.f = zcb;
     zs->ud = ud;
     
     return zs;
 }
+
+
+
+struct zip_source *
+_zip_source_new(struct zip *za)
+{
+    struct zip_source *src;
+
+    if ((src=(struct zip_source *)malloc(sizeof(*src))) == NULL) {
+	_zip_error_set(&za->error, ZIP_ER_MEMORY, 0);
+	return NULL;
+    }
+
+    src->src = NULL;
+    src->cb.f = NULL;
+    src->ud = NULL;
+    src->error_source = ZIP_LES_NONE;
+    src->is_open = 0;
+
+    return src;
+}
diff --git a/lib/zip_source_layered.c b/lib/zip_source_layered.c
index 433c77a..9bdf3c5 100644
--- a/lib/zip_source_layered.c
+++ b/lib/zip_source_layered.c
@@ -48,15 +48,12 @@
     if (za == NULL)
 	return NULL;
 
-    if ((zs=(struct zip_source *)malloc(sizeof(*zs))) == NULL) {
-	_zip_error_set(&za->error, ZIP_ER_MEMORY, 0);
+    if ((zs=_zip_source_new(za)) == NULL)
 	return NULL;
-    }
 
     zs->src = src;
     zs->cb.l = cb;
     zs->ud = ud;
-    zs->error_source = ZIP_LES_NONE;
     
     return zs;
 }
diff --git a/lib/zip_source_open.c b/lib/zip_source_open.c
index 9923bda..a279e20 100644
--- a/lib/zip_source_open.c
+++ b/lib/zip_source_open.c
@@ -42,25 +42,35 @@
 {
     zip_int64_t ret;
 
-    if (src->src == NULL)
-	return src->cb.f(src->ud, NULL, 0, ZIP_SOURCE_OPEN);
-
-    if (zip_source_open(src->src) < 0) {
-	src->error_source = ZIP_LES_LOWER;
+    if (src->is_open) {
+	src->error_source = ZIP_LES_INVAL;
 	return -1;
     }
 
-    ret = src->cb.l(src->src, src->ud, NULL, 0, ZIP_SOURCE_OPEN);
-
-    if (ret < 0) {
-	(void)zip_source_close(src->src);
-
-	if (ret == ZIP_SOURCE_ERR_LOWER)
+    if (src->src == NULL) {
+	if (src->cb.f(src->ud, NULL, 0, ZIP_SOURCE_OPEN) < 0)
+	    return -1;
+    }
+    else {
+	if (zip_source_open(src->src) < 0) {
 	    src->error_source = ZIP_LES_LOWER;
-	else
-	    src->error_source = ZIP_LES_UPPER;
-	return -1;
+	    return -1;
+	}
+
+	ret = src->cb.l(src->src, src->ud, NULL, 0, ZIP_SOURCE_OPEN);
+	
+	if (ret < 0) {
+	    (void)zip_source_close(src->src);
+	    
+	    if (ret == ZIP_SOURCE_ERR_LOWER)
+		src->error_source = ZIP_LES_LOWER;
+	    else
+		src->error_source = ZIP_LES_UPPER;
+	    return -1;
+	}
     }
+
+    src->is_open = 1;
     
     return 0;
 }
diff --git a/lib/zip_source_pop.c b/lib/zip_source_pop.c
index 7352251..88d37da 100644
--- a/lib/zip_source_pop.c
+++ b/lib/zip_source_pop.c
@@ -49,9 +49,15 @@
 
     lower = src->src;
 
-    (void)src->cb.f(src->ud, NULL, 0, ZIP_SOURCE_FREE);
-
-    free(src);
+    if (lower == NULL)
+	zip_source_free(src);
+    else {
+	if (src->is_open)
+	    (void)src->cb.l(src, src->ud, NULL, 0, ZIP_SOURCE_CLOSE);
+	(void)src->cb.l(src, src->ud, NULL, 0, ZIP_SOURCE_FREE);
+	
+	free(src);
+    }
 
     return lower;
 }
diff --git a/lib/zip_source_read.c b/lib/zip_source_read.c
index ef7147d..432243b 100644
--- a/lib/zip_source_read.c
+++ b/lib/zip_source_read.c
@@ -42,7 +42,7 @@
 {
     zip_int64_t ret;
 
-    if (len > ZIP_INT64_MAX || (len > 0 && data == NULL)) {
+    if (!src->is_open || len > ZIP_INT64_MAX || (len > 0 && data == NULL)) {
 	src->error_source = ZIP_LES_INVAL;
 	return -1;
     }
diff --git a/lib/zipint.h b/lib/zipint.h
index 9e9b0cc..f52a996 100644
--- a/lib/zipint.h
+++ b/lib/zipint.h
@@ -184,6 +184,7 @@
     } cb;
     void *ud;
     enum zip_les error_source;
+    int is_open;
 };
 
 /* entry in zip archive directory */
@@ -247,6 +248,7 @@
 struct zip_source *_zip_source_file_or_p(struct zip *, const char *, FILE *,
 					 zip_uint64_t, zip_int64_t, int,
 					 const struct zip_stat *);
+struct zip_source *_zip_source_new(struct zip *);
 
 int _zip_changed(struct zip *, int *);
 void _zip_free(struct zip *);