1# Copyright (c) 1985 Regents of the University of California. 2# All rights reserved. 3# 4# Redistribution and use in source and binary forms are permitted 5# provided that the above copyright notice and this paragraph are 6# duplicated in all such forms and that any documentation, 7# advertising materials, and other materials related to such 8# distribution and use acknowledge that the software was developed 9# by the University of California, Berkeley. The name of the 10# University may not be used to endorse or promote products derived 11# from this software without specific prior written permission. 12# THIS SOFTWARE IS PROVIDED ``AS IS'' AND WITHOUT ANY EXPRESS OR 13# IMPLIED WARRANTIES, INCLUDING, WITHOUT LIMITATION, THE IMPLIED 14# WARRANTIES OF MERCHANTIBILITY AND FITNESS FOR A PARTICULAR PURPOSE. 15# 16# All recipients should regard themselves as participants in an ongoing 17# research project and hence should feel obligated to report their 18# experiences (good or bad) with these elementary function codes, using 19# the sendbug(8) program, to the authors. 20# 21# @(#)argred.s 5.3 (Berkeley) 06/30/88 22# 23 .data 24 .align 2 25_sccsid: 26.asciz "@(#)argred.s 1.1 (Berkeley) 8/21/85; 5.3 (ucb.elefunt) 06/30/88" 27 28# libm$argred implements Bob Corbett's argument reduction and 29# libm$sincos implements Peter Tang's double precision sin/cos. 30# 31# Note: The two entry points libm$argred and libm$sincos are meant 32# to be used only by _sin, _cos and _tan. 33# 34# method: true range reduction to [-pi/4,pi/4], P. Tang & B. Corbett 35# S. McDonald, April 4, 1985 36# 37 .globl libm$argred 38 .globl libm$sincos 39 .text 40 .align 1 41 42libm$argred: 43# 44# Compare the argument with the largest possible that can 45# be reduced by table lookup. r3 := |x| will be used in table_lookup . 46# 47 movd r0,r3 48 bgeq abs1 49 mnegd r3,r3 50abs1: 51 cmpd r3,$0d+4.55530934770520019583e+01 52 blss small_arg 53 jsb trigred 54 rsb 55small_arg: 56 jsb table_lookup 57 rsb 58# 59# At this point, 60# r0 contains the quadrant number, 0, 1, 2, or 3; 61# r2/r1 contains the reduced argument as a D-format number; 62# r3 contains a F-format extension to the reduced argument; 63# r4 contains a 0 or 1 corresponding to a sin or cos entry. 64# 65libm$sincos: 66# 67# Compensate for a cosine entry by adding one to the quadrant number. 68# 69 addl2 r4,r0 70# 71# Polyd clobbers r5-r0 ; save X in r7/r6 . 72# This can be avoided by rewriting trigred . 73# 74 movd r1,r6 75# 76# Likewise, save alpha in r8 . 77# This can be avoided by rewriting trigred . 78# 79 movf r3,r8 80# 81# Odd or even quadrant? cosine if odd, sine otherwise. 82# Save floor(quadrant/2) in r9 ; it determines the final sign. 83# 84 rotl $-1,r0,r9 85 blss cosine 86sine: 87 muld2 r1,r1 # Xsq = X * X 88 cmpw $0x2480,r1 # [zl] Xsq > 2^-56? 89 blss 1f # [zl] yes, go ahead and do polyd 90 clrq r1 # [zl] work around 11/780 FPA polyd bug 911: 92 polyd r1,$7,sin_coef # Q = P(Xsq) , of deg 7 93 mulf3 $0f3.0,r8,r4 # beta = 3 * alpha 94 mulf2 r0,r4 # beta = Q * beta 95 addf2 r8,r4 # beta = alpha + beta 96 muld2 r6,r0 # S(X) = X * Q 97# cvtfd r4,r4 ... r5 = 0 after a polyd. 98 addd2 r4,r0 # S(X) = beta + S(X) 99 addd2 r6,r0 # S(X) = X + S(X) 100 brb done 101cosine: 102 muld2 r6,r6 # Xsq = X * X 103 beql zero_arg 104 mulf2 r1,r8 # beta = X * alpha 105 polyd r6,$7,cos_coef # Q = P'(Xsq) , of deg 7 106 subd3 r0,r8,r0 # beta = beta - Q 107 subw2 $0x80,r6 # Xsq = Xsq / 2 108 addd2 r0,r6 # Xsq = Xsq + beta 109zero_arg: 110 subd3 r6,$0d1.0,r0 # C(X) = 1 - Xsq 111done: 112 blbc r9,even 113 mnegd r0,r0 114even: 115 rsb 116 117.data 118.align 2 119 120sin_coef: 121 .double 0d-7.53080332264191085773e-13 # s7 = 2^-29 -1.a7f2504ffc49f8.. 122 .double 0d+1.60573519267703489121e-10 # s6 = 2^-21 1.611adaede473c8.. 123 .double 0d-2.50520965150706067211e-08 # s5 = 2^-1a -1.ae644921ed8382.. 124 .double 0d+2.75573191800593885716e-06 # s4 = 2^-13 1.71de3a4b884278.. 125 .double 0d-1.98412698411850507950e-04 # s3 = 2^-0d -1.a01a01a0125e7d.. 126 .double 0d+8.33333333333325688985e-03 # s2 = 2^-07 1.11111111110e50 127 .double 0d-1.66666666666666664354e-01 # s1 = 2^-03 -1.55555555555554 128 .double 0d+0.00000000000000000000e+00 # s0 = 0 129 130cos_coef: 131 .double 0d-1.13006966202629430300e-11 # s7 = 2^-25 -1.8D9BA04D1374BE.. 132 .double 0d+2.08746646574796004700e-09 # s6 = 2^-1D 1.1EE632650350BA.. 133 .double 0d-2.75573073031284417300e-07 # s5 = 2^-16 -1.27E4F31411719E.. 134 .double 0d+2.48015872682668025200e-05 # s4 = 2^-10 1.A01A0196B902E8.. 135 .double 0d-1.38888888888464709200e-03 # s3 = 2^-0A -1.6C16C16C11FACE.. 136 .double 0d+4.16666666666664761400e-02 # s2 = 2^-05 1.5555555555539E 137 .double 0d+0.00000000000000000000e+00 # s1 = 0 138 .double 0d+0.00000000000000000000e+00 # s0 = 0 139 140# 141# Multiples of pi/2 expressed as the sum of three doubles, 142# 143# trailing: n * pi/2 , n = 0, 1, 2, ..., 29 144# trailing[n] , 145# 146# middle: n * pi/2 , n = 0, 1, 2, ..., 29 147# middle[n] , 148# 149# leading: n * pi/2 , n = 0, 1, 2, ..., 29 150# leading[n] , 151# 152# where 153# leading[n] := (n * pi/2) rounded, 154# middle[n] := (n * pi/2 - leading[n]) rounded, 155# trailing[n] := (( n * pi/2 - leading[n]) - middle[n]) rounded . 156 157trailing: 158 .double 0d+0.00000000000000000000e+00 # 0 * pi/2 trailing 159 .double 0d+4.33590506506189049611e-35 # 1 * pi/2 trailing 160 .double 0d+8.67181013012378099223e-35 # 2 * pi/2 trailing 161 .double 0d+1.30077151951856714215e-34 # 3 * pi/2 trailing 162 .double 0d+1.73436202602475619845e-34 # 4 * pi/2 trailing 163 .double 0d-1.68390735624352669192e-34 # 5 * pi/2 trailing 164 .double 0d+2.60154303903713428430e-34 # 6 * pi/2 trailing 165 .double 0d-8.16726343231148352150e-35 # 7 * pi/2 trailing 166 .double 0d+3.46872405204951239689e-34 # 8 * pi/2 trailing 167 .double 0d+3.90231455855570147991e-34 # 9 * pi/2 trailing 168 .double 0d-3.36781471248705338384e-34 # 10 * pi/2 trailing 169 .double 0d-1.06379439835298071785e-33 # 11 * pi/2 trailing 170 .double 0d+5.20308607807426856861e-34 # 12 * pi/2 trailing 171 .double 0d+5.63667658458045770509e-34 # 13 * pi/2 trailing 172 .double 0d-1.63345268646229670430e-34 # 14 * pi/2 trailing 173 .double 0d-1.19986217995610764801e-34 # 15 * pi/2 trailing 174 .double 0d+6.93744810409902479378e-34 # 16 * pi/2 trailing 175 .double 0d-8.03640094449267300110e-34 # 17 * pi/2 trailing 176 .double 0d+7.80462911711140295982e-34 # 18 * pi/2 trailing 177 .double 0d-7.16921993148029483506e-34 # 19 * pi/2 trailing 178 .double 0d-6.73562942497410676769e-34 # 20 * pi/2 trailing 179 .double 0d-6.30203891846791677593e-34 # 21 * pi/2 trailing 180 .double 0d-2.12758879670596143570e-33 # 22 * pi/2 trailing 181 .double 0d+2.53800212047402350390e-33 # 23 * pi/2 trailing 182 .double 0d+1.04061721561485371372e-33 # 24 * pi/2 trailing 183 .double 0d+6.11729905311472319056e-32 # 25 * pi/2 trailing 184 .double 0d+1.12733531691609154102e-33 # 26 * pi/2 trailing 185 .double 0d-3.70049587943078297272e-34 # 27 * pi/2 trailing 186 .double 0d-3.26690537292459340860e-34 # 28 * pi/2 trailing 187 .double 0d-1.14812616507957271361e-34 # 29 * pi/2 trailing 188 189middle: 190 .double 0d+0.00000000000000000000e+00 # 0 * pi/2 middle 191 .double 0d+5.72118872610983179676e-18 # 1 * pi/2 middle 192 .double 0d+1.14423774522196635935e-17 # 2 * pi/2 middle 193 .double 0d-3.83475850529283316309e-17 # 3 * pi/2 middle 194 .double 0d+2.28847549044393271871e-17 # 4 * pi/2 middle 195 .double 0d-2.69052076007086676522e-17 # 5 * pi/2 middle 196 .double 0d-7.66951701058566632618e-17 # 6 * pi/2 middle 197 .double 0d-1.54628301484890040587e-17 # 7 * pi/2 middle 198 .double 0d+4.57695098088786543741e-17 # 8 * pi/2 middle 199 .double 0d+1.07001849766246313192e-16 # 9 * pi/2 middle 200 .double 0d-5.38104152014173353044e-17 # 10 * pi/2 middle 201 .double 0d-2.14622680169080983801e-16 # 11 * pi/2 middle 202 .double 0d-1.53390340211713326524e-16 # 12 * pi/2 middle 203 .double 0d-9.21580002543456677056e-17 # 13 * pi/2 middle 204 .double 0d-3.09256602969780081173e-17 # 14 * pi/2 middle 205 .double 0d+3.03066796603896507006e-17 # 15 * pi/2 middle 206 .double 0d+9.15390196177573087482e-17 # 16 * pi/2 middle 207 .double 0d+1.52771359575124969107e-16 # 17 * pi/2 middle 208 .double 0d+2.14003699532492626384e-16 # 18 * pi/2 middle 209 .double 0d-1.68853170360202329427e-16 # 19 * pi/2 middle 210 .double 0d-1.07620830402834670609e-16 # 20 * pi/2 middle 211 .double 0d+3.97700719404595604379e-16 # 21 * pi/2 middle 212 .double 0d-4.29245360338161967602e-16 # 22 * pi/2 middle 213 .double 0d-3.68013020380794313406e-16 # 23 * pi/2 middle 214 .double 0d-3.06780680423426653047e-16 # 24 * pi/2 middle 215 .double 0d-2.45548340466059054318e-16 # 25 * pi/2 middle 216 .double 0d-1.84316000508691335411e-16 # 26 * pi/2 middle 217 .double 0d-1.23083660551323675053e-16 # 27 * pi/2 middle 218 .double 0d-6.18513205939560162346e-17 # 28 * pi/2 middle 219 .double 0d-6.18980636588357585202e-19 # 29 * pi/2 middle 220 221leading: 222 .double 0d+0.00000000000000000000e+00 # 0 * pi/2 leading 223 .double 0d+1.57079632679489661351e+00 # 1 * pi/2 leading 224 .double 0d+3.14159265358979322702e+00 # 2 * pi/2 leading 225 .double 0d+4.71238898038468989604e+00 # 3 * pi/2 leading 226 .double 0d+6.28318530717958645404e+00 # 4 * pi/2 leading 227 .double 0d+7.85398163397448312306e+00 # 5 * pi/2 leading 228 .double 0d+9.42477796076937979208e+00 # 6 * pi/2 leading 229 .double 0d+1.09955742875642763501e+01 # 7 * pi/2 leading 230 .double 0d+1.25663706143591729081e+01 # 8 * pi/2 leading 231 .double 0d+1.41371669411540694661e+01 # 9 * pi/2 leading 232 .double 0d+1.57079632679489662461e+01 # 10 * pi/2 leading 233 .double 0d+1.72787595947438630262e+01 # 11 * pi/2 leading 234 .double 0d+1.88495559215387595842e+01 # 12 * pi/2 leading 235 .double 0d+2.04203522483336561422e+01 # 13 * pi/2 leading 236 .double 0d+2.19911485751285527002e+01 # 14 * pi/2 leading 237 .double 0d+2.35619449019234492582e+01 # 15 * pi/2 leading 238 .double 0d+2.51327412287183458162e+01 # 16 * pi/2 leading 239 .double 0d+2.67035375555132423742e+01 # 17 * pi/2 leading 240 .double 0d+2.82743338823081389322e+01 # 18 * pi/2 leading 241 .double 0d+2.98451302091030359342e+01 # 19 * pi/2 leading 242 .double 0d+3.14159265358979324922e+01 # 20 * pi/2 leading 243 .double 0d+3.29867228626928286062e+01 # 21 * pi/2 leading 244 .double 0d+3.45575191894877260523e+01 # 22 * pi/2 leading 245 .double 0d+3.61283155162826226103e+01 # 23 * pi/2 leading 246 .double 0d+3.76991118430775191683e+01 # 24 * pi/2 leading 247 .double 0d+3.92699081698724157263e+01 # 25 * pi/2 leading 248 .double 0d+4.08407044966673122843e+01 # 26 * pi/2 leading 249 .double 0d+4.24115008234622088423e+01 # 27 * pi/2 leading 250 .double 0d+4.39822971502571054003e+01 # 28 * pi/2 leading 251 .double 0d+4.55530934770520019583e+01 # 29 * pi/2 leading 252 253twoOverPi: 254 .double 0d+6.36619772367581343076e-01 255 .text 256 .align 1 257 258table_lookup: 259 muld3 r3,twoOverPi,r0 260 cvtrdl r0,r0 # n = nearest int to ((2/pi)*|x|) rnded 261 mull3 $8,r0,r5 262 subd2 leading(r5),r3 # p = (|x| - leading n*pi/2) exactly 263 subd3 middle(r5),r3,r1 # q = (p - middle n*pi/2) rounded 264 subd2 r1,r3 # r = (p - q) 265 subd2 middle(r5),r3 # r = r - middle n*pi/2 266 subd2 trailing(r5),r3 # r = r - trailing n*pi/2 rounded 267# 268# If the original argument was negative, 269# negate the reduce argument and 270# adjust the octant/quadrant number. 271# 272 tstw 4(ap) 273 bgeq abs2 274 mnegf r1,r1 275 mnegf r3,r3 276# subb3 r0,$8,r0 ...used for pi/4 reduction -S.McD 277 subb3 r0,$4,r0 278abs2: 279# 280# Clear all unneeded octant/quadrant bits. 281# 282# bicb2 $0xf8,r0 ...used for pi/4 reduction -S.McD 283 bicb2 $0xfc,r0 284 rsb 285# 286# p.0 287 .text 288 .align 2 289# 290# Only 256 (actually 225) bits of 2/pi are needed for VAX double 291# precision; this was determined by enumerating all the nearest 292# machine integer multiples of pi/2 using continued fractions. 293# (8a8d3673775b7ff7 required the most bits.) -S.McD 294# 295 .long 0 296 .long 0 297 .long 0xaef1586d 298 .long 0x9458eaf7 299 .long 0x10e4107f 300 .long 0xd8a5664f 301 .long 0x4d377036 302 .long 0x09d5f47d 303 .long 0x91054a7f 304 .long 0xbe60db93 305bits2opi: 306 .long 0x00000028 307 .long 0 308# 309# Note: wherever you see the word `octant', read `quadrant'. 310# Currently this code is set up for pi/2 argument reduction. 311# By uncommenting/commenting the appropriate lines, it will 312# also serve as a pi/4 argument reduction code. 313# 314 315# p.1 316# Trigred preforms argument reduction 317# for the trigonometric functions. It 318# takes one input argument, a D-format 319# number in r1/r0 . The magnitude of 320# the input argument must be greater 321# than or equal to 1/2 . Trigred produces 322# three results: the number of the octant 323# occupied by the argument, the reduced 324# argument, and an extension of the 325# reduced argument. The octant number is 326# returned in r0 . The reduced argument 327# is returned as a D-format number in 328# r2/r1 . An 8 bit extension of the 329# reduced argument is returned as an 330# F-format number in r3. 331# p.2 332trigred: 333# 334# Save the sign of the input argument. 335# 336 movw r0,-(sp) 337# 338# Extract the exponent field. 339# 340 extzv $7,$7,r0,r2 341# 342# Convert the fraction part of the input 343# argument into a quadword integer. 344# 345 bicw2 $0xff80,r0 346 bisb2 $0x80,r0 # -S.McD 347 rotl $16,r0,r0 348 rotl $16,r1,r1 349# 350# If r1 is negative, add 1 to r0 . This 351# adjustment is made so that the two's 352# complement multiplications done later 353# will produce unsigned results. 354# 355 bgeq posmid 356 incl r0 357posmid: 358# p.3 359# 360# Set r3 to the address of the first quadword 361# used to obtain the needed portion of 2/pi . 362# The address is longword aligned to ensure 363# efficient access. 364# 365 ashl $-3,r2,r3 366 bicb2 $3,r3 367 subl3 r3,$bits2opi,r3 368# 369# Set r2 to the size of the shift needed to 370# obtain the correct portion of 2/pi . 371# 372 bicb2 $0xe0,r2 373# p.4 374# 375# Move the needed 128 bits of 2/pi into 376# r11 - r8 . Adjust the numbers to allow 377# for unsigned multiplication. 378# 379 ashq r2,(r3),r10 380 381 subl2 $4,r3 382 ashq r2,(r3),r9 383 bgeq signoff1 384 incl r11 385signoff1: 386 subl2 $4,r3 387 ashq r2,(r3),r8 388 bgeq signoff2 389 incl r10 390signoff2: 391 subl2 $4,r3 392 ashq r2,(r3),r7 393 bgeq signoff3 394 incl r9 395signoff3: 396# p.5 397# 398# Multiply the contents of r0/r1 by the 399# slice of 2/pi in r11 - r8 . 400# 401 emul r0,r8,$0,r4 402 emul r0,r9,r5,r5 403 emul r0,r10,r6,r6 404 405 emul r1,r8,$0,r7 406 emul r1,r9,r8,r8 407 emul r1,r10,r9,r9 408 emul r1,r11,r10,r10 409 410 addl2 r4,r8 411 adwc r5,r9 412 adwc r6,r10 413# p.6 414# 415# If there are more than five leading zeros 416# after the first two quotient bits or if there 417# are more than five leading ones after the first 418# two quotient bits, generate more fraction bits. 419# Otherwise, branch to code to produce the result. 420# 421 bicl3 $0xc1ffffff,r10,r4 422 beql more1 423 cmpl $0x3e000000,r4 424 bneq result 425more1: 426# p.7 427# 428# generate another 32 result bits. 429# 430 subl2 $4,r3 431 ashq r2,(r3),r5 432 bgeq signoff4 433 434 emul r1,r6,$0,r4 435 addl2 r1,r5 436 emul r0,r6,r5,r5 437 addl2 r0,r6 438 brb addbits1 439 440signoff4: 441 emul r1,r6,$0,r4 442 emul r0,r6,r5,r5 443 444addbits1: 445 addl2 r5,r7 446 adwc r6,r8 447 adwc $0,r9 448 adwc $0,r10 449# p.8 450# 451# Check for massive cancellation. 452# 453 bicl3 $0xc0000000,r10,r6 454# bneq more2 -S.McD Test was backwards 455 beql more2 456 cmpl $0x3fffffff,r6 457 bneq result 458more2: 459# p.9 460# 461# If massive cancellation has occurred, 462# generate another 24 result bits. 463# Testing has shown there will always be 464# enough bits after this point. 465# 466 subl2 $4,r3 467 ashq r2,(r3),r5 468 bgeq signoff5 469 470 emul r0,r6,r4,r5 471 addl2 r0,r6 472 brb addbits2 473 474signoff5: 475 emul r0,r6,r4,r5 476 477addbits2: 478 addl2 r6,r7 479 adwc $0,r8 480 adwc $0,r9 481 adwc $0,r10 482# p.10 483# 484# The following code produces the reduced 485# argument from the product bits contained 486# in r10 - r7 . 487# 488result: 489# 490# Extract the octant number from r10 . 491# 492# extzv $29,$3,r10,r0 ...used for pi/4 reduction -S.McD 493 extzv $30,$2,r10,r0 494# 495# Clear the octant bits in r10 . 496# 497# bicl2 $0xe0000000,r10 ...used for pi/4 reduction -S.McD 498 bicl2 $0xc0000000,r10 499# 500# Zero the sign flag. 501# 502 clrl r5 503# p.11 504# 505# Check to see if the fraction is greater than 506# or equal to one-half. If it is, add one 507# to the octant number, set the sign flag 508# on, and replace the fraction with 1 minus 509# the fraction. 510# 511# bitl $0x10000000,r10 ...used for pi/4 reduction -S.McD 512 bitl $0x20000000,r10 513 beql small 514 incl r0 515 incl r5 516# subl3 r10,$0x1fffffff,r10 ...used for pi/4 reduction -S.McD 517 subl3 r10,$0x3fffffff,r10 518 mcoml r9,r9 519 mcoml r8,r8 520 mcoml r7,r7 521small: 522# p.12 523# 524## Test whether the first 29 bits of the ...used for pi/4 reduction -S.McD 525# Test whether the first 30 bits of the 526# fraction are zero. 527# 528 tstl r10 529 beql tiny 530# 531# Find the position of the first one bit in r10 . 532# 533 cvtld r10,r1 534 extzv $7,$7,r1,r1 535# 536# Compute the size of the shift needed. 537# 538 subl3 r1,$32,r6 539# 540# Shift up the high order 64 bits of the 541# product. 542# 543 ashq r6,r9,r10 544 ashq r6,r8,r9 545 brb mult 546# p.13 547# 548# Test to see if the sign bit of r9 is on. 549# 550tiny: 551 tstl r9 552 bgeq tinier 553# 554# If it is, shift the product bits up 32 bits. 555# 556 movl $32,r6 557 movq r8,r10 558 tstl r10 559 brb mult 560# p.14 561# 562# Test whether r9 is zero. It is probably 563# impossible for both r10 and r9 to be 564# zero, but until proven to be so, the test 565# must be made. 566# 567tinier: 568 beql zero 569# 570# Find the position of the first one bit in r9 . 571# 572 cvtld r9,r1 573 extzv $7,$7,r1,r1 574# 575# Compute the size of the shift needed. 576# 577 subl3 r1,$32,r1 578 addl3 $32,r1,r6 579# 580# Shift up the high order 64 bits of the 581# product. 582# 583 ashq r1,r8,r10 584 ashq r1,r7,r9 585 brb mult 586# p.15 587# 588# The following code sets the reduced 589# argument to zero. 590# 591zero: 592 clrl r1 593 clrl r2 594 clrl r3 595 brw return 596# p.16 597# 598# At this point, r0 contains the octant number, 599# r6 indicates the number of bits the fraction 600# has been shifted, r5 indicates the sign of 601# the fraction, r11/r10 contain the high order 602# 64 bits of the fraction, and the condition 603# codes indicate where the sign bit of r10 604# is on. The following code multiplies the 605# fraction by pi/2 . 606# 607mult: 608# 609# Save r11/r10 in r4/r1 . -S.McD 610 movl r11,r4 611 movl r10,r1 612# 613# If the sign bit of r10 is on, add 1 to r11 . 614# 615 bgeq signoff6 616 incl r11 617signoff6: 618# p.17 619# 620# Move pi/2 into r3/r2 . 621# 622 movq $0xc90fdaa22168c235,r2 623# 624# Multiply the fraction by the portion of pi/2 625# in r2 . 626# 627 emul r2,r10,$0,r7 628 emul r2,r11,r8,r7 629# 630# Multiply the fraction by the portion of pi/2 631# in r3 . 632 emul r3,r10,$0,r9 633 emul r3,r11,r10,r10 634# 635# Add the product bits together. 636# 637 addl2 r7,r9 638 adwc r8,r10 639 adwc $0,r11 640# 641# Compensate for not sign extending r8 above.-S.McD 642# 643 tstl r8 644 bgeq signoff6a 645 decl r11 646signoff6a: 647# 648# Compensate for r11/r10 being unsigned. -S.McD 649# 650 addl2 r2,r10 651 adwc r3,r11 652# 653# Compensate for r3/r2 being unsigned. -S.McD 654# 655 addl2 r1,r10 656 adwc r4,r11 657# p.18 658# 659# If the sign bit of r11 is zero, shift the 660# product bits up one bit and increment r6 . 661# 662 blss signon 663 incl r6 664 ashq $1,r10,r10 665 tstl r9 666 bgeq signoff7 667 incl r10 668signoff7: 669signon: 670# p.19 671# 672# Shift the 56 most significant product 673# bits into r9/r8 . The sign extension 674# will be handled later. 675# 676 ashq $-8,r10,r8 677# 678# Convert the low order 8 bits of r10 679# into an F-format number. 680# 681 cvtbf r10,r3 682# 683# If the result of the conversion was 684# negative, add 1 to r9/r8 . 685# 686 bgeq chop 687 incl r8 688 adwc $0,r9 689# 690# If r9 is now zero, branch to special 691# code to handle that possibility. 692# 693 beql carryout 694chop: 695# p.20 696# 697# Convert the number in r9/r8 into 698# D-format number in r2/r1 . 699# 700 rotl $16,r8,r2 701 rotl $16,r9,r1 702# 703# Set the exponent field to the appropriate 704# value. Note that the extra bits created by 705# sign extension are now eliminated. 706# 707 subw3 r6,$131,r6 708 insv r6,$7,$9,r1 709# 710# Set the exponent field of the F-format 711# number in r3 to the appropriate value. 712# 713 tstf r3 714 beql return 715# extzv $7,$8,r3,r4 -S.McD 716 extzv $7,$7,r3,r4 717 addw2 r4,r6 718# subw2 $217,r6 -S.McD 719 subw2 $64,r6 720 insv r6,$7,$8,r3 721 brb return 722# p.21 723# 724# The following code generates the appropriate 725# result for the unlikely possibility that 726# rounding the number in r9/r8 resulted in 727# a carry out. 728# 729carryout: 730 clrl r1 731 clrl r2 732 subw3 r6,$132,r6 733 insv r6,$7,$9,r1 734 tstf r3 735 beql return 736 extzv $7,$8,r3,r4 737 addw2 r4,r6 738 subw2 $218,r6 739 insv r6,$7,$8,r3 740# p.22 741# 742# The following code makes an needed 743# adjustments to the signs of the 744# results or to the octant number, and 745# then returns. 746# 747return: 748# 749# Test if the fraction was greater than or 750# equal to 1/2 . If so, negate the reduced 751# argument. 752# 753 blbc r5,signoff8 754 mnegf r1,r1 755 mnegf r3,r3 756signoff8: 757# p.23 758# 759# If the original argument was negative, 760# negate the reduce argument and 761# adjust the octant number. 762# 763 tstw (sp)+ 764 bgeq signoff9 765 mnegf r1,r1 766 mnegf r3,r3 767# subb3 r0,$8,r0 ...used for pi/4 reduction -S.McD 768 subb3 r0,$4,r0 769signoff9: 770# 771# Clear all unneeded octant bits. 772# 773# bicb2 $0xf8,r0 ...used for pi/4 reduction -S.McD 774 bicb2 $0xfc,r0 775# 776# Return. 777# 778 rsb 779