| //===------------ udivmodhi4.S - uint16 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 {uint16, uint16} __udivmodhi4(uint16, uint16)`. |
| // The uint16 quotient is returned via R23:R22, and the uint16 remainder is |
| // returned via R25:R24, while R21/R26/R27 are clobbered. |
| // |
| //===----------------------------------------------------------------------===// |
| |
| .text |
| .align 2 |
| |
| .globl __udivmodhi4 |
| .type __udivmodhi4, @function |
| |
| __udivmodhi4: |
| sub r26, r26 |
| sub r27, r27 ; Initialize the remainder to zero. |
| ldi r21, 17 ; Only loop 16 rounds for uint16. |
| |
| __udivmodhi4_loop: |
| adc r24, r24 |
| adc r25, r25 |
| dec r21 |
| breq __udivmodhi4_end |
| adc r26, r26 |
| adc r27, r27 |
| cp r26, r22 |
| cpc r27, r23 ; Compare with the divisor. |
| brcs __udivmodhi4_loop |
| sub r26, r22 |
| sbc r27, r23 ; Subtract the divisor. |
| rjmp __udivmodhi4_loop |
| |
| __udivmodhi4_end: |
| com r24 |
| com r25 |
| mov r22, r24 |
| mov r23, r25 ; The quotient is returned in R23:R22. |
| mov r24, r26 |
| mov r25, r27 ; The remainder is returned in in R25:R24. |
| ret |