Commit b7d8092a authored by Kent Overstreet's avatar Kent Overstreet

bcachefs: do_encrypt() now handles allocation failures

convert to darray, and add a fallback when allocation fails
Signed-off-by: default avatarKent Overstreet <kent.overstreet@linux.dev>
parent 3340dee2
...@@ -100,13 +100,12 @@ static inline int do_encrypt_sg(struct crypto_sync_skcipher *tfm, ...@@ -100,13 +100,12 @@ static inline int do_encrypt_sg(struct crypto_sync_skcipher *tfm,
struct scatterlist *sg, size_t len) struct scatterlist *sg, size_t len)
{ {
SYNC_SKCIPHER_REQUEST_ON_STACK(req, tfm); SYNC_SKCIPHER_REQUEST_ON_STACK(req, tfm);
int ret;
skcipher_request_set_sync_tfm(req, tfm); skcipher_request_set_sync_tfm(req, tfm);
skcipher_request_set_callback(req, 0, NULL, NULL); skcipher_request_set_callback(req, 0, NULL, NULL);
skcipher_request_set_crypt(req, sg, sg, len, nonce.d); skcipher_request_set_crypt(req, sg, sg, len, nonce.d);
ret = crypto_skcipher_encrypt(req); int ret = crypto_skcipher_encrypt(req);
if (ret) if (ret)
pr_err("got error %i from crypto_skcipher_encrypt()", ret); pr_err("got error %i from crypto_skcipher_encrypt()", ret);
...@@ -128,28 +127,41 @@ static inline int do_encrypt(struct crypto_sync_skcipher *tfm, ...@@ -128,28 +127,41 @@ static inline int do_encrypt(struct crypto_sync_skcipher *tfm,
len, offset_in_page(buf)); len, offset_in_page(buf));
return do_encrypt_sg(tfm, nonce, &sg, len); return do_encrypt_sg(tfm, nonce, &sg, len);
} else { } else {
unsigned pages = buf_pages(buf, len); DARRAY_PREALLOCATED(struct scatterlist, 4) sgl;
struct scatterlist *sg; size_t sgl_len = 0;
size_t orig_len = len; int ret;
int ret, i;
sg = kmalloc_array(pages, sizeof(*sg), GFP_KERNEL);
if (!sg)
return -BCH_ERR_ENOMEM_do_encrypt;
sg_init_table(sg, pages); darray_init(&sgl);
for (i = 0; i < pages; i++) { while (len) {
unsigned offset = offset_in_page(buf); unsigned offset = offset_in_page(buf);
unsigned pg_len = min_t(size_t, len, PAGE_SIZE - offset); struct scatterlist sg = {
.page_link = (unsigned long) vmalloc_to_page(buf),
.offset = offset,
.length = min(len, PAGE_SIZE - offset),
};
if (darray_push(&sgl, sg)) {
sg_mark_end(&darray_last(sgl));
ret = do_encrypt_sg(tfm, nonce, sgl.data, sgl_len);
if (ret)
goto err;
sg_set_page(sg + i, vmalloc_to_page(buf), pg_len, offset); nonce = nonce_add(nonce, sgl_len);
buf += pg_len; sgl_len = 0;
len -= pg_len; sgl.nr = 0;
BUG_ON(darray_push(&sgl, sg));
} }
ret = do_encrypt_sg(tfm, nonce, sg, orig_len); buf += sg.length;
kfree(sg); len -= sg.length;
sgl_len += sg.length;
}
sg_mark_end(&darray_last(sgl));
ret = do_encrypt_sg(tfm, nonce, sgl.data, sgl_len);
err:
darray_exit(&sgl);
return ret; return ret;
} }
} }
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment