Calculator on Ada
------------------------------------------------------------------------------
-- --
-- A D A C A L C U L A T O R --
-- --
------------------------------------------------------------------------------
-- Copyright (C) 1999 by Eugene Nonko, cm@liceum.secna.ru --
------------------------------------------------------------------------------
-- expr0 ::= Variable Assign expr1 | expr1
-- expr1 ::= expr2 | expr2 '+' expr2 ... | expr2 '-' expr2 ...
-- expr2 ::= expr3 | expr3 '*' expr3 ... | expr3 '/' expr3 ...
-- expr3 ::= expr4 | expr4 Power expr3
-- expr4 ::= Number | '(' expr1 ')' | '-' expr4 | '+' expr4 |
-- Predefined_Function '(' expr1 ')'
with Ada.Text_IO,
Ada.Characters.Latin_1,
Ada.Strings.Unbounded,
Ada.Numerics,
Ada.Numerics.Generic_Elementary_Functions,
Stack,
List;
use Ada.Text_IO,
Ada.Characters.Latin_1,
Ada.Strings.Unbounded,
Ada.Numerics;
procedure Calculator is
type Value_Type is digits 8;
package Value_Type_IO is
new Float_IO (Value_Type);
use Value_Type_IO;
package Value_Type_Functions is
new Generic_Elementary_Functions (Value_Type);
use Value_Type_Functions;
Syntax_Error, No_Expression_Error, Divide_By_Zero : exception;
type Predefined_Function_Type is
access function (X : Value_Type) return Value_Type;
package Variables is
new List (Value_Type, Unbounded_String);
package Predefined_Functions is
new List (Predefined_Function_Type, Unbounded_String);
package Value_Stack is new Stack (Value_Type, 64);
use Value_Stack;
function Calculate return Value_Type is
type Token_Type is
('+', '-', '*', '/', '(', ')', Power, Assign,
Number, Variable, Predefined_Function, End_Of_Line);
package Token_Stack is new Stack (Token_Type, 8);
procedure Parse_Expr0 is
Token : Token_Type;
Value : Value_Type;
Identifier, Last_Identifier : Unbounded_String;
function Get_Token return Token_Type is
EOL : Boolean;
C : Character;
function Is_Space (C : Character) return Boolean is
begin -- Is_Space
return C = HT or else C = Space;
end Is_Space;
function Is_Letter_Or_Digit (C : Character) return Boolean is
begin -- Is_Letter_Or_Digit
case C is
when 'A' .. 'Z' | 'a' .. 'z' | '0' .. '9' | '_' =>
return True;
when others =>
return False;
end case;
end Is_Letter_Or_Digit;
begin -- Get_Token
if not Token_Stack.Empty then
return Token_Stack.Pop;
end if;
loop
Look_Ahead (C, EOL);
if EOL then
return End_Of_Line;
end if;
exit when not Is_Space (C);
Get (C);
end loop;
case C is
when '+' =>
Get (C);
return '+';
when '-' =>
Get (C);
return '-';
when '*' =>
Get (C);
Look_Ahead (C, EOL);
if EOL or else C /= '*' then
return '*';
end if;
Get (C);
return Power;
when '/' =>
Get (C);
return '/';
when '^' =>
Get (C);
return Power;
when '(' =>
Get (C);
return '(';
when ')' =>
Get (C);
return ')';
when ':' =>
Get (C);
Look_Ahead (C, EOL);
if EOL or else C /= '=' then
raise Syntax_Error;
end if;
Get (C);
return Assign;
when 'A' .. 'Z' | 'a' .. 'z' | '_' =>
Identifier := Null_Unbounded_String;
loop
Identifier := Identifier & C;
Get (C);
Look_Ahead (C, EOL);
exit when EOL or else not Is_Letter_Or_Digit (C);
end loop;
if Predefined_Functions.Search (Identifier) then
return Predefined_Function;
end if;
return Variable;
when '0' .. '9' =>
begin
Get (Value);
return Number;
exception
when others =>
raise Syntax_Error;
end;
when others =>
raise Syntax_Error;
end case;
raise Syntax_Error;
end Get_Token;
procedure Parse_Expr1 is
procedure Parse_Expr2 is
procedure Parse_Expr3 is
procedure Parse_Expr4 is
Func : Predefined_Function_Type;
begin -- Parse_Expr4
Token := Get_Token;
case Token is
when Variable =>
Push (Variables.Get (Identifier));
when Predefined_Function =>
Func := Predefined_Functions.Get (Identifier);
if Get_Token /= '(' then
raise Syntax_Error;
end if;
Parse_Expr1;
if Get_Token /= ')' then
raise Syntax_Error;
end if;
Push (Func (Pop));
when Number =>
Push (Value);
when '+' =>
Parse_Expr4;
when '-' =>
Parse_Expr4;
Push (-Pop);
when '(' =>
Parse_Expr1;
if Get_Token /= ')' then
raise Syntax_Error;
end if;
when others =>
Token_Stack.Push (Token);
end case;
end Parse_Expr4;
begin -- Parse_Expr3
Parse_Expr4;
loop
Token := Get_Token;
case Token is
when Power =>
Parse_Expr3;
Push (Pop ** Pop);
when others =>
Token_Stack.Push (Token);
exit;
end case;
end loop;
end Parse_Expr3;
Divisor : Value_Type;
begin -- Parse_Expr2
Parse_Expr3;
loop
Token := Get_Token;
case Token is
when '*' =>
Parse_Expr3;
Push (Pop * Pop);
when '/' =>
Parse_Expr3;
Divisor := Pop;
if Divisor = Value_Type (0) then
raise Divide_By_Zero;
else
Push (Pop / Divisor);
end if;
when others =>
Token_Stack.Push (Token);
exit;
end case;
end loop;
end Parse_Expr2;
begin -- Parse_Expr1
Parse_Expr2;
loop
Token := Get_Token;
case Token is
when '+' =>
Parse_Expr2;
Push (Pop + Pop);
when '-' =>
Parse_Expr2;
Swap;
Push (Pop - Pop);
when others =>
Token_Stack.Push (Token);
exit;
end case;
end loop;
end Parse_Expr1;
begin -- Parse_Expr0
Token := Get_Token;
if Token = Variable then
Token := Get_Token;
if Token = Assign then
Last_Identifier := Identifier;
Parse_Expr1;
Variables.Set (Last_Identifier, Get);
return;
else
Token_Stack.Push (Token);
Token_Stack.Push (Variable);
end if;
else
Token_Stack.Push (Token);
end if;
Parse_Expr1;
end Parse_Expr0;
begin -- Calculate
Value_Stack.Drop_All;
Token_Stack.Drop_All;
Parse_Expr0;
Skip_Line;
case Capacity is
when 0 =>
raise No_Expression_Error;
when 1 =>
return Pop;
when others =>
raise Syntax_Error;
end case;
exception
when No_Expression_Error =>
raise;
when others =>
Skip_Line;
raise;
end Calculate;
function Arctan (X : Value_Type) return Value_Type is
begin -- Arctan
return Value_Type_Functions.Arctan (X);
end Arctan;
function Arccot (X : Value_Type) return Value_Type is
begin -- Arccot
return Value_Type_Functions.Arccot (X);
end Arccot;
function Abs_Function (X : Value_Type) return Value_Type is
begin -- Abs_Function
return abs X;
end Abs_Function;
function Sgn (X : Value_Type) return Value_Type is
begin -- Sgn
if X > 0.0 then
return 1.0;
elsif X < 0.0 then
return -1.0;
else
return 0.0;
end if;
end Sgn;
procedure Put_Adaptive (X : Value_Type) is
S : String (1 .. 80);
First : Positive := S'First;
Last : Positive := S'Last;
begin -- Put_Adaptive
Put (To => S, Item => X, Aft => Value_Type'Digits, Exp => 0);
while S (First) = ' ' loop
First := First + 1;
end loop;
while S (Last) = '0' loop
Last := Last - 1;
end loop;
if S (Last) = '.' then
Last := Last - 1;
end if;
Put (S (First .. Last));
end Put_Adaptive;
begin -- Calculator
Variables.Add (To_Unbounded_String ("pi"), Pi);
Variables.Add (To_Unbounded_String ("e"), e);
Predefined_Functions.Add (To_Unbounded_String ("abs"), Abs_Function'Access);
Predefined_Functions.Add (To_Unbounded_String ("sgn"), Sgn'Access);
Predefined_Functions.Add (To_Unbounded_String ("sin"), Sin'Access);
Predefined_Functions.Add (To_Unbounded_String ("cos"), Cos'Access);
Predefined_Functions.Add (To_Unbounded_String ("tan"), Tan'Access);
Predefined_Functions.Add (To_Unbounded_String ("cot"), Cot'Access);
Predefined_Functions.Add (To_Unbounded_String ("exp"), Exp'Access);
Predefined_Functions.Add (To_Unbounded_String ("log"), Log'Access);
Predefined_Functions.Add (To_Unbounded_String ("sqrt"), Sqrt'Access);
Predefined_Functions.Add (To_Unbounded_String ("arcsin"), Arcsin'Access);
Predefined_Functions.Add (To_Unbounded_String ("arccos"), Arccos'Access);
Predefined_Functions.Add (To_Unbounded_String ("arctan"), Arctan'Access);
Predefined_Functions.Add (To_Unbounded_String ("arccot"), Arccot'Access);
Predefined_Functions.Add (To_Unbounded_String ("sinh"), Sinh'Access);
Predefined_Functions.Add (To_Unbounded_String ("cosh"), Cosh'Access);
Predefined_Functions.Add (To_Unbounded_String ("tanh"), Tanh'Access);
Predefined_Functions.Add (To_Unbounded_String ("coth"), Coth'Access);
Predefined_Functions.Add (To_Unbounded_String ("arcsinh"), Arcsinh'Access);
Predefined_Functions.Add (To_Unbounded_String ("arccosh"), Arccosh'Access);
Predefined_Functions.Add (To_Unbounded_String ("arctanh"), Arctanh'Access);
Predefined_Functions.Add (To_Unbounded_String ("arccoth"), Arccoth'Access);
loop
declare
Result : Value_Type;
begin
Put ("? ");
Result := Calculate;
Put ("= ");
Put_Adaptive (Result);
New_Line;
exception
when Overflow_Error =>
Put_Line ("stack overflow");
when Underflow_Error | Syntax_Error =>
Put_Line ("syntax error");
when Variables.Item_Not_Found_Error =>
Put_Line ("undefined variable");
when Argument_Error | Constraint_Error =>
Put_Line ("invalid argument");
when Divide_By_Zero =>
Put_Line ("divide by zero");
when No_Expression_Error =>
exit;
end;
end loop;
end Calculator;
package body List is
type Node;
type Node_Pointer is access Node;
type Node is
record
Next : Node_Pointer;
Identifier : Identifier_Type;
Value : Item_Type;
end record;
First_Node : Node_Pointer := null;
procedure Add (Identifier : Identifier_Type; Value : Item_Type) is
New_Node : Node_Pointer := new Node'
(Next => First_Node, Identifier => Identifier, Value => Value);
begin -- Add
First_Node := New_Node;
end Add;
function Get (Identifier : Identifier_Type) return Item_Type is
Cur_Node : Node_Pointer := First_Node;
begin -- Get
while Cur_Node /= null loop
if Cur_Node.Identifier = Identifier then
return Cur_Node.Value;
end if;
Cur_Node := Cur_Node.Next;
end loop;
raise Item_Not_Found_Error;
end Get;
procedure Set (Identifier : Identifier_Type; Value : Item_Type) is
Cur_Node : Node_Pointer := First_Node;
begin -- Set
while Cur_Node /= null loop
if Cur_Node.Identifier = Identifier then
Cur_Node.Value := Value;
return;
end if;
Cur_Node := Cur_Node.Next;
end loop;
Add (Identifier, Value);
end Set;
function Search (Identifier : Identifier_Type) return Boolean is
Cur_Node : Node_Pointer := First_Node;
begin -- Search
while Cur_Node /= null loop
if Cur_Node.Identifier = Identifier then
return True;
end if;
Cur_Node := Cur_Node.Next;
end loop;
return False;
end Search;
end List;
-- Generic list package, used to create variable and function lists
generic
type Item_Type is private;
type Identifier_Type is private;
package List is
-- Adds item to list without duplication checks
procedure Add (Identifier : Identifier_Type; Value : Item_Type);
-- Gets item value, if item is not present, then raise Item_Not_Found_Error
function Get (Identifier : Identifier_Type) return Item_Type;
-- Sets item's value to Value, adds item if needed
procedure Set (Identifier : Identifier_Type; Value : Item_Type);
-- Returns true if item present in list
function Search (Identifier : Identifier_Type) return Boolean;
Item_Not_Found_Error : exception;
end List;
package body Stack is
Stack_Depth_Constant : constant Positive := Stack_Depth;
subtype Stack_Pointer_Type is Positive range 1 .. Stack_Depth_Constant;
Stack_Data : array (Stack_Pointer_Type) of Item_Type;
Stack_Pointer : Stack_Pointer_Type := 1;
procedure Push (Item : Item_Type) is
begin -- Push
if Stack_Pointer = Stack_Depth_Constant then
raise Overflow_Error;
else
Stack_Data (Stack_Pointer) := Item;
Stack_Pointer := Stack_Pointer + 1;
end if;
end Push;
function Pop return Item_Type is
begin -- Pop
if Stack_Pointer = 1 then
raise Underflow_Error;
else
Stack_Pointer := Stack_Pointer - 1;
return Stack_Data (Stack_Pointer);
end if;
end Pop;
function Get return Item_Type is
begin -- Get
if Stack_Pointer = 1 then
raise Underflow_Error;
else
return Stack_Data (Stack_Pointer - 1);
end if;
end Get;
procedure Empty is
begin -- Empty
Stack_Pointer := 1;
end Empty;
function Is_Empty return Boolean is
begin -- Is_Empty
return Stack_Pointer = 1;
end Is_Empty;
procedure Swap is
Temp : Item_Type := Stack_Data (Stack_Pointer - 1);
begin -- Swap
if Stack_Pointer < 3 then
raise Underflow_Error;
else
Stack_Data (Stack_Pointer - 1) := Stack_Data (Stack_Pointer - 2);
Stack_Data (Stack_Pointer - 2) := Temp;
end if;
end Swap;
procedure Drop is
begin -- Drop
if Stack_Pointer = 1 then
raise Underflow_Error;
else
Stack_Pointer := Stack_Pointer - 1;
end if;
end Drop;
function Capacity return Natural is
begin -- Capacity
return Natural (Stack_Pointer) - 1;
end Capacity;
end Stack;
-- Generic stack package, used to create value and token stacks
generic
type Item_Type is private;
Stack_Depth : Positive;
package Stack is
-- Pushes item
procedure Push (Item : Item_Type);
-- Pops item
function Pop return Item_Type;
-- Gets topmost item without popping
function Get return Item_Type;
-- Swaps topmost items
procedure Swap;
-- Drops topmost item
procedure Drop;
-- Empties stack (drop all items)
procedure Empty;
-- Returns True if stack is empty and False otherwise
function Is_Empty return Boolean;
-- Return current capacity of stack
function Capacity return Natural;
Overflow_Error, Underflow_Error : exception;
end Stack;
Contributed by: Eugene Nonko
Contributed on: May 31, 1999
License: Public Domain
Back