[SparForte][Banner]
[Top Main Menu] Intro | Tutorials | Reference | Packages | Examples | Contributors   [Back Page]      [Next Page]  

Fundamental Types

All AdaScript variables have a type.  Variables must declared so that SparForte knows what that type is.  When new variables are created by assignment at the command prompt, SparForte choses an appropriate type for the new variable.  In scripts, all variables must be declared.


[SparForte Type Tree]
 
Figure: SparForte Predefined Scalar Data Types and their Relationships


Universal Types

There are three "universal types":

Type Name Content Usage
universal_string unlimited length, 8-bit characters text
universal_numeric double precision floating point values numbers
universal_typeless same as universal_string/numeric text or numbers

The first two types form the basis for all other AdaScript string and numeric types.  Variables declared as universal_typeless change their type between universal_string and universal_numeric depending how they are used.  If AdaScript cannot decide in a particular context, the variable type defaults to universal_string.  Typeless variables correspond to variables in Bourne shells and are provided for as a command line convenience.

Universal types automatically match any type derived from the same universal type. A universal_numeric variable accepts integer or floating point values.  A universal_string variable accepts single characters or strings containing many characters.

Universal types are used for all AdaScript literals.  For example, a string literal like "hello" is a universal_string and be assigned to any string type. The numeric literal 45.5 is a universal_numeric can be used with float, long_float, or any other numeric type.

Using these three built-in types will give you a minimum amount of type checking, suitable for short scripts or quick command line calculations.  Universal types should not be used in large scripts because they will greatly reduce the number of errors SparForte can detect.

Ada: Ada has no usable universal types but the Ada rationale uses universal types to describe the type system.

Predefined, Non-Universal Types

For more extensive scripts, AdaScript extends the universal string and numeric types into all the predefined Ada language types, plus some AdaScript-specific types:

Type Name Base Type Usage GCC Ada Equivalent
integer universal_numeric number without a radix point integer
natural universal_numeric 0 or integer larger than zero natural
positive universal_numeric integer larger than zero positive
short_short_integer universal_numeric very small integer short_short_integer
short_integer universal_numeric small integer long_integer
long_integer universal_numeric large integer long_long_integer
long_long_integer universal_numeric very large integer float
float universal_numeric floating point number float
short_float universal_numeric small floating point number short_float
long_float universal_numeric large floating point number long_float
character universal_string 8-bit character character
string universal_string unlimited length string almost unbounded_string
unbounded_string universal_string unlimited length string almost unbounded_string
duration universal_numeric time, float seconds duration
boolean enumerated type true or false boolean
file_mode enumerated type in_file, out_file or append_file file_mode
command limited type alias for an operating system command  -
file_type limited record type operating system files file_type
socket_type limited record type operating system sockets -
complex record type complex numbers complex

The built-in packages may define additional types.

By default, all numeric variables are initialized without any value. Any attempt to use uninitialized numeric variables in an expression will cause an error or exception.

=> i : integer -- no value specified
=> i := i + 1
i := i + 1;
          ^ exception raised

Most types are incompatible with each other. Types may be typecast into related types.

=> i : integer := 5
=> f : float := 0
=> f := i
f := i;
      ^ type 'float' is not compatible with type 'integer'
=> f := float(i) -- typecast
=> ? f
5

You cannot typecast a numeric type into a string type directly.  There are functions in the numerics and strings packages to do these conversions.

String Gotchas

Since all literals have a universal type, this can cause some unusual side-effects. A character variable can contain more than one character if you really want it to by assigning a string literal. Characters are stored as internally as a string and a string literal is a universal_string type.  AdaScript will allow the assignment.  However, the type checking will prevent a character variable from being assigned to a string variable.

=> i : integer := 5
=> c : character;
=> c := "hello"; -- confusing, perhaps stupid, but legal
=> s : string := c; -- ILLEGAL
                  ^ type string is not compatible with type character
=> c := character( "hello" ); -- explicity typecast is not fooled
c := character( "hello" );
                         ^ character value must be one character long

In this example, "hello" is a literal so it is a universal_string. The variable c is a character, a type derived from universal_string. So SparForte allows the assignment. Internally, SparForte uses strings for characters so the entire value of "hello" is assigned to c and c is now a character variable with 5 characters in it. However, SparForte's type rules prevent c from being assigned to a string, and typecasting will check the length of a character variable. (This will likely get addressed in the future.)

AdaScript strings are an unbounded string type--that is, they are stored as an Ada.Strings.Unbounded.Unbounded_String variable.  They have an unlimited length and are not implemented as any sort of array. Ada "fixed" strings, the default string type for GCC Ada, are not used because they have limited length and can create array bounds exceptions. Strings are an integral part of scripts and unbounded strings are used instead. AdaScript includes an unbounded_string type (it has the same properties as the string type) which may make porting to GCC Ada easier.

Constants

Constants can be declared with the word "constant" for any type. The use of "constant" doesn't not affect the type of the variable--it simply prevents new values from being assigned by making the variable "read-only".

program_name : constant string := "Nightly FTP Transfer";

Limited Types

file_type and socket_type variables are known as limited type variables. Limited types cannot be assigned a new value with an assignment statement

=> f : file_type
=> g : file_type
=> f := g
f := g;
      ^ limited variables cannot be assigned a value

SparForte manages the contents of these variables and scripts are not allowed to change the contents.

Complex Types

Variables declared complex represent complex numbers.  A complex number is a record with long_float real and imaginary parts.  The real field is designated "re" and the imaginary field is designated "im".  (See below for more information about using records.)

=> c : complex := ( 1.0, 2.0)
=> ? c.re
 1.0
=> ? c.im
 2.0
=> c.re := 5.0
=> ? c.re
 5.0
=> numerics.set_re( c, 3.0 )
=> ? c.re
 3.0

Some of the numerics package subprograms work with complex types.

Complex types are not fully implemented in this version of SparForte: some of the Ada complex number functions are not available.

Command Types

External operating system commands can be declared using command variables. When a command is declared, SparForte will ensure that the command exists and is runnable.

Command types are similar to limited types and have several restrictions.

  • they must be declared as constants
  • they cannot be assigned to other variables
  • they cannot be used in expressions (although they can appear by themselves and are treated as strings)

=> l : constant command := "/bin/ls"
=> m : constant command := "/bin/lt"
m : constant command := "/bin/lt";
                                 ^ "/bin/lt" is not an executable command
=> n : constant command := l;
n : constant command := l;
                         ^ type universal_string is inherently different from a command
=> ? l & " is the path"
? l & " is the path";
                    ^ type command is inherently different from a universal_string
=> ? l
/bin/ls

 
[Right Submenu]

 AdaScript versus GCC

 Case Sensitivity

 Reserved Words

 Comments

 Literals

 Bourne Shell Word Expansions

 Fundamental Types

 User-defined Types

 Enumerated Types

 Arrays

 Records

 Basic Assignment

 The @ and % Operands

 Command Argument Shortcuts

 Redirection and Pipelines

 Command Line Interaction

 Built-in Shell Commands

 The Current Directory

 Database Commands

 Flow of Control

 Other Statements/ Subprograms

 External Commands

 Block Statements and Subprograms

 TCP/IP Sockets

 Numeric Formatting with Put

 Interpreter Directives

 Command Line Options

 Command Reference

 ASCII and Latin_1 Character Sets

 Common Error Messages

 Common PHP Functions and the SparForte Equivalent

[Back to Top] Back To Top [Small Forte Symbol]