Path: blob/master/libs/compiler-rt/lib/builtins/udivmodti4.c
4395 views
/* ===-- udivmodti4.c - Implement __udivmodti4 -----------------------------===1*2* The LLVM Compiler Infrastructure3*4* This file is dual licensed under the MIT and the University of Illinois Open5* Source Licenses. See LICENSE.TXT for details.6*7* ===----------------------------------------------------------------------===8*9* This file implements __udivmodti4 for the compiler_rt library.10*11* ===----------------------------------------------------------------------===12*/1314#include "int_lib.h"1516#ifdef CRT_HAS_128BIT1718/* Effects: if rem != 0, *rem = a % b19* Returns: a / b20*/2122/* Translated from Figure 3-40 of The PowerPC Compiler Writer's Guide */2324COMPILER_RT_ABI tu_int25__udivmodti4(tu_int a, tu_int b, tu_int* rem)26{27const unsigned n_udword_bits = sizeof(du_int) * CHAR_BIT;28const unsigned n_utword_bits = sizeof(tu_int) * CHAR_BIT;29utwords n;30n.all = a;31utwords d;32d.all = b;33utwords q;34utwords r;35unsigned sr;36/* special cases, X is unknown, K != 0 */37if (n.s.high == 0)38{39if (d.s.high == 0)40{41/* 0 X42* ---43* 0 X44*/45if (rem)46*rem = n.s.low % d.s.low;47return n.s.low / d.s.low;48}49/* 0 X50* ---51* K X52*/53if (rem)54*rem = n.s.low;55return 0;56}57/* n.s.high != 0 */58if (d.s.low == 0)59{60if (d.s.high == 0)61{62/* K X63* ---64* 0 065*/66if (rem)67*rem = n.s.high % d.s.low;68return n.s.high / d.s.low;69}70/* d.s.high != 0 */71if (n.s.low == 0)72{73/* K 074* ---75* K 076*/77if (rem)78{79r.s.high = n.s.high % d.s.high;80r.s.low = 0;81*rem = r.all;82}83return n.s.high / d.s.high;84}85/* K K86* ---87* K 088*/89if ((d.s.high & (d.s.high - 1)) == 0) /* if d is a power of 2 */90{91if (rem)92{93r.s.low = n.s.low;94r.s.high = n.s.high & (d.s.high - 1);95*rem = r.all;96}97return n.s.high >> __builtin_ctzll(d.s.high);98}99/* K K100* ---101* K 0102*/103sr = __builtin_clzll(d.s.high) - __builtin_clzll(n.s.high);104/* 0 <= sr <= n_udword_bits - 2 or sr large */105if (sr > n_udword_bits - 2)106{107if (rem)108*rem = n.all;109return 0;110}111++sr;112/* 1 <= sr <= n_udword_bits - 1 */113/* q.all = n.all << (n_utword_bits - sr); */114q.s.low = 0;115q.s.high = n.s.low << (n_udword_bits - sr);116/* r.all = n.all >> sr; */117r.s.high = n.s.high >> sr;118r.s.low = (n.s.high << (n_udword_bits - sr)) | (n.s.low >> sr);119}120else /* d.s.low != 0 */121{122if (d.s.high == 0)123{124/* K X125* ---126* 0 K127*/128if ((d.s.low & (d.s.low - 1)) == 0) /* if d is a power of 2 */129{130if (rem)131*rem = n.s.low & (d.s.low - 1);132if (d.s.low == 1)133return n.all;134sr = __builtin_ctzll(d.s.low);135q.s.high = n.s.high >> sr;136q.s.low = (n.s.high << (n_udword_bits - sr)) | (n.s.low >> sr);137return q.all;138}139/* K X140* ---141* 0 K142*/143sr = 1 + n_udword_bits + __builtin_clzll(d.s.low)144- __builtin_clzll(n.s.high);145/* 2 <= sr <= n_utword_bits - 1146* q.all = n.all << (n_utword_bits - sr);147* r.all = n.all >> sr;148*/149if (sr == n_udword_bits)150{151q.s.low = 0;152q.s.high = n.s.low;153r.s.high = 0;154r.s.low = n.s.high;155}156else if (sr < n_udword_bits) // 2 <= sr <= n_udword_bits - 1157{158q.s.low = 0;159q.s.high = n.s.low << (n_udword_bits - sr);160r.s.high = n.s.high >> sr;161r.s.low = (n.s.high << (n_udword_bits - sr)) | (n.s.low >> sr);162}163else // n_udword_bits + 1 <= sr <= n_utword_bits - 1164{165q.s.low = n.s.low << (n_utword_bits - sr);166q.s.high = (n.s.high << (n_utword_bits - sr)) |167(n.s.low >> (sr - n_udword_bits));168r.s.high = 0;169r.s.low = n.s.high >> (sr - n_udword_bits);170}171}172else173{174/* K X175* ---176* K K177*/178sr = __builtin_clzll(d.s.high) - __builtin_clzll(n.s.high);179/*0 <= sr <= n_udword_bits - 1 or sr large */180if (sr > n_udword_bits - 1)181{182if (rem)183*rem = n.all;184return 0;185}186++sr;187/* 1 <= sr <= n_udword_bits188* q.all = n.all << (n_utword_bits - sr);189* r.all = n.all >> sr;190*/191q.s.low = 0;192if (sr == n_udword_bits)193{194q.s.high = n.s.low;195r.s.high = 0;196r.s.low = n.s.high;197}198else199{200r.s.high = n.s.high >> sr;201r.s.low = (n.s.high << (n_udword_bits - sr)) | (n.s.low >> sr);202q.s.high = n.s.low << (n_udword_bits - sr);203}204}205}206/* Not a special case207* q and r are initialized with:208* q.all = n.all << (n_utword_bits - sr);209* r.all = n.all >> sr;210* 1 <= sr <= n_utword_bits - 1211*/212su_int carry = 0;213for (; sr > 0; --sr)214{215/* r:q = ((r:q) << 1) | carry */216r.s.high = (r.s.high << 1) | (r.s.low >> (n_udword_bits - 1));217r.s.low = (r.s.low << 1) | (q.s.high >> (n_udword_bits - 1));218q.s.high = (q.s.high << 1) | (q.s.low >> (n_udword_bits - 1));219q.s.low = (q.s.low << 1) | carry;220/* carry = 0;221* if (r.all >= d.all)222* {223* r.all -= d.all;224* carry = 1;225* }226*/227const ti_int s = (ti_int)(d.all - r.all - 1) >> (n_utword_bits - 1);228carry = s & 1;229r.all -= d.all & s;230}231q.all = (q.all << 1) | carry;232if (rem)233*rem = r.all;234return q.all;235}236237#endif /* CRT_HAS_128BIT */238239240