aboutsummaryrefslogtreecommitdiffstats
path: root/contrib/libs/cxxsupp/builtins/avr/udivmodqi4.S
blob: 99aec34429361f6e6e96c5023e5b6568c859b922 (plain) (blame)
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
//===------------ udivmodqi4.S - uint8 div & mod --------------------------===//
//
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
// See https://llvm.org/LICENSE.txt for license information.
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
//
//===----------------------------------------------------------------------===//
//
// As described at
// https://gcc.gnu.org/wiki/avr-gcc#Exceptions_to_the_Calling_Convention, the
// prototype is `struct {uint8, uint8} __udivmodqi4(uint8, uint8)`.
// The uint8 quotient is returned via R24, and the uint8 remainder is returned
// via R25, while R23 is clobbered.
//
//===----------------------------------------------------------------------===//

	.text
	.align 2

	.globl __udivmodqi4
	.type  __udivmodqi4, @function

__udivmodqi4:
	sub     r25, r25           ; Initialize the remainder to zero.
	ldi     r23, 9             ; Only loop 8 rounds for uint8.

__udivmodqi4_loop:
	adc     r24, r24
	dec     r23
	breq    __udivmodqi4_end
	adc     r25, r25
	cp      r25, r22           ; Compare with the divisor.
	brcs    __udivmodqi4_loop
	sub     r25, r22           ; Subtract the divisor.
	rjmp    __udivmodqi4_loop

__udivmodqi4_end:
	com     r24                ; The uint8 quotient is returned via R24.
	ret                        ; The uint8 remainder is returned via R25.