* \param[in,out] hash_len     pointer to hash buffer size, if \a hdesc = NULL
  *                             only free \a hdesc instead of computing the hash
  *
- * \retval     -ENOSPC if \a hash = NULL, or \a hash_len < digest size
  * \retval     0 for success
+ * \retval     -EOVERFLOW if hash_len is too small for the hash digest
  * \retval     negative errno for other errors from lower layers
  */
 int cfs_crypto_hash_final(struct cfs_crypto_hash_desc *hdesc,
        struct ahash_request *req = (void *)hdesc;
        int size = crypto_ahash_digestsize(crypto_ahash_reqtfm(req));
 
-       if (!hash_len) {
-               crypto_free_ahash(crypto_ahash_reqtfm(req));
-               ahash_request_free(req);
-               return 0;
+       if (!hash || !hash_len) {
+               err = 0;
+               goto free_ahash;
        }
-       if (!hash || *hash_len < size) {
-               *hash_len = size;
-               return -ENOSPC;
+       if (*hash_len < size) {
+               err = -EOVERFLOW;
+               goto free_ahash;
        }
+
        ahash_request_set_crypt(req, NULL, hash, 0);
        err = crypto_ahash_final(req);
-
-       if (err < 0) {
-               /* May be caller can fix error */
-               return err;
-       }
+       if (!err)
+               *hash_len = size;
+free_ahash:
        crypto_free_ahash(crypto_ahash_reqtfm(req));
        ahash_request_free(req);
        return err;
 
                i++;
        }
 
-       bufsize = 4;
+       bufsize = sizeof(cksum);
        err = cfs_crypto_hash_final(hdesc, (unsigned char *)&cksum, &bufsize);
 
-       if (err)
-               cfs_crypto_hash_final(hdesc, NULL, NULL);
-
        /* For sending we only compute the wrong checksum instead
         * of corrupting the data so it is still correct on a redo
         */
 
 #define DEBUG_SUBSYSTEM S_SEC
 
 #include "../../include/linux/libcfs/libcfs.h"
-#include <linux/crypto.h>
 
 #include "../include/obd.h"
 #include "../include/obd_cksum.h"
 {
        struct cfs_crypto_hash_desc *hdesc;
        int hashsize;
-       char hashbuf[64];
        unsigned int bufsize;
        int i, err;
 
                                  desc->bd_iov[i].kiov_offset & ~CFS_PAGE_MASK,
                                  desc->bd_iov[i].kiov_len);
        }
+
        if (hashsize > buflen) {
+               unsigned char hashbuf[CFS_CRYPTO_HASH_DIGESTSIZE_MAX];
+
                bufsize = sizeof(hashbuf);
-               err = cfs_crypto_hash_final(hdesc, (unsigned char *)hashbuf,
-                                           &bufsize);
+               LASSERTF(bufsize >= hashsize, "bufsize = %u < hashsize %u\n",
+                        bufsize, hashsize);
+               err = cfs_crypto_hash_final(hdesc, hashbuf, &bufsize);
                memcpy(buf, hashbuf, buflen);
        } else {
                bufsize = buflen;
                err = cfs_crypto_hash_final(hdesc, buf, &bufsize);
        }
 
-       if (err)
-               cfs_crypto_hash_final(hdesc, NULL, NULL);
        return err;
 }
 EXPORT_SYMBOL(sptlrpc_get_bulk_checksum);