v3.14 -------- Made the following enhancements to improve R7RS support: - Updated the parser for strings, symbols, and character literals. The reader now understands Unicode hex values, strings and symbols now allow mnemonic and numeric escape sequences, and the list of named characters has been extended. - Added support for the `=>` syntax to the `case` conditional. - Added `cond-expand` syntax to statically expand different expressions depending upon whether features are present in the Scheme implementation. This could help allow the same code to work in husk as well as other R7RS implementations. - Added an optional second argument to `log` to allow specifying the base. - Added the following functions: `nan?`, `finite?`, `infinite?`, `exact-integer?`, `exact?`, `inexact?`, `square`, `boolean=?`, `symbol=?`, `read-line`, `flush-output-port`, `eof-object`. - Added support for `include`, `letrec*`, `syntax-error`, `unless`, and `when` syntax. Added a library to compute simple random numbers, based on [this Stack Overflow answer](http://stackoverflow.com/a/14675103/101258): (import (husk random)) (random num) ; Seed the RNG (random) ; Random number from 0 to 1 (randint lo hi) ; Generate random integer between lo (optional) and hi v3.13 -------- - Added the command line flag `--revision 7` (or `-r7` for short) to allow huski and huskc to start in R7RS mode. - Added most of the standard R7RS libraries: `(scheme base)`, `(scheme char)`, etc. - Extended syntax-rules to allow another identifier to be used to specify the ellipsis symbol, per R7RS. For example, `:::` could be used instead: (define-syntax and (syntax-rules ::: () ((and test1 test2 :::) (if test1 (and test2 :::) #f)))) - Added the following functions from R7RS: `make-list` , `list-copy` , `list-set!` , `vector-copy` , `vector-map` , `vector-for-each` , `vector-append` , `string-map` , `string-for-each` , `string->vector` , `vector->string` , `vector-copy!` , `string-copy!` v3.12 -------- Significant enhancements have been made to the huski REPL: - Allow using huski to run Scheme scripts from the shell, as specified by SRFI 22. The script needs to start with the line `#! /usr/bin/env huski` or equivalent, and a `main` function may be defined to receive command line arguments. The `examples/scripts` directory contains example programs `cat.scm` and `sum.scm` that demonstrate how this works in practice. - Add tab completion for Scheme variables and special forms. Tab completion will still fill in filenames when tab is pressed within double-quotes. This makes it easy to find a file in certain cases such as for a `load`. - Accept (and ignore) inputs of just whitespace. Previously this would display a nasty error message. This release also includes the following features: - Added the `(scheme time)` library from R7RS. - Added the `system` function to make system calls from a husk program. The syntax is `(system "command")`. An integer status code is returned with the same value that the executing program returned to the OS. Bug fixes: - Duplicate or otherwise invalid lambda parameters are not allowed, and will throw an error. For example: `(lambda (a a 1) a)` - It is now a parse error to have a form that includes an empty car cell, for example: `'( . 1)` - Ensure all of husk's exports are included in the Haskell API documentation. v3.11 -------- The major change in this release is support for R7RS style library syntax in the compiler. This enables functionality that was previously only available in the interpreter, and sets the stage for husk to begin adding support for R7RS. API changes: - The Compiler Haskell API has been reorganized to use a Types module, and a new Libraries module has been added to store code for libraries. Bug fixes: - Allow the husk compiler to reference variables that are defined later in the program. For example, the following now compiles: `(define (foo) (bar))` `(define (bar) (foo))` - Fixed `string-set!` to allow an expression to be passed as the character argument. - Fixed `string-ref` to work when the string argument is passed using a variable. v3.10 -------- This release includes many important fixes and enhancements to the compiler: - Significantly reduced the amount of compiled Haskell code generated by huskc, leading to a 40% reduction in the code generated by the compiler test suite. Note any actual percentages will depend upon the scheme code being compiled. - Added `load` as a compiler special form, to allow code to be loaded from other files and compiled alongside the main program. - Added optimizations to evaluate primitive expressions at compile time, and to generate more efficient code for functions that are only passed literal arguments. - Enhanced the compiler to detect undefined variables and report an error at compile time, instead of generating code that will throw a runtime error. - Fixed an issue where the compiler would not cache macro expansions in its local memory when compiling function definitions. This bug caused problems for explicit renaming macros, as it was possible for the evaluator to evaluate a special form thinking that it was a function, resulting in an "unbound variable" error. - Modified the test suite to work with both the interpreter and compiler. Note that some test cases are not executed by the compiler because they are not supported (see below). - SRFI 1 is not supported by the compiler at this time. v3.9 -------- - Fixed a syntax error that prevented the FFI module from building. Thanks to Ricardo Lanziano for bringing this to my attention! - Enhanced the compiler to use the name of the source file for the compiled Haskell output instead of hardcoding the file to `_tmp.hs`. For example, `my-file.scm` will compile to `my-file.hs`. - Removed extraneous quotes when printing a lambda form. v3.8 -------- This release introduces several performance improvements: - Macro expansions are now cached, significantly improving performance when repeatedly calling a function containing macros. - A `Pointer` type was added in version 3.6 as part of the changes to enhance the variable storage model. Unfortunately the initial implementation naively checked for pointers prior to calling into any Haskell function. This release eliminates those inefficiencies by allowing primitive functions to deal with the Pointer type directly, instead of attempting to convert values before passing them to primitive functions. - Restructured code in the Macro module to eliminate redundant calls to `Data.Map.lookup`. The example game of life program `examples/game-of-life/life.scm` demonstrates these performance improvements, as it now runs over 4.5 times faster than in the previous release. This release also adds the following library from R7RS: - `(scheme r5rs)` - Exposes the full husk R5RS environment And, R5RS versions of the scheme libraries have been relocated to underneath `(scheme r5rs)`. Each of these libraries exposes a husk subset of the functions recommended by R7RS: - `(scheme r5rs base)` - `(scheme r5rs char)` - `(scheme r5rs complex)` - `(scheme r5rs cxr)` - `(scheme r5rs eval)` - `(scheme r5rs file)` - `(scheme r5rs inexact)` - `(scheme r5rs lazy)` - `(scheme r5rs load)` - `(scheme r5rs read)` - `(scheme r5rs write)` Changes to the Haskell API: - Introduced a new type of function, `CustFunc`, which is now the recommended way to define your own Haskell functions when using the Haskell API. This type allows you to avoid having to handle Pointer types directly in your Haskell code. If you know what you are doing, though, you can handle Pointer types and avoid the overhead of checking for pointers prior to calling into your function code. - Moved `runIOThrows` into Core, and removed obsolete functions `trapError` and `extractValue`. Bug fixes: - Updated `map` and `for-each` to accept multiple list arguments, per R5RS. v3.7 -------- A major change for this release is the introduction of Scheme libraries using R7RS library syntax. For an example of how to use libraries, see `examples/hello-library/hello.scm` in the husk source tree. Note that since R7RS is not currently implemented by husk, the library system only has the built-in import `(r5rs base)` to allow you to import the standard husk R5RS environment. Also, please keep in mind this is still a beta feature that is not yet implemented by the compiler. This release also contains many improvements to the Haskell API: - Added `r5rsEnv` to the Core module to expose the full environment, including functions loaded from the Scheme standard library. - Added `getDataFileFullPath` to the Core module to allow third party code to reference Scheme files such as `stdlib.scm` that are installed alongside husk. - Modified `NumArgs` to optionally require an explicit number of arguments. This helps when writing variable-length functions where specifying a single number of arguments may be misleading. - Added a new module `Language.Scheme.Util` to contain general purpose utility functions. Bug fixes: - Updated the parser to accept floating point numbers that contain only a fractional component, such as `.5`. - Enhanced numerical comparison operators (`=`, `<`, `<=`, `>`, `>=`) to be able to accept an unlimited number of arguments, per R5RS. v3.6.3 -------- Added support for R7RS bytevectors. Improved support for using husk as an extension language by adding Haskell function `evalLisp'`. This function evaluates a lisp data structure and returns the `LispVal` or `LispError` result directly: evalLisp' :: Env -> LispVal -> IO (ThrowsError LispVal) This makes it much easier to retrieve results when using husk as an extension language: result <- evalLisp' env $ List [Atom "/", Number 1, Number 0] case result of Left err -> putStrLn $ "Error: " ++ (show err) Right val -> putStrLn $ show val Finally, fixed a bug where setting a variable to refer back to itself would result in an infinite loop. For example, the last line of the following code would cause `huski` to hang: (define a '()) (define b a) (define a b) v3.6.2 -------- This release adds support for nested quasi-quotation forms, which now respect depth level. This was done by replacing the quasi-quotation special form with a macro based on the one from chibi scheme. A nice side-benefit is that by removing the special forms, quasi-quotation now works in the compiler. Also added support for SRFI 2, `and-let*`. From the SRFI document: > Like an ordinary AND, an AND-LET\* special form evaluates its arguments - expressions - one after another in order, until the first one that yields #f. Unlike AND, however, a non-#f result of one expression can be bound to a fresh variable and used in the subsequent expressions. AND-LET\* is a cross-breed between LET\* and AND. And added support for environment specifiers, including the following functions: - `(interaction-environment)` - `(current-environment)` - `(make-environment)` - `(null-environment version)` - `(load filename environment-specifier)` - `(eval expression environment-specifier)` This release also includes the following bug fixes: - Fixed a bug where nested explicit renaming macros may not always expand properly. - Unfortunately, the storage model changes introduced in 3.6 cause problems with hash table literals defined using the `#hash()` syntax. For now, hash table literals have been removed to prevent further problems. This feature may be added back in the future. - Fixed the code for `require-extension` to allow passing multiple SRFI numbers in the same call. For example: `(require-extension (srfi 1 2))`. - Improved compiler support by diverting renamed variables back into the enclosing environment. Such variables would previously throw a runtime error when accessed by the compiled program. - Improved compiler support by loading macros defined using `define-syntax` so they are available at runtime. v3.6.1 -------- Added support for GHC 7.6. v3.6 -------- Enhanced the variable storage model to correctly store references to objects. For example, consider the following: (define x (list 'a 'b 'c)) (define y x) (set-cdr! x 4) After executing this code, previous versions of husk assigned `(a b c)` to `y`. With this release, husk now evaluates `y` to the expected value of `(a . 4)`. The more general problem is that certain data types denote a memory location which may be modified by mutator functions such as `set-cdr!`. This is discussed specifically in section 3.4 Storage Model: > > Variables and objects such as pairs, vectors, and strings implicitly denote locations or sequences of locations. A string, for example, denotes as many locations as there are characters in the string. (These locations need not correspond to a full machine word.) A new value may be stored into one of these locations using the string-set! procedure, but the string continues to denote the same locations as before. > Internally husk uses Haskell data types, so the husk model differs slightly from the one in R5RS - references are used instead of individual memory locations. This has implications for the `set-car!` and `set-cdr!` special forms, where circular lists and similar low-level optimizations are not possible as Haskell lists are used instead of raw pointers. However, these issues aside, the enhanced storage model is a big step forward to bringing husk's variable support closer in line to that of other Schemes. v3.5.7 -------- The major change in this release is support for explicit renaming macros. This low-level macro system provides the ability to break macro hygiene, if necessary, and offers a macro system that is similar to `defmacro`. In addition, all of the character functions from R5RS have been implemented. v3.5.6 -------- - Enhanced the compiler to accept `load-ffi` as a special form, so a compiled version of a program does not have to wait for a module to be dynamically loaded. Instead, the module is included at compile time. This offers a nice speed improvement:
$ time huski ffi-cputime.scm
Seconds of CPU time spent: 2.756171
$ time ./ffi-cputime
Seconds of CPU time spent: 2.4001e-2
- Allow a hash table to be defined directly using `#hash(alist)` - for example, `#hash()` for an empty table or `#hash((a 1) (b . 2))` for a table with two elements. This is not part of R5RS but seems less clumsy than the standard way, and a similar language feature is provided by Racket.
- Added support for `hash-table-set!` and `hash-table-delete!` to the compiler.
- Fixed a bug where `integer?` would always return false for negative rational numbers. For example, `(integer? -2/2)` should evaluate to `#t`.
v3.5.5
--------
- Added support for SRFI-1 (List Library), which can be loaded using `(require-extension (srfi 1))`. Note that linear update functions (such as `map!`, `take!`, etc) and circular lists are not supported at this time.
- Added a new LispVal type called `Opaque` for Haskell integration, courtesy of Josh Triplett. The Opaque type allows a native Haskell function to package an arbitrary Haskell type for use by other native code called by husk. See `examples/ffi/Opaque.hs` for an example of how to use this feature.
- Implemented `file-exists?`, `delete-file`, `char-ready?`, `rationalize`, `gcd`, and `lcm`.
- Enhanced the parser to read numbers in scientific notation such as `1e3` and `4.2e1`.
- Modified `numerator` and `denominator` to add support for integers and floating point numbers.
- Fixed a bug in `set-cdr!` where an unsimplified list may be output. For example, `(3 . (2 . (1 . ())))` instead of `(3 2 1)`.
- Allow `apply` to receive multiple arguments, as long as the last one is a list.
v3.5.4
--------
- Enhanced `huski` and `huskc` to print errors to console when a runtime exception occurs,
and to not print any results to console unless the user calls an I/O function
such as `write` or `display`.
- Added a special form `expand` that can be used to see the result of a macro
expansion, for debugging purposes. For example: `(expand (let ((x 1)) x))`.
- Allow a list to be enclosed by matched brackets as well as parentheses. For example: `(+ 1 2 [+ 3 4])`.
- Internal change - cleaned up code by using Language pragmas instead of explicitly using extensions in the cabal and make files.
v3.5.3.2
--------
- Modified huski to escape baskslashes in the path to the standard library, to guarantee stdlib is loaded when husk is installed on a Windows machine.
v3.5.3.1
--------
- Removed an unnecessary dependency on `/dev/null` to allow `huski` to interpret files when run on Windows.
- Prevent divide-by-zero errors when dividing by a single number, or processing a rational number with zero as the denominator.
v3.5.3
------
This release adds full support for GHC 7.2.2 / 7.4.1 as well as a number of small enhancements.
- Fixed the FFI to work in both GHC 7.2.2 and GHC 7.4.1.
- Implemented SRFI 23, which adds error reporting via the `(error)` function.
- Added support for `let-syntax`, `letrec-syntax`, `set-car!`, `set-cdr!`, `vector-set!`, and `lambda` (pair syntax) to the huskc compiler.
- Added the `--dynamic` option to the compiler to use dynamic Haskell libraries.
- Added the `--extra` option to the compiler to allow passing of arbitrary arguments directly to ghc.
- Fixed huski to allow any redefinition of `let-syntax` and `letrec-syntax`.
v3.5.2.x
------
This is a series of quick bug-fix releases that allows husk to build under GHC 7.4.1.
v3.5.2
------
- Added an experimental compiler, huskc. For more information see [Issue #62](https://github.com/justinethier/husk-scheme/issues/62).
- Streamlined the cabal file so that each source file is only compiled a single time.
v3.5.1
------
- Improved support for comparing instances of functions using `eq?`, `eqv?`, etc.
- Reduced variable access time by using a Map to store variables within an environment.
- Various internal changes such as renaming the tests directory and integrating R5RS pitfalls with the unit tests.
v3.4.4
------
This release continues the trend of quick point releases for the 3.4.x series. The key change is support for GHC 7.2:
- husk now compiles in GHC 7.2. This ended up being a simple change - I had mis-interpreted the value of a pre-processor directive. Sorry about that!
- Replaced the definition of `letrec` with the macro from R5RS.
- Allow a continuation to call into another continuation via call/cc - see R5RS pitfall 1.2
v3.4.3
------
- Fixed `let-syntax` and `letrec-syntax` to prevent conflicts between identifiers of the same name in the outer scope and within the macro body.
- Per R5RS, `(if