diff options
author | dos-reis <gdr@axiomatics.org> | 2011-09-13 12:50:14 +0000 |
---|---|---|
committer | dos-reis <gdr@axiomatics.org> | 2011-09-13 12:50:14 +0000 |
commit | 5c07eb9f4fea3835b7f2d272aa4790b0638a512c (patch) | |
tree | 4aba0944eaa0e549284af0143c4388c4a780a2cb /src/utils | |
parent | 10244e3b6dcf312f45105ac531a24bcdc1af9302 (diff) | |
download | open-axiom-5c07eb9f4fea3835b7f2d272aa4790b0638a512c.tar.gz |
* utils/Makefile.in (libOpenAxiom_HEADERS): Include vm.H.
(libOpenAxiom_SOURCES): Include vm.cc
* utils/vm.cc: New.
* utils/vm.H: Likwise.
Diffstat (limited to 'src/utils')
-rw-r--r-- | src/utils/Makefile.in | 6 | ||||
-rw-r--r-- | src/utils/vm.H | 269 | ||||
-rw-r--r-- | src/utils/vm.cc | 46 |
3 files changed, 318 insertions, 3 deletions
diff --git a/src/utils/Makefile.in b/src/utils/Makefile.in index dd2d3154..e9734fd2 100644 --- a/src/utils/Makefile.in +++ b/src/utils/Makefile.in @@ -36,14 +36,14 @@ hammer_SOURCES = hammer.cc hammer_OBJECTS = $(hammer_SOURCES:.cc=.lo) hammer_LDADD = -L. -lOpenAxiom -libOpenAxiom_HEADERS = storage.H hash-table.H string-pool.H sexpr.H +libOpenAxiom_HEADERS = storage.H hash-table.H string-pool.H sexpr.H vm.H libOpenAxiom_SOURCES = \ storage.cc string-pool.cc sexpr.cc command.cc \ - filesystem.cc + filesystem.cc vm.cc libOpenAxiom_OBJECTS = $(libOpenAxiom_SOURCES:.cc=.lo) -oa_public_headers = storage hash-table string-pool sexpr +oa_public_headers = storage hash-table string-pool sexpr vm ## Where we store public header files oa_target_headerdir = $(oa_target_includedir)/open-axiom diff --git a/src/utils/vm.H b/src/utils/vm.H new file mode 100644 index 00000000..981c6bad --- /dev/null +++ b/src/utils/vm.H @@ -0,0 +1,269 @@ +// Copyright (C) 2011, Gabriel Dos Reis. +// All rights reserved. +// +// Redistribution and use in source and binary forms, with or without +// modification, are permitted provided that the following conditions are +// met: +// +// - Redistributions of source code must retain the above copyright +// notice, this list of conditions and the following disclaimer. +// +// - Redistributions in binary form must reproduce the above copyright +// notice, this list of conditions and the following disclaimer in +// the documentation and/or other materials provided with the +// distribution. +// +// - Neither the name of OpenAxiom nor the names of its contributors +// may be used to endorse or promote products derived from this +// software without specific prior written permission. +// +// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS +// IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED +// TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A +// PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER +// OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, +// EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, +// PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR +// PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF +// LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING +// NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS +// SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + +// --% Author: Gabriel Dos Reis +// --% Description: +// --% Interface and implementation of basic services of the +// --% OpenAxiom Virtual Machine. + +#ifndef OPENAXIOM_VM_INCLUDED +#define OPENAXIOM_VM_INCLUDED + +#include <open-axiom/storage> +#if HAVE_STDINT_H +# include <stdint.h> +#endif +#include <open-axiom/string-pool> + +namespace OpenAxiom { + namespace VM { + // --% + // --% Value representation + // --% + // A far reaching design decision is to provide a uniform + // representation for values. That is all values, irrespective + // of type have fit in a fixed format, i.e. a scalar register. + // This means that values that are more complicated than a scalar, + // that is the vast majority and most interesting values, have to + // be stored in allocated objects and addresses of their container + // objects used in place of the actual values. This is folklore + // in the communities of garbage collected language implementations. + // An unfortunate but widely held belief is that AXIOM-based + // systems (and computer algebra systems in general) are + // Lisp-based systems. Nothing could be further from the truth + // for OpenAxiom. The type system is believed to support + // erasure semantics, at least for values. + // + // However,the current implementation being Lisp-based, it does + // unwittingly make use of some Lisp features that are not + // strictly necessary, but which would take a certain amount + // of effort to get rid of. Consequently, we must cope -- at least + // for now -- with the notion of uniform value representation and + // use runtime predicates to descriminate between values. + // However, we do not want to carry an unduly expensive abstraction + // penalty for perfectly well behaved and well disciplined programs. + // Here are a few constraints: + // 1. Small integers should represent themselves, not allocated. + // Furthermore, the maximum range should be sought. + // 2. Since we have to deal with characters, they should be + // directly represented. + // 3. List values and list manipulation should be efficient. + // Ideally, a pair should not occupy more than what it + // takes to store two values. + // 4. Idealy, pointers to foreign objects (at least) should be + // left unmolested. + // + // * Assumptions: + // (a) the host machine has sizeof(Value) quo 4 = 0. + // (b) allocatd objects can be aligned on sizeof(Value) boundary. + // (c) the host machine has 2's complement arithmetic. + // + // If: + // -- we use a dedicated allocation pool for cons cells + // -- we allocate the first cell in each cons-storage arena + // on a 8-byte boundary + // -- we use exactly 2 * sizeof(Value) to store a cons cell + // therefore realizing constraint (1) + // then: + // each pointer to a cons cell will have its last 3 bits cleared. + // + // Therefore, we can use the last 3 bits to tag a cons value, instead + // of storing the tag inside the cons cell. we can't leave those + // bits cleared for we would not be able to easily and cheaply + // distinguish a pointer to a cons cell from a pointer to other objects. + // + // To meet constraint (1), we must logically use at least one bit + // to distinguish a small integer from a pointer to a cons cell. + // The good news is that we need not need more than that if pointers + // to foreign pointers do not have the last bit set. Which is + // pretty much the case in practice as crystallized in assumption (a). + // Therefore we arrive at the first design: + // I. the value representation of small integer always have the + // the least significant bit set. All other bits are + // significant in representing small integers. + // As a consequence, the last bit of all other values must be cleared. + // + // Next, cons cell values: + // II. Cons cells are represented by their addresses with the + // last 3 bits set to 0b110. + // + // III. Any allocated object obj shall saisfy sizeof obj quo 4 = 0. + // All allocated objects shall have alignment divisible by 4. + // In consequence, the last two bits of object addresses + // are cleared, i.e. 0b00. + // Finally: + // IV. The representation of a character shall have the last two + // bits set to 0b10. + // + // Note: These choices do not satisfy constraint 4. If we could + // restrict pointer to foreign objects to address aligned + // to multiple of 4 boundaries, then we can make different choices + // that represents foreign address directly. + // In the current design, foriegn addresses are allocated. + + + // ----------- + // -- Value -- + // ----------- + // All VM values fit in a universal value datatype. + typedef uintptr_t Value; + + // ------------- + // -- Pointer -- + // ------------- + // Allocated objects are represented by their addresses. + using Memory::Pointer; + + const Value ptr_tag = 0x0; + + inline bool is_pointer(Value v) { + return (v & 0x3) == ptr_tag; + } + + inline Pointer to_pointer(Value v) { + return Pointer(v); + } + + inline Value from_pointer(Pointer p) { + return Value(p); + } + + // ------------- + // -- Fixnum --- + // ------------- + // VM integers are divided into classes: small numbers, + // and large numbers. A small number fits entirely in a register. + // A large number is allocated and represented by its address. + typedef intptr_t Fixnum; + + const Value fix_tag = 0x1; + + inline bool is_fixnum(Value v) { + return (v & 0x1) == fix_tag; + } + + inline Fixnum to_fixnum(Value v) { + return Fixnum(v >> 1); + } + + inline Value from_fixnum(Fixnum i) { + return (Fixnum(i) << 1 ) | fix_tag; + } + + // --------------- + // -- Character -- + // --------------- + + // This datatype is prepared for Uncode characters even if + // we do not handle UCN characters. + typedef Value Character; + + const Value char_tag = 0x2; + + inline bool is_character(Value v) { + return (v & 0x3) == char_tag; + } + + inline Character to_character(Value v) { + return Character(v >> 2); + } + + inline Value from_character(Character c) { + return (Value(c) << 2) | char_tag; + } + + // ---------- + // -- Pair -- + // ---------- + struct ConsCell { + Value head; + Value tail; + ConsCell(Value h, Value t) : head(h), tail(t) { } + }; + + typedef ConsCell* Pair; + + const Value pair_tag = 0x6; + + inline bool is_pair(Value v) { + return (v & 0x7) == pair_tag; + } + + inline Pair to_pair(Value v) { + return Pair(v & ~0x7); + } + + inline Value from_pair(Pair p) { + return Value(p) | pair_tag; + } + + // If `v' designates a pair, return a pointer to its + // concrete representation. + inline Pair pair_if_can(Value v) { + return is_pair(v) ? to_pair(v) : 0; + } + + // -- + // -- Builtin Operations + // -- + // Types for native implementation of builtin operators. + struct BasicContext; + typedef Value (*NullaryCode)(BasicContext*); + typedef Value (*UnaryCode)(BasicContext*, Value); + typedef Value (*BinaryCode)(BasicContext*, Value, Value); + typedef Value (*TernaryCode)(BasicContext*, Value, Value, Value); + + // ------------ + // -- Object -- + // ------------ + template<typename T> + struct Object { + Value dict; + T data; + }; + + // ------------------ + // -- BasicContext -- + // ------------------ + // Provides basic evaluation services. + struct BasicContext : StringPool { + BasicContext(); + + Pair make_cons(Value, Value); + + protected: + Memory::Factory<ConsCell> conses; + }; + }; +} + +#endif // OPENAXIOM_VM_INCLUDED + diff --git a/src/utils/vm.cc b/src/utils/vm.cc new file mode 100644 index 00000000..291eda03 --- /dev/null +++ b/src/utils/vm.cc @@ -0,0 +1,46 @@ +// Copyright (C) 2011, Gabriel Dos Reis. +// All rights reserved. +// +// Redistribution and use in source and binary forms, with or without +// modification, are permitted provided that the following conditions are +// met: +// +// - Redistributions of source code must retain the above copyright +// notice, this list of conditions and the following disclaimer. +// +// - Redistributions in binary form must reproduce the above copyright +// notice, this list of conditions and the following disclaimer in +// the documentation and/or other materials provided with the +// distribution. +// +// - Neither the name of OpenAxiom nor the names of its contributors +// may be used to endorse or promote products derived from this +// software without specific prior written permission. +// +// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS +// IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED +// TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A +// PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER +// OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, +// EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, +// PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR +// PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF +// LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING +// NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS +// SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + +// --% Author: Gabriel Dos Reis + +#include "vm.H" + +namespace OpenAxiom { + namespace VM { + // -- BasicContext -- + Pair BasicContext::make_cons(Value h, Value t) { + return conses.make(h, t); + } + + BasicContext::BasicContext() { + } + } +} |