VirtualBox

source: vbox/trunk/src/VBox/Runtime/common/compiler/vcc/x86-alldvrm.asm

Last change on this file was 98508, checked in by vboxsync, 16 months ago

IPRT/vcc: Optimized unsigned 64-bit division implementation for the 32-bit Visual C++ compilers. bugref:10261

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 5.1 KB
Line 
1; $Id: x86-alldvrm.asm 98508 2023-02-08 15:31:06Z vboxsync $
2;; @file
3; IPRT - Visual C++ Compiler - signed 64-bit division support, x86.
4;
5
6;
7; Copyright (C) 2023 Oracle and/or its affiliates.
8;
9; This file is part of VirtualBox base platform packages, as
10; available from https://www.virtualbox.org.
11;
12; This program is free software; you can redistribute it and/or
13; modify it under the terms of the GNU General Public License
14; as published by the Free Software Foundation, in version 3 of the
15; License.
16;
17; This program is distributed in the hope that it will be useful, but
18; WITHOUT ANY WARRANTY; without even the implied warranty of
19; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
20; General Public License for more details.
21;
22; You should have received a copy of the GNU General Public License
23; along with this program; if not, see <https://www.gnu.org/licenses>.
24;
25; The contents of this file may alternatively be used under the terms
26; of the Common Development and Distribution License Version 1.0
27; (CDDL), a copy of it is provided in the "COPYING.CDDL" file included
28; in the VirtualBox distribution, in which case the provisions of the
29; CDDL are applicable instead of those of the GPL.
30;
31; You may elect to license modified versions of this file under the
32; terms and conditions of either the GPL or the CDDL or both.
33;
34; SPDX-License-Identifier: GPL-3.0-only OR CDDL-1.0
35;
36
37
38;*********************************************************************************************************************************
39;* Header Files *
40;*********************************************************************************************************************************
41%include "iprt/asmdefs.mac"
42
43
44;*********************************************************************************************************************************
45;* External Symbols *
46;*********************************************************************************************************************************
47extern __aulldvrm
48
49
50;;
51; Division of signed 64-bit values, returning both the quotient and remainder.
52;
53; @returns EDX:EAX Quotient, EBX:ECX Remainder.
54; @param [esp+04h] [ebp+08h] Dividend (64-bit)
55; @param [esp+0ch] [ebp+10h] Divisor (64-bit)
56;
57; @note The remainder registers are swapped compared to Watcom's I8D and U8D.
58;
59BEGINPROC_RAW __alldvrm
60 ;
61 ; Load high parts so we can examine them for negativity.
62 ;
63 mov edx, [esp + 08h] ; dividend_hi
64 mov ecx, [esp + 10h] ; divisor_hi
65
66 ;
67 ; We use __aulldvrm to do the work, we take care of the signedness.
68 ;
69 or edx, edx
70 js .negative_dividend
71
72 or ecx, ecx
73 js .negative_divisor_positive_dividend
74
75 ; Both positive, so same as unsigned division.
76 jmp __aulldvrm
77
78
79 ;
80 ; The rest of the code sets up a stack frame using EBP as it makes
81 ; calls rather than tail jumps.
82 ;
83
84.negative_divisor_positive_dividend:
85 push ebp
86 mov ebp, esp
87
88 ; Load the low values to as we will be pushing them and probably negating them.
89 mov eax, dword [ebp + 08h] ; dividend_lo
90 mov ebx, dword [ebp + 10h] ; divisor_lo
91
92 ; negate the divisor, do unsigned division, and negate the quotient.
93 neg ecx
94 neg ebx
95 sbb ecx, 0
96
97 push ecx
98 push ebx
99 push edx
100 push eax
101 call __aulldvrm ; cleans up the the stack.
102
103 neg edx
104 neg eax
105 sbb edx, 0
106
107 leave
108 ret 10h
109
110.negative_dividend:
111 push ebp
112 mov ebp, esp
113
114 ; Load the low values to as we will be pushing them and probably negating them.
115 mov eax, dword [ebp + 08h] ; dividend_lo
116 mov ebx, dword [ebp + 10h] ; divisor_lo
117
118 neg edx
119 neg eax
120 sbb edx, 0
121
122 or ecx, ecx
123 js .negative_dividend_negative_divisor
124
125.negative_dividend_positive_divisor:
126 ; negate the dividend (above), do unsigned division, and negate both quotient and remainder
127
128 push ecx
129 push ebx
130 push edx
131 push eax
132 call __aulldvrm ; cleans up the the stack.
133
134 neg edx
135 neg eax
136 sbb edx, 0
137
138.return_negated_remainder:
139 neg ebx
140 neg ecx
141 sbb ebx, 0
142
143 leave
144 ret 10h
145
146.negative_dividend_negative_divisor:
147 ; negate both dividend (above) and divisor, do unsigned division, and negate the remainder.
148 neg ecx
149 neg ebx
150 sbb ecx, 0
151
152 push ecx
153 push ebx
154 push edx
155 push eax
156 call __aulldvrm ; cleans up the the stack.
157
158 jmp .return_negated_remainder
159ENDPROC_RAW __alldvrm
160
Note: See TracBrowser for help on using the repository browser.

© 2023 Oracle
ContactPrivacy policyTerms of Use