David Brazdil | 0f672f6 | 2019-12-10 10:32:29 +0000 | [diff] [blame] | 1 | // SPDX-License-Identifier: GPL-2.0-only |
Andrew Scull | b4b6d4a | 2019-01-02 15:54:55 +0000 | [diff] [blame] | 2 | /* |
| 3 | * Copyright 2002, 2003 Andi Kleen, SuSE Labs. |
Andrew Scull | b4b6d4a | 2019-01-02 15:54:55 +0000 | [diff] [blame] | 4 | * |
| 5 | * Wrappers of assembly checksum functions for x86-64. |
| 6 | */ |
| 7 | #include <asm/checksum.h> |
| 8 | #include <linux/export.h> |
| 9 | #include <linux/uaccess.h> |
| 10 | #include <asm/smap.h> |
| 11 | |
| 12 | /** |
Olivier Deprez | 157378f | 2022-04-04 15:47:50 +0200 | [diff] [blame^] | 13 | * csum_and_copy_from_user - Copy and checksum from user space. |
Andrew Scull | b4b6d4a | 2019-01-02 15:54:55 +0000 | [diff] [blame] | 14 | * @src: source address (user space) |
| 15 | * @dst: destination address |
| 16 | * @len: number of bytes to be copied. |
| 17 | * @isum: initial sum that is added into the result (32bit unfolded) |
| 18 | * @errp: set to -EFAULT for an bad source address. |
| 19 | * |
| 20 | * Returns an 32bit unfolded checksum of the buffer. |
| 21 | * src and dst are best aligned to 64bits. |
| 22 | */ |
| 23 | __wsum |
Olivier Deprez | 157378f | 2022-04-04 15:47:50 +0200 | [diff] [blame^] | 24 | csum_and_copy_from_user(const void __user *src, void *dst, int len) |
Andrew Scull | b4b6d4a | 2019-01-02 15:54:55 +0000 | [diff] [blame] | 25 | { |
Olivier Deprez | 157378f | 2022-04-04 15:47:50 +0200 | [diff] [blame^] | 26 | __wsum sum; |
| 27 | |
Andrew Scull | b4b6d4a | 2019-01-02 15:54:55 +0000 | [diff] [blame] | 28 | might_sleep(); |
Olivier Deprez | 157378f | 2022-04-04 15:47:50 +0200 | [diff] [blame^] | 29 | if (!user_access_begin(src, len)) |
| 30 | return 0; |
| 31 | sum = csum_partial_copy_generic((__force const void *)src, dst, len); |
| 32 | user_access_end(); |
| 33 | return sum; |
Andrew Scull | b4b6d4a | 2019-01-02 15:54:55 +0000 | [diff] [blame] | 34 | } |
Olivier Deprez | 157378f | 2022-04-04 15:47:50 +0200 | [diff] [blame^] | 35 | EXPORT_SYMBOL(csum_and_copy_from_user); |
Andrew Scull | b4b6d4a | 2019-01-02 15:54:55 +0000 | [diff] [blame] | 36 | |
| 37 | /** |
Olivier Deprez | 157378f | 2022-04-04 15:47:50 +0200 | [diff] [blame^] | 38 | * csum_and_copy_to_user - Copy and checksum to user space. |
Andrew Scull | b4b6d4a | 2019-01-02 15:54:55 +0000 | [diff] [blame] | 39 | * @src: source address |
| 40 | * @dst: destination address (user space) |
| 41 | * @len: number of bytes to be copied. |
| 42 | * @isum: initial sum that is added into the result (32bit unfolded) |
| 43 | * @errp: set to -EFAULT for an bad destination address. |
| 44 | * |
| 45 | * Returns an 32bit unfolded checksum of the buffer. |
| 46 | * src and dst are best aligned to 64bits. |
| 47 | */ |
| 48 | __wsum |
Olivier Deprez | 157378f | 2022-04-04 15:47:50 +0200 | [diff] [blame^] | 49 | csum_and_copy_to_user(const void *src, void __user *dst, int len) |
Andrew Scull | b4b6d4a | 2019-01-02 15:54:55 +0000 | [diff] [blame] | 50 | { |
Olivier Deprez | 157378f | 2022-04-04 15:47:50 +0200 | [diff] [blame^] | 51 | __wsum sum; |
Andrew Scull | b4b6d4a | 2019-01-02 15:54:55 +0000 | [diff] [blame] | 52 | |
| 53 | might_sleep(); |
Olivier Deprez | 157378f | 2022-04-04 15:47:50 +0200 | [diff] [blame^] | 54 | if (!user_access_begin(dst, len)) |
Andrew Scull | b4b6d4a | 2019-01-02 15:54:55 +0000 | [diff] [blame] | 55 | return 0; |
Olivier Deprez | 157378f | 2022-04-04 15:47:50 +0200 | [diff] [blame^] | 56 | sum = csum_partial_copy_generic(src, (void __force *)dst, len); |
| 57 | user_access_end(); |
| 58 | return sum; |
Andrew Scull | b4b6d4a | 2019-01-02 15:54:55 +0000 | [diff] [blame] | 59 | } |
Olivier Deprez | 157378f | 2022-04-04 15:47:50 +0200 | [diff] [blame^] | 60 | EXPORT_SYMBOL(csum_and_copy_to_user); |
Andrew Scull | b4b6d4a | 2019-01-02 15:54:55 +0000 | [diff] [blame] | 61 | |
| 62 | /** |
| 63 | * csum_partial_copy_nocheck - Copy and checksum. |
| 64 | * @src: source address |
| 65 | * @dst: destination address |
| 66 | * @len: number of bytes to be copied. |
| 67 | * @sum: initial sum that is added into the result (32bit unfolded) |
| 68 | * |
| 69 | * Returns an 32bit unfolded checksum of the buffer. |
| 70 | */ |
| 71 | __wsum |
Olivier Deprez | 157378f | 2022-04-04 15:47:50 +0200 | [diff] [blame^] | 72 | csum_partial_copy_nocheck(const void *src, void *dst, int len) |
Andrew Scull | b4b6d4a | 2019-01-02 15:54:55 +0000 | [diff] [blame] | 73 | { |
Olivier Deprez | 157378f | 2022-04-04 15:47:50 +0200 | [diff] [blame^] | 74 | return csum_partial_copy_generic(src, dst, len); |
Andrew Scull | b4b6d4a | 2019-01-02 15:54:55 +0000 | [diff] [blame] | 75 | } |
| 76 | EXPORT_SYMBOL(csum_partial_copy_nocheck); |
| 77 | |
| 78 | __sum16 csum_ipv6_magic(const struct in6_addr *saddr, |
| 79 | const struct in6_addr *daddr, |
| 80 | __u32 len, __u8 proto, __wsum sum) |
| 81 | { |
| 82 | __u64 rest, sum64; |
| 83 | |
| 84 | rest = (__force __u64)htonl(len) + (__force __u64)htons(proto) + |
| 85 | (__force __u64)sum; |
| 86 | |
| 87 | asm(" addq (%[saddr]),%[sum]\n" |
| 88 | " adcq 8(%[saddr]),%[sum]\n" |
| 89 | " adcq (%[daddr]),%[sum]\n" |
| 90 | " adcq 8(%[daddr]),%[sum]\n" |
| 91 | " adcq $0,%[sum]\n" |
| 92 | |
| 93 | : [sum] "=r" (sum64) |
| 94 | : "[sum]" (rest), [saddr] "r" (saddr), [daddr] "r" (daddr)); |
| 95 | |
| 96 | return csum_fold( |
| 97 | (__force __wsum)add32_with_carry(sum64 & 0xffffffff, sum64>>32)); |
| 98 | } |
| 99 | EXPORT_SYMBOL(csum_ipv6_magic); |