aboutsummaryrefslogtreecommitdiff
path: root/src/libmp/386/mpvecdigmuladd.s
blob: 8c92f61f426fe282f1bacd88133637bcee1ec2e7 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
#
#	mpvecdigmul(mpdigit *b, int n, mpdigit m, mpdigit *p)
#
#	p += b*m
#
#	each step look like:
#		hi,lo = m*b[i]
#		lo += oldhi + carry
#		hi += carry
#		p[i] += lo
#		oldhi = hi
#
#	the registers are:
#		hi = DX		- constrained by hardware
#		lo = AX		- constrained by hardware
#		b+n = SI	- can't be BP
#		p+n = DI	- can't be BP
#		i-n = BP
#		m = BX
#		oldhi = CX
#		
 
.text

.p2align 2,0x90
.globl mpvecdigmuladd
mpvecdigmuladd:
	# Prelude 
	pushl %ebp		# save on stack 
	pushl %ebx
	pushl %esi
	pushl %edi

	leal 20(%esp), %ebp		# %ebp = FP for now 
	movl	0(%ebp), %esi		# b 
	movl	4(%ebp), %ecx		# n 
	movl	8(%ebp), %ebx		# m 
	movl	12(%ebp), %edi		# p 
	movl	%ecx, %ebp
	negl	%ebp			# BP = -n 
	shll	$2, %ecx
	addl	%ecx, %esi		# SI = b + n 
	addl	%ecx, %edi		# DI = p + n 
	xorl	%ecx, %ecx
_muladdloop:
	movl	(%esi, %ebp, 4), %eax	# lo = b[i] 
	mull	%ebx			# hi, lo = b[i] * m 
	addl	%ecx,%eax		# lo += oldhi 
	jae	_muladdnocarry1
	incl	%edx			# hi += carry 
_muladdnocarry1:
	addl	%eax, (%edi, %ebp, 4)	# p[i] += lo 
	jae	_muladdnocarry2
	incl	%edx			# hi += carry 
_muladdnocarry2:
	movl	%edx, %ecx		# oldhi = hi 
	incl	%ebp			# i++ 
	jnz	_muladdloop
	xorl	%eax, %eax
	addl	%ecx, (%edi, %ebp, 4)	# p[n] + oldhi 
	adcl	%eax, %eax		# return carry out of p[n] 

	# Postlude 
	popl %edi
	popl %esi
	popl %ebx
	popl %ebp
	ret