cryptoloop.c 4.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217
  1. /*
  2. Linux loop encryption enabling module
  3. Copyright (C) 2002 Herbert Valerio Riedel <hvr@gnu.org>
  4. Copyright (C) 2003 Fruhwirth Clemens <clemens@endorphin.org>
  5. This module is free software; you can redistribute it and/or modify
  6. it under the terms of the GNU General Public License as published by
  7. the Free Software Foundation; either version 2 of the License, or
  8. (at your option) any later version.
  9. This module is distributed in the hope that it will be useful,
  10. but WITHOUT ANY WARRANTY; without even the implied warranty of
  11. MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  12. GNU General Public License for more details.
  13. You should have received a copy of the GNU General Public License
  14. along with this module; if not, write to the Free Software
  15. Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
  16. */
  17. #include <linux/module.h>
  18. #include <linux/init.h>
  19. #include <linux/string.h>
  20. #include <linux/crypto.h>
  21. #include <linux/blkdev.h>
  22. #include <linux/loop.h>
  23. #include <linux/scatterlist.h>
  24. #include <asm/uaccess.h>
  25. MODULE_LICENSE("GPL");
  26. MODULE_DESCRIPTION("loop blockdevice transferfunction adaptor / CryptoAPI");
  27. MODULE_AUTHOR("Herbert Valerio Riedel <hvr@gnu.org>");
  28. #define LOOP_IV_SECTOR_BITS 9
  29. #define LOOP_IV_SECTOR_SIZE (1 << LOOP_IV_SECTOR_BITS)
  30. static int
  31. cryptoloop_init(struct loop_device *lo, const struct loop_info64 *info)
  32. {
  33. int err = -EINVAL;
  34. int cipher_len;
  35. int mode_len;
  36. char cms[LO_NAME_SIZE]; /* cipher-mode string */
  37. char *cipher;
  38. char *mode;
  39. char *cmsp = cms; /* c-m string pointer */
  40. struct crypto_blkcipher *tfm;
  41. /* encryption breaks for non sector aligned offsets */
  42. if (info->lo_offset % LOOP_IV_SECTOR_SIZE)
  43. goto out;
  44. strncpy(cms, info->lo_crypt_name, LO_NAME_SIZE);
  45. cms[LO_NAME_SIZE - 1] = 0;
  46. cipher = cmsp;
  47. cipher_len = strcspn(cmsp, "-");
  48. mode = cmsp + cipher_len;
  49. mode_len = 0;
  50. if (*mode) {
  51. mode++;
  52. mode_len = strcspn(mode, "-");
  53. }
  54. if (!mode_len) {
  55. mode = "cbc";
  56. mode_len = 3;
  57. }
  58. if (cipher_len + mode_len + 3 > LO_NAME_SIZE)
  59. return -EINVAL;
  60. memmove(cms, mode, mode_len);
  61. cmsp = cms + mode_len;
  62. *cmsp++ = '(';
  63. memcpy(cmsp, info->lo_crypt_name, cipher_len);
  64. cmsp += cipher_len;
  65. *cmsp++ = ')';
  66. *cmsp = 0;
  67. tfm = crypto_alloc_blkcipher(cms, 0, CRYPTO_ALG_ASYNC);
  68. if (IS_ERR(tfm))
  69. return PTR_ERR(tfm);
  70. err = crypto_blkcipher_setkey(tfm, info->lo_encrypt_key,
  71. info->lo_encrypt_key_size);
  72. if (err != 0)
  73. goto out_free_tfm;
  74. lo->key_data = tfm;
  75. return 0;
  76. out_free_tfm:
  77. crypto_free_blkcipher(tfm);
  78. out:
  79. return err;
  80. }
  81. typedef int (*encdec_cbc_t)(struct blkcipher_desc *desc,
  82. struct scatterlist *sg_out,
  83. struct scatterlist *sg_in,
  84. unsigned int nsg);
  85. static int
  86. cryptoloop_transfer(struct loop_device *lo, int cmd,
  87. struct page *raw_page, unsigned raw_off,
  88. struct page *loop_page, unsigned loop_off,
  89. int size, sector_t IV)
  90. {
  91. struct crypto_blkcipher *tfm = lo->key_data;
  92. struct blkcipher_desc desc = {
  93. .tfm = tfm,
  94. .flags = CRYPTO_TFM_REQ_MAY_SLEEP,
  95. };
  96. struct scatterlist sg_out;
  97. struct scatterlist sg_in;
  98. encdec_cbc_t encdecfunc;
  99. struct page *in_page, *out_page;
  100. unsigned in_offs, out_offs;
  101. int err;
  102. sg_init_table(&sg_out, 1);
  103. sg_init_table(&sg_in, 1);
  104. if (cmd == READ) {
  105. in_page = raw_page;
  106. in_offs = raw_off;
  107. out_page = loop_page;
  108. out_offs = loop_off;
  109. encdecfunc = crypto_blkcipher_crt(tfm)->decrypt;
  110. } else {
  111. in_page = loop_page;
  112. in_offs = loop_off;
  113. out_page = raw_page;
  114. out_offs = raw_off;
  115. encdecfunc = crypto_blkcipher_crt(tfm)->encrypt;
  116. }
  117. while (size > 0) {
  118. const int sz = min(size, LOOP_IV_SECTOR_SIZE);
  119. u32 iv[4] = { 0, };
  120. iv[0] = cpu_to_le32(IV & 0xffffffff);
  121. sg_set_page(&sg_in, in_page, sz, in_offs);
  122. sg_set_page(&sg_out, out_page, sz, out_offs);
  123. desc.info = iv;
  124. err = encdecfunc(&desc, &sg_out, &sg_in, sz);
  125. if (err)
  126. return err;
  127. IV++;
  128. size -= sz;
  129. in_offs += sz;
  130. out_offs += sz;
  131. }
  132. return 0;
  133. }
  134. static int
  135. cryptoloop_ioctl(struct loop_device *lo, int cmd, unsigned long arg)
  136. {
  137. return -EINVAL;
  138. }
  139. static int
  140. cryptoloop_release(struct loop_device *lo)
  141. {
  142. struct crypto_blkcipher *tfm = lo->key_data;
  143. if (tfm != NULL) {
  144. crypto_free_blkcipher(tfm);
  145. lo->key_data = NULL;
  146. return 0;
  147. }
  148. printk(KERN_ERR "cryptoloop_release(): tfm == NULL?\n");
  149. return -EINVAL;
  150. }
  151. static struct loop_func_table cryptoloop_funcs = {
  152. .number = LO_CRYPT_CRYPTOAPI,
  153. .init = cryptoloop_init,
  154. .ioctl = cryptoloop_ioctl,
  155. .transfer = cryptoloop_transfer,
  156. .release = cryptoloop_release,
  157. .owner = THIS_MODULE
  158. };
  159. static int __init
  160. init_cryptoloop(void)
  161. {
  162. int rc = loop_register_transfer(&cryptoloop_funcs);
  163. if (rc)
  164. printk(KERN_ERR "cryptoloop: loop_register_transfer failed\n");
  165. return rc;
  166. }
  167. static void __exit
  168. cleanup_cryptoloop(void)
  169. {
  170. if (loop_unregister_transfer(LO_CRYPT_CRYPTOAPI))
  171. printk(KERN_ERR
  172. "cryptoloop: loop_unregister_transfer failed\n");
  173. }
  174. module_init(init_cryptoloop);
  175. module_exit(cleanup_cryptoloop);