copy_to_user.S 1.4 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273
  1. /* SPDX-License-Identifier: GPL-2.0-only */
  2. /*
  3. * Copyright (C) 2012 ARM Ltd.
  4. */
  5. #include <linux/linkage.h>
  6. #include <asm/asm-uaccess.h>
  7. #include <asm/assembler.h>
  8. #include <asm/cache.h>
  9. /*
  10. * Copy to user space from a kernel buffer (alignment handled by the hardware)
  11. *
  12. * Parameters:
  13. * x0 - to
  14. * x1 - from
  15. * x2 - n
  16. * Returns:
  17. * x0 - bytes not copied
  18. */
  19. .macro ldrb1 reg, ptr, val
  20. ldrb \reg, [\ptr], \val
  21. .endm
  22. .macro strb1 reg, ptr, val
  23. user_ldst 9998f, sttrb, \reg, \ptr, \val
  24. .endm
  25. .macro ldrh1 reg, ptr, val
  26. ldrh \reg, [\ptr], \val
  27. .endm
  28. .macro strh1 reg, ptr, val
  29. user_ldst 9997f, sttrh, \reg, \ptr, \val
  30. .endm
  31. .macro ldr1 reg, ptr, val
  32. ldr \reg, [\ptr], \val
  33. .endm
  34. .macro str1 reg, ptr, val
  35. user_ldst 9997f, sttr, \reg, \ptr, \val
  36. .endm
  37. .macro ldp1 reg1, reg2, ptr, val
  38. ldp \reg1, \reg2, [\ptr], \val
  39. .endm
  40. .macro stp1 reg1, reg2, ptr, val
  41. user_stp 9997f, \reg1, \reg2, \ptr, \val
  42. .endm
  43. end .req x5
  44. srcin .req x15
  45. SYM_FUNC_START(__arch_copy_to_user)
  46. add end, x0, x2
  47. mov srcin, x1
  48. #include "copy_template.S"
  49. mov x0, #0
  50. ret
  51. // Exception fixups
  52. 9997: cmp dst, dstin
  53. b.ne 9998f
  54. // Before being absolutely sure we couldn't copy anything, try harder
  55. ldrb tmp1w, [srcin]
  56. USER(9998f, sttrb tmp1w, [dst])
  57. add dst, dst, #1
  58. 9998: sub x0, end, dst // bytes not copied
  59. ret
  60. SYM_FUNC_END(__arch_copy_to_user)
  61. EXPORT_SYMBOL(__arch_copy_to_user)