Skip to content

Latest commit

 

History

History
565 lines (368 loc) · 14.6 KB

README.md

File metadata and controls

565 lines (368 loc) · 14.6 KB

logo

JW-Basic

LoC

A toy language that is somewhat like QBasic.

Features:

  • Graphics: 160x96 (255 colors & transparent)
  • Text: 32x16 (4x5 font)
  • Character set: ASCII (32-127)
  • Keyboard input
  • Multidimensional arrays

Design:

  • Parses tokens using nom and nom_locate
  • Syntax & expression tree parsed with from tokens
  • Assembly-like instructions emitted from syntax
  • Instructions executed using a register-based virtual machine
  • Graphics & text output using Screen-13
  • Operates as a library or command-line program

Language demonstration:

' This is a comment! Types you may use:
'  ?: Boolean
'  @: Unsigned byte
'  %: Signed 32-bit integer
'  !: Single-precision float
'  $: String

DIM myVar$ = "Variable initialization is optional"
DIM answer% = 42, question? = TRUE
DIM latitude!

' Type specifiers are not required:
latitude = 100.0

IF question THEN
    latitude! = -100.0!
END IF

CLS
PRINT "Hello, there!", myVar

' A diagonal red line
LINE (0, 13) - (159, 21), 4@

' Some colorful boxes
FOR c = 25 TO 95 STEP 3
    RECTANGLE (c - 5, c - 3) - (159, c), BYTE(c), TRUE
NEXT

See full specification below

Example Programs

This repository contains several example programs. Access the helpfile with --help:

A BASIC language interpreter. Does not conform to existing standards. Mostly a toy.

Usage: jw-basic [OPTIONS] <PATH>

Arguments:
  <PATH>
          File to load and interpret (.bas format)

Options:
  -h, --help
          Print help (see a summary with '-h')

  -V, --version
          Print version

Hello, world!

examples/hello_world.bas

cargo run examples/hello_world.bas

Preview

Raycast

examples/raycast.bas

cargo run examples/raycast.bas

Preview

Language Specifications

ABS[! | %](expr) | COS[!](expr) | SIN[!](expr)

    Math functions.

    expr: Any expression.

    Examples:

        ABS(-1.0) + COS(4.5) ' Radians of course!


CLS

    CLS clears the screen of text and graphics. Color 0 is used.


COLOR foreground@[, background@]

    COLOR sets text and graphics colors.

    foreground: The color of characters and any lines or rectangles which do not specify a color.
    backround:  The backgroud of text characters or the fill-color of rectangles which do not
                specify a color.

    Examples:

        COLOR 4, 14 ' Red on yellow, danger!


BOOLEAN[?](expr) | BYTE[@](expr) | FLOAT[!](expr) | INT[%](expr) | STR[$](expr)

    Converts an expression to another type.

    expr: Any expression.

    Examples:

        STR(1.0)


DIM var[type][(subscripts)] [= value] [, var[type][(subscripts)]] [= value] ...

    DIM declares variables and arrays. Variables may also be simply assigned without DIM.

    var:        The name of a variable.
    type:       Type of data which may be stored:
                    ? (boolean)
                    @ (byte)
                    ! (float)
                    % (integer)
                    $ (string)
    subscripts: [lower% TO] upper% [, [lower% TO] upper%] ...
                lower: Lower bound of the array. The default bound is zero.
                upper: Upper bound of the array. Inclusive.
    value:      Any expression. Not supported with arrays yet.

    Examples:

        DIM name$, myMatrix!(2, -2 TO 2)
        DIM total% = 5
        myMatrix(2, -2) = 10.0


DO [{WHILE | UNTIL} condition]
   [..]
LOOP

DO
   [..]
LOOP [{WHILE | UNTIL} condition]

    Repeats a block of statements while a condition is true or until a condition becomes true.

    condition:  Any expression which evaluates to a boolean.

    Examples:

        i% = 0
        PRINT "Value of i% at beginning "; i%
        DO WHILE i% < 10
            i% = i% + 1
        LOOP
        PRINT "Value of i% at end "; i%


END {IF | FUNCTION | SUB}

    Ends a program, procedure or block.

    Examples:

        PRINT "Game over."
        END


EXIT {DO | FOR | WHILE | FUNCTION | SUB}

    Exits a DO, FOR or WHILE loop or a FUNCTION or SUB procedure.

    Examples:

        i% = 0
        DO
            i% = i% + 1
            IF i% = 500 THEN
                EXIT DO
            END IF
        LOOP
        PRINT "EXIT at"; i%


FOR var = start TO end [STEP step]
    [..]
NEXT [var]

    Loop where `var` is incremented (or decremented) from `start` to `end` in `step` increments.

    var:   A byte, float, or integer variable defined for the body of the FOR..NEXT statement.
    start: Any expression evaluated to become the initial value of `var`.
    end:   Any expression evaluated to become the inclusive final value of `var`.
    step:  Any expression evaluated to be added to `var` for each iteration.

    Examples:

        FOR temperature = 96.0 to 104.5 STEP 0.1
            PRINT temperature
        NEXT


FUNCTION name{type}[([var{type}][, var{type}] ... )]
    [..]
END FUNCTION

    Declare a function which returns a value. May only be used in the outermost scope, and not
    within any other block. Has global access to preceeding variables and functions. Function name
    is used as a variable to return a value.

    name:   The name of the function.
    type:   Type of data which may be returned or used as an argument:
                    ? (boolean)
                    @ (byte)
                    ! (float)
                    % (integer)
                    $ (string)
    var:    The name of a variable argument.

    Examples:

        FUNCTION areEqual?(lhs$, rhs$)
            IF lhs = rhs THEN
                areEqual = TRUE
            ELSE THEN
                areEqual = FALSE
            END IF
        END FUNCTION

        myGlobal = 0

        FUNCTION changeGlobal%
            myGlobal = myGlobal + 1
            changeGlobal = myGlobal
        END FUNCTION

        PRINT areEqual("Apples", "Oranges")     ' Prints "FALSE"
        PRINT changeGlobal()                    ' Prints "1"


GET (x0, y1) - (x1, y2), arrayname[(index)]
PUT (x0, y1), (width, height), arrayname[(index)][, actionverb]

    GET captures a graphics screen image. PUT displays an image captured by GET.
    PUT Defaults to TSET.

    x0, y0, x1, y1: Any expressions which evaluates to integers.
    width, height:  Any expressions which evaluates to integers.
    arrayname:      The name of the array where the image is stored.
    index:          The integer array index at which storage of the image begins.
    actionverb:     A keyword indicating how the image is displayed:

                    Keyword    Action
                    ═══════    ═════════════════════════════════════════════
                    AND        Merges stored image with an existing image.
                    OR         Superimposes stored image on existing image.
                    PSET       Draws stored image, erasing existing image.
                    PRESET     Draws stored image in reverse colors, erasing
                               existing image.
                    XOR        Draws a stored image or erases a previously
                               drawn image while preserving the background,
                               producing animation effects.
                    TSET       Draws stored image, preserving background
                               where the stored image value is 255.

    Examples:

        DIM fullScreen@(0 TO (160 * 96) - 1)
        GET (0, 0) - (159, 95), fullScreen
        CLS
        PUT (0, 0), (160, 96), fullScreen


GOTO [label | line number]

    Jumps directly to a given labelled or numbered line. Fun at parties.

    Examples:

        Again:
        PRINT "Dance!"
        GOTO Again


IF expr THEN
    [..]
[ELSE IF expr THEN]
    [..]
[ELSE THEN]
    [..]
END IF

    Branching logic tree.

    expr: Any expression which evaluates to a boolean.


KEYDOWN[@](expr)

    Returns TRUE when a given key is pressed.

    expr: Any expression which evaluates to a byte, see source code for the keys which have
          been setup.


LINE [(x0, y0) -] (x1, y1), color

    Draws a line between two points.

    x0, y0, x1, y1: Any expressions which evaluates to integers.
    color:          Any expression which evaluates to a byte.


LOCATE row[, col]

    Moves the text output location of the following PRINT statements.


numeric-expression1 MOD numeric-expression2

    Divides one number by another and returns the remainder.

    numeric-expression1, numeric-expression2:   Any numeric expressions.

    Examples:

        PRINT 19.0 MOD 6.7  'Output is 5.6000004
        PRINT 21 MOD 2      'Output is 1


PALETTE color, r, g, b

    Changes the currently active palette allowing for colorful animation without re-drawing the
    screen.

    color:   Any expression which evaluates to a byte in the 0-254 range. 255 (&hFF@) is
             transparent.
    r, g, b: Any expression which evaluates to a byte.


PEEK[? | @ | ! | % | $](address)

    Returns a byte value (by default) stored at a specified memory location.

    address:    A byte position in the zero-initialized heap memory; a value in the
                range 0 through 16,383.

    Examples:

        myByte = PEEK(420)
        someFloat = PEEK!(128)


POKE address, expr

    Writes a value to a specified memory location.

    address:    A byte position in the zero-initialized heap memory; a value in the
                range 0 through 16,383.
    expr:       Any expression.

    Examples:

        POKE 420, 255@
        POKE 128, 42.0


PRINT [expr][; expr][, expr]

    PRINT displays text using the current foreground and background colors at the current cursor
    location.

    expr:       Any expression.
    semicolon:  Prints the following expression with zero additional spaces.
    comma:      Prints the following expression with one additional space.

    Examples:

        PRINT "Hello " + name$ + ". Nice to meet you!", "Welcome to day ", dayOfWeek%; "!"


PSET (x, y), color

    Draw a specified point on the screen.

    x, y:   Any expression which evaluates to an integer.
    color:  Any expression which evaluates to a byte.


RECTANGLE [(x0, y0) -] (x1, y1), color[, filled]

    Draws a rectangle between two points.

    x0, y0, x1, y1: Any expression which evaluates to an integer.
    color:          Any expression which evaluates to a byte.
    filled:         Any expression which evaluates to a boolean.


RND[!][()]

    Returns a random float number uniformly distributed in the half-open range [0, 1).

    Examples:

        x% = INT(RND * 6.0) + 1
        y% = INT(RND * 6.0) + 1
        PRINT "Roll of two dice: "; x%; "and "; y%


SELECT CASE testexpression
    CASE expressionlist1
        [statementblock-1]
    CASE expressionlist2
        [statementblock-2]...
    CASE ELSE
        [statementblock-n]
END SELECT

    Executes one of several statement blocks depending on the value of an expression. The
    expressionlist arguments can have any of these forms or a combination of them, separated by
    commas:

        expression[, expression]...
        expression TO expression
        IS relational-operator expression

    testexpression:     Any expression.
    expression:         Any expression which evaluates to same type as testexpression.
    relation-operator:  One of the following relational operators: <, <=, >, >=, <>, or =. Boolean
                        and string expressions may only use <> and = operators.

    Examples:

        someValue = 45.0
        SELECT CASE someValue
            CASE IS <= 20.0: PRINT "Too low!"
            CASE someValue - 5.0 TO someValue + 5.0: PRINT "Close enough!"
            CASE IS > 100.0
                PRINT "Too high!"
            CASE ELSE
                PRINT "Over, but not too high!"
        END SELECT


SUB name[([var{type}][, var{type}] ... )]
    [..]
END SUB

    Declare a subroutine. May only be used in the outermost scope, and not within any other block.
    Has global access to preceeding variables and functions. May be called optionally using CALL.

    When calling, if you omit the CALL keyword, also omit the optional parentheses around the
    arguments.

    name:   The name of the sub.
    var:    The name of a variable argument.
    type:   Type of data which may be used as an argument:
                    ? (boolean)
                    @ (byte)
                    ! (float)
                    % (integer)
                    $ (string)

    Examples:

        SUB PrintName(name$)
            PRINT "Hello", name$ + "!"
        END SUB

        PrintName "John"
        CALL PrintName("Keli")


TIMER[%]()

    Returns the number of microseconds since the program began execution.


WHILE expr
    [..]
WEND

    Loop which begins if `expr` is TRUE and continues until it is FALSE.

    expr: Any expression which evaluates to a boolean.


YIELD

    Pause execution of a program until the next update of the interpreter. Without calling this
    execution will continue until the final statement is executed.

Tests

In addition to the test programs, there are unit and integration tests of the language. When something goes wrong you should receive an error indicating the line and column number which caused the issue.

Some Vulkan drivers will fail if too many devices are created at once. When testing you may need to limit the number of threads.

Running the tests:

cargo test -- --test-threads=1

Performance Profiling

To enable profiling with puffin, use the profile-with-puffin feature:

cargo run --features profile-with-puffin -- examples/raycast.bas

Flamegraph of performance data

Credits

This project was designed completely for fun and to learn how a language might be developed. I hope you find something useful that you can bring to your projects, just like I was able to find sources of inspiration for JW-Basic.

Feel free to submit PRs if you would like to enhance this code or fill out remaining features.