xref: /original-bsd/lib/libm/vax/argred.s (revision f052b07a)
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