scatterwalk.h 2.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105
  1. /* SPDX-License-Identifier: GPL-2.0-or-later */
  2. /*
  3. * Cryptographic scatter and gather helpers.
  4. *
  5. * Copyright (c) 2002 James Morris <[email protected]>
  6. * Copyright (c) 2002 Adam J. Richter <[email protected]>
  7. * Copyright (c) 2004 Jean-Luc Cooke <[email protected]>
  8. * Copyright (c) 2007 Herbert Xu <[email protected]>
  9. */
  10. #ifndef _CRYPTO_SCATTERWALK_H
  11. #define _CRYPTO_SCATTERWALK_H
  12. #include <crypto/algapi.h>
  13. #include <linux/highmem.h>
  14. #include <linux/mm.h>
  15. #include <linux/scatterlist.h>
  16. static inline void scatterwalk_crypto_chain(struct scatterlist *head,
  17. struct scatterlist *sg, int num)
  18. {
  19. if (sg)
  20. sg_chain(head, num, sg);
  21. else
  22. sg_mark_end(head);
  23. }
  24. static inline unsigned int scatterwalk_pagelen(struct scatter_walk *walk)
  25. {
  26. unsigned int len = walk->sg->offset + walk->sg->length - walk->offset;
  27. unsigned int len_this_page = offset_in_page(~walk->offset) + 1;
  28. return len_this_page > len ? len : len_this_page;
  29. }
  30. static inline unsigned int scatterwalk_clamp(struct scatter_walk *walk,
  31. unsigned int nbytes)
  32. {
  33. unsigned int len_this_page = scatterwalk_pagelen(walk);
  34. return nbytes > len_this_page ? len_this_page : nbytes;
  35. }
  36. static inline void scatterwalk_advance(struct scatter_walk *walk,
  37. unsigned int nbytes)
  38. {
  39. walk->offset += nbytes;
  40. }
  41. static inline struct page *scatterwalk_page(struct scatter_walk *walk)
  42. {
  43. return sg_page(walk->sg) + (walk->offset >> PAGE_SHIFT);
  44. }
  45. static inline void scatterwalk_unmap(void *vaddr)
  46. {
  47. kunmap_atomic(vaddr);
  48. }
  49. static inline void scatterwalk_start(struct scatter_walk *walk,
  50. struct scatterlist *sg)
  51. {
  52. walk->sg = sg;
  53. walk->offset = sg->offset;
  54. }
  55. static inline void *scatterwalk_map(struct scatter_walk *walk)
  56. {
  57. return kmap_atomic(scatterwalk_page(walk)) +
  58. offset_in_page(walk->offset);
  59. }
  60. static inline void scatterwalk_pagedone(struct scatter_walk *walk, int out,
  61. unsigned int more)
  62. {
  63. if (out) {
  64. struct page *page;
  65. page = sg_page(walk->sg) + ((walk->offset - 1) >> PAGE_SHIFT);
  66. flush_dcache_page(page);
  67. }
  68. if (more && walk->offset >= walk->sg->offset + walk->sg->length)
  69. scatterwalk_start(walk, sg_next(walk->sg));
  70. }
  71. static inline void scatterwalk_done(struct scatter_walk *walk, int out,
  72. int more)
  73. {
  74. if (!more || walk->offset >= walk->sg->offset + walk->sg->length ||
  75. !(walk->offset & (PAGE_SIZE - 1)))
  76. scatterwalk_pagedone(walk, out, more);
  77. }
  78. void scatterwalk_copychunks(void *buf, struct scatter_walk *walk,
  79. size_t nbytes, int out);
  80. void *scatterwalk_map(struct scatter_walk *walk);
  81. void scatterwalk_map_and_copy(void *buf, struct scatterlist *sg,
  82. unsigned int start, unsigned int nbytes, int out);
  83. struct scatterlist *scatterwalk_ffwd(struct scatterlist dst[2],
  84. struct scatterlist *src,
  85. unsigned int len);
  86. #endif /* _CRYPTO_SCATTERWALK_H */