return glue_xts_req_128bit(&aesni_enc_xts, req,
                                   XTS_TWEAK_CAST(aesni_xts_tweak),
                                   aes_ctx(ctx->raw_tweak_ctx),
-                                  aes_ctx(ctx->raw_crypt_ctx));
+                                  aes_ctx(ctx->raw_crypt_ctx),
+                                  false);
 }
 
 static int xts_decrypt(struct skcipher_request *req)
        return glue_xts_req_128bit(&aesni_dec_xts, req,
                                   XTS_TWEAK_CAST(aesni_xts_tweak),
                                   aes_ctx(ctx->raw_tweak_ctx),
-                                  aes_ctx(ctx->raw_crypt_ctx));
+                                  aes_ctx(ctx->raw_crypt_ctx),
+                                  true);
 }
 
 static int
 
 
        return glue_xts_req_128bit(&camellia_enc_xts, req,
                                   XTS_TWEAK_CAST(camellia_enc_blk),
-                                  &ctx->tweak_ctx, &ctx->crypt_ctx);
+                                  &ctx->tweak_ctx, &ctx->crypt_ctx, false);
 }
 
 static int xts_decrypt(struct skcipher_request *req)
 
        return glue_xts_req_128bit(&camellia_dec_xts, req,
                                   XTS_TWEAK_CAST(camellia_enc_blk),
-                                  &ctx->tweak_ctx, &ctx->crypt_ctx);
+                                  &ctx->tweak_ctx, &ctx->crypt_ctx, true);
 }
 
 static struct skcipher_alg camellia_algs[] = {
 
 
        return glue_xts_req_128bit(&camellia_enc_xts, req,
                                   XTS_TWEAK_CAST(camellia_enc_blk),
-                                  &ctx->tweak_ctx, &ctx->crypt_ctx);
+                                  &ctx->tweak_ctx, &ctx->crypt_ctx, false);
 }
 
 static int xts_decrypt(struct skcipher_request *req)
 
        return glue_xts_req_128bit(&camellia_dec_xts, req,
                                   XTS_TWEAK_CAST(camellia_enc_blk),
-                                  &ctx->tweak_ctx, &ctx->crypt_ctx);
+                                  &ctx->tweak_ctx, &ctx->crypt_ctx, true);
 }
 
 static struct skcipher_alg camellia_algs[] = {
 
 
        return glue_xts_req_128bit(&cast6_enc_xts, req,
                                   XTS_TWEAK_CAST(__cast6_encrypt),
-                                  &ctx->tweak_ctx, &ctx->crypt_ctx);
+                                  &ctx->tweak_ctx, &ctx->crypt_ctx, false);
 }
 
 static int xts_decrypt(struct skcipher_request *req)
 
        return glue_xts_req_128bit(&cast6_dec_xts, req,
                                   XTS_TWEAK_CAST(__cast6_encrypt),
-                                  &ctx->tweak_ctx, &ctx->crypt_ctx);
+                                  &ctx->tweak_ctx, &ctx->crypt_ctx, true);
 }
 
 static struct skcipher_alg cast6_algs[] = {
 
 #include <crypto/b128ops.h>
 #include <crypto/gf128mul.h>
 #include <crypto/internal/skcipher.h>
+#include <crypto/scatterwalk.h>
 #include <crypto/xts.h>
 #include <asm/crypto/glue_helper.h>
 
 int glue_xts_req_128bit(const struct common_glue_ctx *gctx,
                        struct skcipher_request *req,
                        common_glue_func_t tweak_fn, void *tweak_ctx,
-                       void *crypt_ctx)
+                       void *crypt_ctx, bool decrypt)
 {
+       const bool cts = (req->cryptlen % XTS_BLOCK_SIZE);
        const unsigned int bsize = 128 / 8;
+       struct skcipher_request subreq;
        struct skcipher_walk walk;
        bool fpu_enabled = false;
-       unsigned int nbytes;
+       unsigned int nbytes, tail;
        int err;
 
+       if (req->cryptlen < XTS_BLOCK_SIZE)
+               return -EINVAL;
+
+       if (unlikely(cts)) {
+               struct crypto_skcipher *tfm = crypto_skcipher_reqtfm(req);
+
+               tail = req->cryptlen % XTS_BLOCK_SIZE + XTS_BLOCK_SIZE;
+
+               skcipher_request_set_tfm(&subreq, tfm);
+               skcipher_request_set_callback(&subreq,
+                                             crypto_skcipher_get_flags(tfm),
+                                             NULL, NULL);
+               skcipher_request_set_crypt(&subreq, req->src, req->dst,
+                                          req->cryptlen - tail, req->iv);
+               req = &subreq;
+       }
+
        err = skcipher_walk_virt(&walk, req, false);
        nbytes = walk.nbytes;
-       if (!nbytes)
+       if (err)
                return err;
 
        /* set minimum length to bsize, for tweak_fn */
                nbytes = walk.nbytes;
        }
 
+       if (unlikely(cts)) {
+               u8 *next_tweak, *final_tweak = req->iv;
+               struct scatterlist *src, *dst;
+               struct scatterlist s[2], d[2];
+               le128 b[2];
+
+               dst = src = scatterwalk_ffwd(s, req->src, req->cryptlen);
+               if (req->dst != req->src)
+                       dst = scatterwalk_ffwd(d, req->dst, req->cryptlen);
+
+               if (decrypt) {
+                       next_tweak = memcpy(b, req->iv, XTS_BLOCK_SIZE);
+                       gf128mul_x_ble(b, b);
+               } else {
+                       next_tweak = req->iv;
+               }
+
+               skcipher_request_set_crypt(&subreq, src, dst, XTS_BLOCK_SIZE,
+                                          next_tweak);
+
+               err = skcipher_walk_virt(&walk, req, false) ?:
+                     skcipher_walk_done(&walk,
+                               __glue_xts_req_128bit(gctx, crypt_ctx, &walk));
+               if (err)
+                       goto out;
+
+               scatterwalk_map_and_copy(b, dst, 0, XTS_BLOCK_SIZE, 0);
+               memcpy(b + 1, b, tail - XTS_BLOCK_SIZE);
+               scatterwalk_map_and_copy(b, src, XTS_BLOCK_SIZE,
+                                        tail - XTS_BLOCK_SIZE, 0);
+               scatterwalk_map_and_copy(b, dst, 0, tail, 1);
+
+               skcipher_request_set_crypt(&subreq, dst, dst, XTS_BLOCK_SIZE,
+                                          final_tweak);
+
+               err = skcipher_walk_virt(&walk, req, false) ?:
+                     skcipher_walk_done(&walk,
+                               __glue_xts_req_128bit(gctx, crypt_ctx, &walk));
+       }
+
+out:
        glue_fpu_end(fpu_enabled);
 
        return err;
 
 
        return glue_xts_req_128bit(&serpent_enc_xts, req,
                                   XTS_TWEAK_CAST(__serpent_encrypt),
-                                  &ctx->tweak_ctx, &ctx->crypt_ctx);
+                                  &ctx->tweak_ctx, &ctx->crypt_ctx, false);
 }
 
 static int xts_decrypt(struct skcipher_request *req)
 
        return glue_xts_req_128bit(&serpent_dec_xts, req,
                                   XTS_TWEAK_CAST(__serpent_encrypt),
-                                  &ctx->tweak_ctx, &ctx->crypt_ctx);
+                                  &ctx->tweak_ctx, &ctx->crypt_ctx, true);
 }
 
 static struct skcipher_alg serpent_algs[] = {
 
 
        return glue_xts_req_128bit(&serpent_enc_xts, req,
                                   XTS_TWEAK_CAST(__serpent_encrypt),
-                                  &ctx->tweak_ctx, &ctx->crypt_ctx);
+                                  &ctx->tweak_ctx, &ctx->crypt_ctx, false);
 }
 
 static int xts_decrypt(struct skcipher_request *req)
 
        return glue_xts_req_128bit(&serpent_dec_xts, req,
                                   XTS_TWEAK_CAST(__serpent_encrypt),
-                                  &ctx->tweak_ctx, &ctx->crypt_ctx);
+                                  &ctx->tweak_ctx, &ctx->crypt_ctx, true);
 }
 
 static struct skcipher_alg serpent_algs[] = {
 
 
        return glue_xts_req_128bit(&twofish_enc_xts, req,
                                   XTS_TWEAK_CAST(twofish_enc_blk),
-                                  &ctx->tweak_ctx, &ctx->crypt_ctx);
+                                  &ctx->tweak_ctx, &ctx->crypt_ctx, false);
 }
 
 static int xts_decrypt(struct skcipher_request *req)
 
        return glue_xts_req_128bit(&twofish_dec_xts, req,
                                   XTS_TWEAK_CAST(twofish_enc_blk),
-                                  &ctx->tweak_ctx, &ctx->crypt_ctx);
+                                  &ctx->tweak_ctx, &ctx->crypt_ctx, true);
 }
 
 static struct skcipher_alg twofish_algs[] = {
 
 extern int glue_xts_req_128bit(const struct common_glue_ctx *gctx,
                               struct skcipher_request *req,
                               common_glue_func_t tweak_fn, void *tweak_ctx,
-                              void *crypt_ctx);
+                              void *crypt_ctx, bool decrypt);
 
 extern void glue_xts_crypt_128bit_one(void *ctx, u128 *dst, const u128 *src,
                                      le128 *iv, common_glue_func_t fn);