aes-armv8-bcm-glue.c 3.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124
  1. /*
  2. * linux/arch/arm64/crypto/aes-armv8-bcm-glue.c - wrapper code for ARMv8 Aarch32 CE
  3. *
  4. * Copyright (C) 2014 Mediatek Inc.
  5. *
  6. * This program is free software; you can redistribute it and/or modify
  7. * it under the terms of the GNU General Public License version 2 as
  8. * published by the Free Software Foundation.
  9. */
  10. #include <asm/neon.h>
  11. #include <crypto/aes.h>
  12. #include <crypto/ablk_helper.h>
  13. #include <crypto/algapi.h>
  14. #include <linux/module.h>
  15. #include <linux/cpufeature.h>
  16. asmlinkage void aes_v8_cbc_encrypt(u8 out[], u8 const in[], u8 const rk[],
  17. int rounds, int blocks, u8 iv[], int first);
  18. asmlinkage void aes_v8_cbc_decrypt(u8 out[], u8 const in[], u8 const rk[],
  19. int rounds, int blocks, u8 iv[], int first);
  20. static int cbc_encrypt(struct blkcipher_desc *desc, struct scatterlist *dst,
  21. struct scatterlist *src, unsigned int nbytes)
  22. {
  23. struct crypto_aes_ctx *ctx = crypto_blkcipher_ctx(desc->tfm);
  24. int err, first, rounds = 6 + ctx->key_length / 4;
  25. struct blkcipher_walk walk;
  26. unsigned int blocks;
  27. desc->flags &= ~CRYPTO_TFM_REQ_MAY_SLEEP;
  28. blkcipher_walk_init(&walk, dst, src, nbytes);
  29. err = blkcipher_walk_virt(desc, &walk);
  30. kernel_neon_begin();
  31. for (first = 1; (blocks = (walk.nbytes / AES_BLOCK_SIZE)); first = 0) {
  32. aes_v8_cbc_encrypt(walk.dst.virt.addr, walk.src.virt.addr,
  33. (u8 *)ctx->key_enc, rounds, blocks, walk.iv,
  34. first);
  35. err = blkcipher_walk_done(desc, &walk, walk.nbytes % AES_BLOCK_SIZE);
  36. }
  37. kernel_neon_end();
  38. return err;
  39. }
  40. static int cbc_decrypt(struct blkcipher_desc *desc, struct scatterlist *dst,
  41. struct scatterlist *src, unsigned int nbytes)
  42. {
  43. struct crypto_aes_ctx *ctx = crypto_blkcipher_ctx(desc->tfm);
  44. int err, first, rounds = 6 + ctx->key_length / 4;
  45. struct blkcipher_walk walk;
  46. unsigned int blocks;
  47. desc->flags &= ~CRYPTO_TFM_REQ_MAY_SLEEP;
  48. blkcipher_walk_init(&walk, dst, src, nbytes);
  49. err = blkcipher_walk_virt(desc, &walk);
  50. kernel_neon_begin();
  51. for (first = 1; (blocks = (walk.nbytes / AES_BLOCK_SIZE)); first = 0) {
  52. aes_v8_cbc_decrypt(walk.dst.virt.addr, walk.src.virt.addr,
  53. (u8 *)ctx->key_dec, rounds, blocks, walk.iv,
  54. first);
  55. err = blkcipher_walk_done(desc, &walk, walk.nbytes % AES_BLOCK_SIZE);
  56. }
  57. kernel_neon_end();
  58. return err;
  59. }
  60. static struct crypto_alg aes_algs[] = { {
  61. .cra_name = "__cbc-aes-ce" ,
  62. .cra_driver_name = "__driver-cbc-aes-ce" ,
  63. .cra_priority = 0,
  64. .cra_flags = CRYPTO_ALG_TYPE_BLKCIPHER,
  65. .cra_blocksize = AES_BLOCK_SIZE,
  66. .cra_ctxsize = sizeof(struct crypto_aes_ctx),
  67. .cra_alignmask = 7,
  68. .cra_type = &crypto_blkcipher_type,
  69. .cra_module = THIS_MODULE,
  70. .cra_blkcipher = {
  71. .min_keysize = AES_MIN_KEY_SIZE,
  72. .max_keysize = AES_MAX_KEY_SIZE,
  73. .ivsize = AES_BLOCK_SIZE,
  74. .setkey = crypto_aes_set_key,
  75. .encrypt = cbc_encrypt,
  76. .decrypt = cbc_decrypt,
  77. },
  78. }, {
  79. .cra_name = "cbc(aes)",
  80. .cra_driver_name = "cbc-aes-ce",
  81. .cra_priority = 300,
  82. .cra_flags = CRYPTO_ALG_TYPE_ABLKCIPHER|CRYPTO_ALG_ASYNC,
  83. .cra_blocksize = AES_BLOCK_SIZE,
  84. .cra_ctxsize = sizeof(struct async_helper_ctx),
  85. .cra_alignmask = 7,
  86. .cra_type = &crypto_ablkcipher_type,
  87. .cra_module = THIS_MODULE,
  88. .cra_init = ablk_init,
  89. .cra_exit = ablk_exit,
  90. .cra_ablkcipher = {
  91. .min_keysize = AES_MIN_KEY_SIZE,
  92. .max_keysize = AES_MAX_KEY_SIZE,
  93. .ivsize = AES_BLOCK_SIZE,
  94. .setkey = ablk_set_key,
  95. .encrypt = ablk_encrypt,
  96. .decrypt = ablk_decrypt,
  97. }
  98. } };
  99. static int __init aes_init(void)
  100. {
  101. return crypto_register_algs(aes_algs, ARRAY_SIZE(aes_algs));
  102. }
  103. static void __exit aes_exit(void)
  104. {
  105. crypto_unregister_algs(aes_algs, ARRAY_SIZE(aes_algs));
  106. }
  107. module_init(aes_init);
  108. module_exit(aes_exit);
  109. MODULE_DESCRIPTION("Bit sliced AES in CBC modes using CE");
  110. MODULE_AUTHOR("Mediatek Inc.");
  111. MODULE_LICENSE("GPL");