mirror of
https://mirrors.bfsu.edu.cn/git/linux.git
synced 2024-11-19 02:04:19 +08:00
588cb88ced
Based on 1 normalized pattern(s): this program is free software you can redistribute it and or modify it under the terms of the gnu general public license as published by the free software foundation either version 2 of the license or at your option any later version this program is distributed in the hope that it will be useful but without any warranty without even the implied warranty of merchantability or fitness for a particular purpose see the gnu general public license for more details you should have received a copy of the gnu general public license along with this program if not see the file copying or write to the free software foundation inc extracted by the scancode license scanner the SPDX license identifier GPL-2.0-or-later has been chosen to replace the boilerplate/reference in 12 file(s). Signed-off-by: Thomas Gleixner <tglx@linutronix.de> Reviewed-by: Richard Fontana <rfontana@redhat.com> Reviewed-by: Allison Randal <allison@lohutok.net> Cc: linux-spdx@vger.kernel.org Link: https://lkml.kernel.org/r/20190523091651.231300438@linutronix.de Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
61 lines
1.7 KiB
C
61 lines
1.7 KiB
C
// SPDX-License-Identifier: GPL-2.0-or-later
|
|
/*
|
|
*/
|
|
|
|
#include <linux/export.h>
|
|
#include <linux/libgcc.h>
|
|
|
|
#define W_TYPE_SIZE 32
|
|
|
|
#define __ll_B ((unsigned long) 1 << (W_TYPE_SIZE / 2))
|
|
#define __ll_lowpart(t) ((unsigned long) (t) & (__ll_B - 1))
|
|
#define __ll_highpart(t) ((unsigned long) (t) >> (W_TYPE_SIZE / 2))
|
|
|
|
/* If we still don't have umul_ppmm, define it using plain C. */
|
|
#if !defined(umul_ppmm)
|
|
#define umul_ppmm(w1, w0, u, v) \
|
|
do { \
|
|
unsigned long __x0, __x1, __x2, __x3; \
|
|
unsigned short __ul, __vl, __uh, __vh; \
|
|
\
|
|
__ul = __ll_lowpart(u); \
|
|
__uh = __ll_highpart(u); \
|
|
__vl = __ll_lowpart(v); \
|
|
__vh = __ll_highpart(v); \
|
|
\
|
|
__x0 = (unsigned long) __ul * __vl; \
|
|
__x1 = (unsigned long) __ul * __vh; \
|
|
__x2 = (unsigned long) __uh * __vl; \
|
|
__x3 = (unsigned long) __uh * __vh; \
|
|
\
|
|
__x1 += __ll_highpart(__x0); /* this can't give carry */\
|
|
__x1 += __x2; /* but this indeed can */ \
|
|
if (__x1 < __x2) /* did we get it? */ \
|
|
__x3 += __ll_B; /* yes, add it in the proper pos */ \
|
|
\
|
|
(w1) = __x3 + __ll_highpart(__x1); \
|
|
(w0) = __ll_lowpart(__x1) * __ll_B + __ll_lowpart(__x0);\
|
|
} while (0)
|
|
#endif
|
|
|
|
#if !defined(__umulsidi3)
|
|
#define __umulsidi3(u, v) ({ \
|
|
DWunion __w; \
|
|
umul_ppmm(__w.s.high, __w.s.low, u, v); \
|
|
__w.ll; \
|
|
})
|
|
#endif
|
|
|
|
long long notrace __muldi3(long long u, long long v)
|
|
{
|
|
const DWunion uu = {.ll = u};
|
|
const DWunion vv = {.ll = v};
|
|
DWunion w = {.ll = __umulsidi3(uu.s.low, vv.s.low)};
|
|
|
|
w.s.high += ((unsigned long) uu.s.low * (unsigned long) vv.s.high
|
|
+ (unsigned long) uu.s.high * (unsigned long) vv.s.low);
|
|
|
|
return w.ll;
|
|
}
|
|
EXPORT_SYMBOL(__muldi3);
|