cryptoloop.c 6.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268
  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 <asm/semaphore.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. char cms[LO_NAME_SIZE]; /* cipher-mode string */
  35. char *cipher;
  36. char *mode;
  37. char *cmsp = cms; /* c-m string pointer */
  38. struct crypto_tfm *tfm = NULL;
  39. /* encryption breaks for non sector aligned offsets */
  40. if (info->lo_offset % LOOP_IV_SECTOR_SIZE)
  41. goto out;
  42. strncpy(cms, info->lo_crypt_name, LO_NAME_SIZE);
  43. cms[LO_NAME_SIZE - 1] = 0;
  44. cipher = strsep(&cmsp, "-");
  45. mode = strsep(&cmsp, "-");
  46. if (mode == NULL || strcmp(mode, "cbc") == 0)
  47. tfm = crypto_alloc_tfm(cipher, CRYPTO_TFM_MODE_CBC);
  48. else if (strcmp(mode, "ecb") == 0)
  49. tfm = crypto_alloc_tfm(cipher, CRYPTO_TFM_MODE_ECB);
  50. if (tfm == NULL)
  51. return -EINVAL;
  52. err = tfm->crt_u.cipher.cit_setkey(tfm, info->lo_encrypt_key,
  53. info->lo_encrypt_key_size);
  54. if (err != 0)
  55. goto out_free_tfm;
  56. lo->key_data = tfm;
  57. return 0;
  58. out_free_tfm:
  59. crypto_free_tfm(tfm);
  60. out:
  61. return err;
  62. }
  63. typedef int (*encdec_ecb_t)(struct crypto_tfm *tfm,
  64. struct scatterlist *sg_out,
  65. struct scatterlist *sg_in,
  66. unsigned int nsg);
  67. static int
  68. cryptoloop_transfer_ecb(struct loop_device *lo, int cmd,
  69. struct page *raw_page, unsigned raw_off,
  70. struct page *loop_page, unsigned loop_off,
  71. int size, sector_t IV)
  72. {
  73. struct crypto_tfm *tfm = (struct crypto_tfm *) lo->key_data;
  74. struct scatterlist sg_out = { NULL, };
  75. struct scatterlist sg_in = { NULL, };
  76. encdec_ecb_t encdecfunc;
  77. struct page *in_page, *out_page;
  78. unsigned in_offs, out_offs;
  79. if (cmd == READ) {
  80. in_page = raw_page;
  81. in_offs = raw_off;
  82. out_page = loop_page;
  83. out_offs = loop_off;
  84. encdecfunc = tfm->crt_u.cipher.cit_decrypt;
  85. } else {
  86. in_page = loop_page;
  87. in_offs = loop_off;
  88. out_page = raw_page;
  89. out_offs = raw_off;
  90. encdecfunc = tfm->crt_u.cipher.cit_encrypt;
  91. }
  92. while (size > 0) {
  93. const int sz = min(size, LOOP_IV_SECTOR_SIZE);
  94. sg_in.page = in_page;
  95. sg_in.offset = in_offs;
  96. sg_in.length = sz;
  97. sg_out.page = out_page;
  98. sg_out.offset = out_offs;
  99. sg_out.length = sz;
  100. encdecfunc(tfm, &sg_out, &sg_in, sz);
  101. size -= sz;
  102. in_offs += sz;
  103. out_offs += sz;
  104. }
  105. return 0;
  106. }
  107. typedef int (*encdec_cbc_t)(struct crypto_tfm *tfm,
  108. struct scatterlist *sg_out,
  109. struct scatterlist *sg_in,
  110. unsigned int nsg, u8 *iv);
  111. static int
  112. cryptoloop_transfer_cbc(struct loop_device *lo, int cmd,
  113. struct page *raw_page, unsigned raw_off,
  114. struct page *loop_page, unsigned loop_off,
  115. int size, sector_t IV)
  116. {
  117. struct crypto_tfm *tfm = (struct crypto_tfm *) lo->key_data;
  118. struct scatterlist sg_out = { NULL, };
  119. struct scatterlist sg_in = { NULL, };
  120. encdec_cbc_t encdecfunc;
  121. struct page *in_page, *out_page;
  122. unsigned in_offs, out_offs;
  123. if (cmd == READ) {
  124. in_page = raw_page;
  125. in_offs = raw_off;
  126. out_page = loop_page;
  127. out_offs = loop_off;
  128. encdecfunc = tfm->crt_u.cipher.cit_decrypt_iv;
  129. } else {
  130. in_page = loop_page;
  131. in_offs = loop_off;
  132. out_page = raw_page;
  133. out_offs = raw_off;
  134. encdecfunc = tfm->crt_u.cipher.cit_encrypt_iv;
  135. }
  136. while (size > 0) {
  137. const int sz = min(size, LOOP_IV_SECTOR_SIZE);
  138. u32 iv[4] = { 0, };
  139. iv[0] = cpu_to_le32(IV & 0xffffffff);
  140. sg_in.page = in_page;
  141. sg_in.offset = in_offs;
  142. sg_in.length = sz;
  143. sg_out.page = out_page;
  144. sg_out.offset = out_offs;
  145. sg_out.length = sz;
  146. encdecfunc(tfm, &sg_out, &sg_in, sz, (u8 *)iv);
  147. IV++;
  148. size -= sz;
  149. in_offs += sz;
  150. out_offs += sz;
  151. }
  152. return 0;
  153. }
  154. static int
  155. cryptoloop_transfer(struct loop_device *lo, int cmd,
  156. struct page *raw_page, unsigned raw_off,
  157. struct page *loop_page, unsigned loop_off,
  158. int size, sector_t IV)
  159. {
  160. struct crypto_tfm *tfm = (struct crypto_tfm *) lo->key_data;
  161. if(tfm->crt_cipher.cit_mode == CRYPTO_TFM_MODE_ECB)
  162. {
  163. lo->transfer = cryptoloop_transfer_ecb;
  164. return cryptoloop_transfer_ecb(lo, cmd, raw_page, raw_off,
  165. loop_page, loop_off, size, IV);
  166. }
  167. if(tfm->crt_cipher.cit_mode == CRYPTO_TFM_MODE_CBC)
  168. {
  169. lo->transfer = cryptoloop_transfer_cbc;
  170. return cryptoloop_transfer_cbc(lo, cmd, raw_page, raw_off,
  171. loop_page, loop_off, size, IV);
  172. }
  173. /* This is not supposed to happen */
  174. printk( KERN_ERR "cryptoloop: unsupported cipher mode in cryptoloop_transfer!\n");
  175. return -EINVAL;
  176. }
  177. static int
  178. cryptoloop_ioctl(struct loop_device *lo, int cmd, unsigned long arg)
  179. {
  180. return -EINVAL;
  181. }
  182. static int
  183. cryptoloop_release(struct loop_device *lo)
  184. {
  185. struct crypto_tfm *tfm = (struct crypto_tfm *) lo->key_data;
  186. if (tfm != NULL) {
  187. crypto_free_tfm(tfm);
  188. lo->key_data = NULL;
  189. return 0;
  190. }
  191. printk(KERN_ERR "cryptoloop_release(): tfm == NULL?\n");
  192. return -EINVAL;
  193. }
  194. static struct loop_func_table cryptoloop_funcs = {
  195. .number = LO_CRYPT_CRYPTOAPI,
  196. .init = cryptoloop_init,
  197. .ioctl = cryptoloop_ioctl,
  198. .transfer = cryptoloop_transfer,
  199. .release = cryptoloop_release,
  200. .owner = THIS_MODULE
  201. };
  202. static int __init
  203. init_cryptoloop(void)
  204. {
  205. int rc = loop_register_transfer(&cryptoloop_funcs);
  206. if (rc)
  207. printk(KERN_ERR "cryptoloop: loop_register_transfer failed\n");
  208. return rc;
  209. }
  210. static void __exit
  211. cleanup_cryptoloop(void)
  212. {
  213. if (loop_unregister_transfer(LO_CRYPT_CRYPTOAPI))
  214. printk(KERN_ERR
  215. "cryptoloop: loop_unregister_transfer failed\n");
  216. }
  217. module_init(init_cryptoloop);
  218. module_exit(cleanup_cryptoloop);