added more bigint code
This commit is contained in:
parent
f78a95283c
commit
bee0815c1e
@ -49,13 +49,21 @@ package body Bigint is
|
|||||||
return R.High;
|
return R.High;
|
||||||
end Get_High;
|
end Get_High;
|
||||||
|
|
||||||
function Is_Less (X: in Object_Pointer;
|
function Make_Word (L: in Object_Half_Word;
|
||||||
Y: in Object_Pointer) return Standard.Boolean is
|
H: in Object_Half_Word) return Object_Word is
|
||||||
|
W: Object_Word;
|
||||||
|
R: Word_Record;
|
||||||
|
for R'Address use W'Address;
|
||||||
begin
|
begin
|
||||||
if X.Sign /= Y.Sign then
|
R.Low := L;
|
||||||
return X.Sign = Negative_Sign;
|
R.High := H;
|
||||||
end if;
|
return W;
|
||||||
|
end Make_Word;
|
||||||
|
|
||||||
|
function Is_Less_Unsigned (X: in Object_Pointer;
|
||||||
|
Y: in Object_Pointer) return Standard.Boolean is
|
||||||
|
pragma Inline (Is_Less_Unsigned);
|
||||||
|
begin
|
||||||
if X.Size /= Y.Size then
|
if X.Size /= Y.Size then
|
||||||
return X.Size < Y.Size;
|
return X.Size < Y.Size;
|
||||||
end if;
|
end if;
|
||||||
@ -67,6 +75,15 @@ package body Bigint is
|
|||||||
end loop;
|
end loop;
|
||||||
|
|
||||||
return Standard.False;
|
return Standard.False;
|
||||||
|
end Is_Less_Unsigned;
|
||||||
|
|
||||||
|
function Is_Less (X: in Object_Pointer;
|
||||||
|
Y: in Object_Pointer) return Standard.Boolean is
|
||||||
|
begin
|
||||||
|
if X.Sign /= Y.Sign then
|
||||||
|
return X.Sign = Negative_Sign;
|
||||||
|
end if;
|
||||||
|
return Is_Less_Unsigned (X, Y);
|
||||||
end Is_Less;
|
end Is_Less;
|
||||||
|
|
||||||
function Is_Equal (X: in Object_Pointer;
|
function Is_Equal (X: in Object_Pointer;
|
||||||
@ -77,9 +94,48 @@ package body Bigint is
|
|||||||
X.Half_Word_Slot = Y.Half_Word_Slot;
|
X.Half_Word_Slot = Y.Half_Word_Slot;
|
||||||
end Is_Equal;
|
end Is_Equal;
|
||||||
|
|
||||||
function Add (Interp: access Interpreter_Record;
|
function Is_Zero (X: in Object_Pointer) return Standard.Boolean is
|
||||||
X: in Object_Pointer;
|
begin
|
||||||
Y: in Object_Pointer) return Object_Pointer is
|
return X.Size = 1 and then X.Half_Word_Slot(1) = 0;
|
||||||
|
end Is_Zero;
|
||||||
|
|
||||||
|
function Normalize (X: in Object_Pointer) return Object_Pointer is
|
||||||
|
begin
|
||||||
|
case X.Size is
|
||||||
|
when 1 =>
|
||||||
|
if X.Sign = Negative_Sign then
|
||||||
|
return Integer_To_Pointer(-Object_Integer(X.Half_Word_Slot(1)));
|
||||||
|
else
|
||||||
|
return Integer_To_Pointer(Object_Integer(X.Half_Word_Slot(1)));
|
||||||
|
end if;
|
||||||
|
when 2 =>
|
||||||
|
declare
|
||||||
|
W: Object_Word := Make_Word (X.Half_Word_Slot(1), X.Half_Word_Slot(2));
|
||||||
|
begin
|
||||||
|
if X.Sign = Negative_Sign then
|
||||||
|
if W in 0 .. Object_Word(-Object_Signed_Word(Object_Integer'First)) then
|
||||||
|
return Integer_To_Pointer(-Object_Integer(W));
|
||||||
|
else
|
||||||
|
return X;
|
||||||
|
end if;
|
||||||
|
else
|
||||||
|
if W in 0 .. Object_Word(Object_Integer'Last) then
|
||||||
|
return Integer_To_Pointer(Object_Integer(W));
|
||||||
|
else
|
||||||
|
return X;
|
||||||
|
end if;
|
||||||
|
end if;
|
||||||
|
end;
|
||||||
|
|
||||||
|
when others =>
|
||||||
|
return X;
|
||||||
|
end case;
|
||||||
|
|
||||||
|
end Normalize;
|
||||||
|
|
||||||
|
function Add_Unsigned (Interp: access Interpreter_Record;
|
||||||
|
X: in Object_Pointer;
|
||||||
|
Y: in Object_Pointer) return Object_Pointer is
|
||||||
--pragma Assert (Is_Integer(X) or else Is_Bigint(X));
|
--pragma Assert (Is_Integer(X) or else Is_Bigint(X));
|
||||||
--pragma Assert (Is_Integer(Y) or else Is_Bigint(Y));
|
--pragma Assert (Is_Integer(Y) or else Is_Bigint(Y));
|
||||||
pragma Assert (Is_Bigint(X));
|
pragma Assert (Is_Bigint(X));
|
||||||
@ -94,7 +150,7 @@ package body Bigint is
|
|||||||
Push_Top (Interp.all, B'Unchecked_Access);
|
Push_Top (Interp.all, B'Unchecked_Access);
|
||||||
Push_Top (Interp.all, Z'Unchecked_Access);
|
Push_Top (Interp.all, Z'Unchecked_Access);
|
||||||
|
|
||||||
if X.Size > Y.Size then
|
if X.Size >= Y.Size then
|
||||||
A := X;
|
A := X;
|
||||||
B := Y;
|
B := Y;
|
||||||
Last := X.Size + 1;
|
Last := X.Size + 1;
|
||||||
@ -141,11 +197,11 @@ end loop;
|
|||||||
end;
|
end;
|
||||||
Pop_Tops (Interp.all, 3);
|
Pop_Tops (Interp.all, 3);
|
||||||
return Z;
|
return Z;
|
||||||
end Add;
|
end Add_Unsigned;
|
||||||
|
|
||||||
function Subtract (Interp: access Interpreter_Record;
|
function Subtract_Unsigned (Interp: access Interpreter_Record;
|
||||||
X: in Object_Pointer;
|
X: in Object_Pointer;
|
||||||
Y: in Object_Pointer) return Object_Pointer is
|
Y: in Object_Pointer) return Object_Pointer is
|
||||||
A, B, Z: aliased Object_Pointer;
|
A, B, Z: aliased Object_Pointer;
|
||||||
Last: Half_Word_Object_Size;
|
Last: Half_Word_Object_Size;
|
||||||
Borrow: Object_Signed_Word;
|
Borrow: Object_Signed_Word;
|
||||||
@ -155,7 +211,7 @@ end;
|
|||||||
Push_Top (Interp.all, B'Unchecked_Access);
|
Push_Top (Interp.all, B'Unchecked_Access);
|
||||||
Push_Top (Interp.all, Z'Unchecked_Access);
|
Push_Top (Interp.all, Z'Unchecked_Access);
|
||||||
|
|
||||||
if X.Size > Y.Size then
|
if X.Size >= Y.Size then
|
||||||
A := X;
|
A := X;
|
||||||
B := Y;
|
B := Y;
|
||||||
Last := X.Size;
|
Last := X.Size;
|
||||||
@ -190,7 +246,7 @@ end;
|
|||||||
end if;
|
end if;
|
||||||
end loop;
|
end loop;
|
||||||
|
|
||||||
ada.text_io.put_line ("-------------");
|
ada.text_io.put_line ("------------SUBTRACT-");
|
||||||
declare
|
declare
|
||||||
package Int_IO is new ada.text_io.modular_IO(object_half_word);
|
package Int_IO is new ada.text_io.modular_IO(object_half_word);
|
||||||
begin
|
begin
|
||||||
@ -200,7 +256,120 @@ ada.text_io.new_line;
|
|||||||
end loop;
|
end loop;
|
||||||
end;
|
end;
|
||||||
return Z;
|
return Z;
|
||||||
end Subtract;
|
end Subtract_Unsigned;
|
||||||
|
|
||||||
|
function Add (Interp: access Interpreter_Record;
|
||||||
|
X: in Object_Pointer;
|
||||||
|
Y: in Object_Pointer) return Object_Pointer is
|
||||||
|
|
||||||
|
Z: Object_Pointer;
|
||||||
|
A: aliased Object_Pointer;
|
||||||
|
B: aliased Object_Pointer;
|
||||||
|
begin
|
||||||
|
if Is_Integer(X) and then Is_Integer(Y) then
|
||||||
|
declare
|
||||||
|
G: Object_Integer := Pointer_To_Integer(X);
|
||||||
|
H: Object_Integer := Pointer_To_Integer(Y);
|
||||||
|
T: Object_Integer;
|
||||||
|
begin
|
||||||
|
T := G + H;
|
||||||
|
return Integer_To_Pointer(T);
|
||||||
|
exception
|
||||||
|
when Constraint_Error =>
|
||||||
|
Push_Top (Interp.all, A'Unchecked_Access);
|
||||||
|
Push_Top (Interp.all, B'Unchecked_Access);
|
||||||
|
A := Make_Bigint(Interp, G);
|
||||||
|
B := Make_Bigint(Interp, H);
|
||||||
|
end;
|
||||||
|
else
|
||||||
|
Push_Top (Interp.all, A'Unchecked_Access);
|
||||||
|
Push_Top (Interp.all, B'Unchecked_Access);
|
||||||
|
A := X;
|
||||||
|
B := Y;
|
||||||
|
if Is_Integer(A) then
|
||||||
|
A := Make_Bigint(Interp, Pointer_To_Integer(A));
|
||||||
|
end if;
|
||||||
|
if Is_Integer(B) then
|
||||||
|
B := Make_Bigint(Interp, Pointer_To_Integer(B));
|
||||||
|
end if;
|
||||||
|
end if;
|
||||||
|
|
||||||
|
if A.Sign /= B.Sign then
|
||||||
|
if A.Sign = Negative_Sign then
|
||||||
|
Z := Subtract (Interp, B, A);
|
||||||
|
else
|
||||||
|
Z := Subtract (Interp, A, B);
|
||||||
|
end if;
|
||||||
|
else
|
||||||
|
Z := Add_Unsigned (Interp, A, B);
|
||||||
|
Z.Sign := A.Sign;
|
||||||
|
end if;
|
||||||
|
|
||||||
|
Pop_Tops (Interp.all, 2);
|
||||||
|
return Normalize(Z);
|
||||||
|
end Add;
|
||||||
|
|
||||||
|
function Subtract (Interp: access Interpreter_Record;
|
||||||
|
X: in Object_Pointer;
|
||||||
|
Y: in Object_Pointer) return Object_Pointer is
|
||||||
|
Z: Object_Pointer;
|
||||||
|
A: aliased Object_Pointer;
|
||||||
|
B: aliased Object_Pointer;
|
||||||
|
begin
|
||||||
|
if Is_Integer(X) and then Is_Integer(Y) then
|
||||||
|
declare
|
||||||
|
G: Object_Integer := Pointer_To_Integer(X);
|
||||||
|
H: Object_Integer := Pointer_To_Integer(Y);
|
||||||
|
T: Object_Integer;
|
||||||
|
begin
|
||||||
|
T := G + H;
|
||||||
|
return Integer_To_Pointer(T);
|
||||||
|
exception
|
||||||
|
when Constraint_Error =>
|
||||||
|
Push_Top (Interp.all, A'Unchecked_Access);
|
||||||
|
Push_Top (Interp.all, B'Unchecked_Access);
|
||||||
|
A := Make_Bigint(Interp, G);
|
||||||
|
B := Make_Bigint(Interp, H);
|
||||||
|
end;
|
||||||
|
else
|
||||||
|
Push_Top (Interp.all, A'Unchecked_Access);
|
||||||
|
Push_Top (Interp.all, B'Unchecked_Access);
|
||||||
|
A := X;
|
||||||
|
B := Y;
|
||||||
|
if Is_Integer(A) then
|
||||||
|
A := Make_Bigint(Interp, Pointer_To_Integer(A));
|
||||||
|
end if;
|
||||||
|
if Is_Integer(B) then
|
||||||
|
B := Make_Bigint(Interp, Pointer_To_Integer(B));
|
||||||
|
end if;
|
||||||
|
end if;
|
||||||
|
|
||||||
|
if A.Sign /= B.Sign then
|
||||||
|
Z := Add_Unsigned (Interp, A, B);
|
||||||
|
Z.Sign := A.Sign;
|
||||||
|
--if A.Sign = Negative_Sign then
|
||||||
|
-- Z.Sign := Negative_Sign;
|
||||||
|
-- Z.Sign := Negative_Sign;
|
||||||
|
--else
|
||||||
|
-- Z := Add_Unsigned (Interp, A, B);
|
||||||
|
--end if;
|
||||||
|
else
|
||||||
|
if Is_Less_Unsigned(A, B) then
|
||||||
|
Z := Subtract_Unsigned (Interp, B, A);
|
||||||
|
--Z.Sign := Object_Sign'Val(not Object_Sign'Pos(A.Sign)); -- opposite A.Sign
|
||||||
|
if A.Sign = Negative_Sign then
|
||||||
|
Z.Sign := Positive_Sign;
|
||||||
|
else
|
||||||
|
Z.Sign := Negative_Sign;
|
||||||
|
end if;
|
||||||
|
else
|
||||||
|
Z := Subtract_Unsigned (Interp, A, B);
|
||||||
|
Z.Sign := A.Sign;
|
||||||
|
end if;
|
||||||
|
end if;
|
||||||
|
|
||||||
|
Pop_Tops (Interp.all, 2);
|
||||||
|
return Normalize(Z);
|
||||||
|
end Subtract;
|
||||||
end Bigint;
|
end Bigint;
|
||||||
|
|
||||||
|
@ -2659,8 +2659,10 @@ Push_Top (Interp, B'Unchecked_Access);
|
|||||||
--A := Bigint.Add (Interp.Self, A, B);
|
--A := Bigint.Add (Interp.Self, A, B);
|
||||||
--end loop;
|
--end loop;
|
||||||
A := Make_Bigint (Interp.Self, Value => 16#FFFF_00000001#);
|
A := Make_Bigint (Interp.Self, Value => 16#FFFF_00000001#);
|
||||||
B := Make_Bigint (Interp.Self, Value => 16#0000_0000000F#);
|
B := Make_Bigint (Interp.Self, Value => 16#FFFF_0000000F#);
|
||||||
A := Bigint.Subtract (Interp.Self, A, B);
|
--A := Bigint.Subtract (Interp.Self, integer_to_pointer(16), B);
|
||||||
|
A := Bigint.Subtract (Interp.Self, B, integer_to_pointer(16));
|
||||||
|
print (interp, A);
|
||||||
Pop_tops (Interp, 2);
|
Pop_tops (Interp, 2);
|
||||||
end;
|
end;
|
||||||
Ada.Text_IO.Put_LINE ("=== BYE ===");
|
Ada.Text_IO.Put_LINE ("=== BYE ===");
|
||||||
|
@ -467,6 +467,9 @@ package H2.Scheme is
|
|||||||
function Make_Bigint (Interp: access Interpreter_Record;
|
function Make_Bigint (Interp: access Interpreter_Record;
|
||||||
Size: Half_Word_Object_Size) return Object_Pointer;
|
Size: Half_Word_Object_Size) return Object_Pointer;
|
||||||
|
|
||||||
|
function Make_Bigint (Interp: access Interpreter_Record;
|
||||||
|
Value: Object_Integer) return Object_Pointer;
|
||||||
|
|
||||||
-- -----------------------------------------------------------------------------
|
-- -----------------------------------------------------------------------------
|
||||||
|
|
||||||
|
|
||||||
@ -579,9 +582,12 @@ private
|
|||||||
|
|
||||||
function Get_Low (W: Object_Word) return Object_Half_Word;
|
function Get_Low (W: Object_Word) return Object_Half_Word;
|
||||||
function Get_High (W: Object_Word) return Object_Half_Word;
|
function Get_High (W: Object_Word) return Object_Half_Word;
|
||||||
|
function Make_Word (L: Object_Half_Word;
|
||||||
|
H: Object_Half_Word) return Object_Word;
|
||||||
|
|
||||||
pragma Inline (Get_High);
|
pragma Inline (Get_High);
|
||||||
pragma Inline (Get_Low);
|
pragma Inline (Get_Low);
|
||||||
|
pragma Inline (Make_Word);
|
||||||
|
|
||||||
function Add (Interp: access Interpreter_Record;
|
function Add (Interp: access Interpreter_Record;
|
||||||
X: in Object_Pointer;
|
X: in Object_Pointer;
|
||||||
|
Loading…
Reference in New Issue
Block a user