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