Dont over-read fd, use function pointer for hash function.
diff --git a/include/libbb.h b/include/libbb.h
index 88f43d0..22a77e6 100644
--- a/include/libbb.h
+++ b/include/libbb.h
@@ -479,7 +479,7 @@
 
 #define HASH_SHA1	1
 #define HASH_MD5	2
-extern int hash_fd(int fd, const off_t size, const uint8_t hash_algo, uint8_t *hashval);
+extern int hash_fd(int fd, const size_t size, const uint8_t hash_algo, uint8_t *hashval);
 
 
 #endif /* __LIBCONFIG_H__ */
diff --git a/libbb/hash_fd.c b/libbb/hash_fd.c
index ad0e4f4..770ba09 100644
--- a/libbb/hash_fd.c
+++ b/libbb/hash_fd.c
@@ -166,10 +166,11 @@
 
 /* SHA1 hash data in an array of bytes into hash buffer and call the        */
 /* hash_compile function as required.                                       */
-static void sha1_hash(const unsigned char data[], unsigned int len, sha1_ctx_t *ctx)
+static void sha1_hash(const void *data, size_t len, void *ctx_v)
 {
-	uint32_t pos = (uint32_t) (ctx->count[0] & SHA1_MASK),
-		freeb = SHA1_BLOCK_SIZE - pos;
+	sha1_ctx_t *ctx = (sha1_ctx_t *) ctx_v;
+	uint32_t pos = (uint32_t) (ctx->count[0] & SHA1_MASK);
+	uint32_t freeb = SHA1_BLOCK_SIZE - pos;
 	const unsigned char *sp = data;
 
 	if ((ctx->count[0] += len) < len)
@@ -706,6 +707,15 @@
 	}
 }
 
+static void md5_hash(const void *buffer, size_t length, void *md5_ctx)
+{
+	if (length % 64 == 0) {
+		md5_hash_block(buffer, length, md5_ctx);
+	} else {
+		md5_hash_bytes(buffer, length, md5_ctx);
+	}
+}
+
 /* Process the remaining bytes in the buffer and put result from CTX
  * in first 16 bytes following RESBUF.  The result is always in little
  * endian byte order, so that a byte-wise output yields to the wanted
@@ -760,13 +770,17 @@
 
 
 
-extern int hash_fd(int src_fd, const off_t size, const uint8_t hash_algo,
+extern int hash_fd(int src_fd, const size_t size, const uint8_t hash_algo,
 				   uint8_t * hashval)
 {
 	int result = EXIT_SUCCESS;
-	off_t hashed_count = 0;
-	unsigned int blocksize = 0;
+//	size_t hashed_count = 0;
+	size_t blocksize = 0;
+	size_t remaining = size;
 	unsigned char *buffer = NULL;
+	void (*hash_fn_ptr)(const void *, size_t, void *) = NULL;
+	void *cx = NULL;
+
 #ifdef CONFIG_SHA1SUM
 	sha1_ctx_t sha1_cx;
 #endif
@@ -779,13 +793,17 @@
 	if (hash_algo == HASH_SHA1) {
 		/* Ensure that BLOCKSIZE is a multiple of 64.  */
 		blocksize = 65536;
-		buffer = malloc(blocksize);
+		buffer = xmalloc(blocksize);
+		hash_fn_ptr = sha1_hash;
+		cx = &sha1_cx;
 	}
 #endif
 #ifdef CONFIG_MD5SUM
 	if (hash_algo == HASH_MD5) {
 		blocksize = 4096;
-		buffer = malloc(blocksize + 72);
+		buffer = xmalloc(blocksize + 72);
+		hash_fn_ptr = md5_hash;
+		cx = &md5_cx;
 	}
 #endif
 	
@@ -801,33 +819,29 @@
 	}
 #endif
 	/* Iterate over full file contents.  */
-	do {
-		const ssize_t count = bb_full_read(src_fd, buffer, blocksize);
+	while ((remaining == (size_t) -1) || (remaining > 0)) {
+		size_t read_try;
+		ssize_t read_got;
 
-		if (count < 1) {
+		if (remaining > blocksize) {
+			read_try = blocksize;
+		} else {
+			read_try = remaining;
+		}
+		read_got = bb_full_read(src_fd, buffer, read_try);
+		if (read_got < 1) {
 			/* count == 0 means short read
 			 * count == -1 means read error */
-			result = count - 1;
+			result = read_got - 1;
 			break;
 		}
-		hashed_count += count;
+		if (remaining != (size_t) -1) {
+			remaining -= read_got;
+		}
 
 		/* Process buffer */
-#ifdef CONFIG_SHA1SUM
-		if (hash_algo == HASH_SHA1) {
-			sha1_hash(buffer, count, &sha1_cx);
-		}
-#endif
-#ifdef CONFIG_MD5SUM
-		if (hash_algo == HASH_MD5) {
-			if (count % 64 == 0) {
-				md5_hash_block(buffer, count, &md5_cx);
-			} else {
-				md5_hash_bytes(buffer, count, &md5_cx);
-			}
-		}
-#endif
-	} while ((size == (off_t) - 1) || (hashed_count < size));
+		hash_fn_ptr(buffer, read_got, cx);
+	}
 
 	/* Finalize and write the hash into our buffer.  */
 #ifdef CONFIG_SHA1SUM