Compare commits
55 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| 7690dce3f6 | |||
| 1ed1993489 | |||
| 9b5327b83d | |||
| 10dbe8f8f1 | |||
| 9e4ad7a9a5 | |||
| 35bfa1b9f1 | |||
| 7f692b4cb4 | |||
| 0cef0abdb5 | |||
| 6ba42b5a9b | |||
| 3cb62274c9 | |||
| 60b561d809 | |||
| 70b2dcd829 | |||
| 0955b3ff38 | |||
| 4137935468 | |||
| ebeabcb9d4 | |||
| 4d33a9473a | |||
| 61e3cdba82 | |||
| 3b7d2be87e | |||
| fa175203c9 | |||
| b4a3e9b42b | |||
| 6347778ead | |||
| ba98624e82 | |||
| 24ce965e76 | |||
| 29c130135c | |||
| 01eca37560 | |||
| 8e5ec7d847 | |||
| 9ed6383630 | |||
| 1ec0f63f76 | |||
| d2341ae227 | |||
| 3782f2aaaa | |||
| b636ab9e31 | |||
| cdfe17ad53 | |||
| 3d7d1179c9 | |||
| c3c46b4fc6 | |||
| 0e41b00ef4 | |||
| f6ec6a8c73 | |||
| 2157b2f540 | |||
| 1481216e69 | |||
| f25f389b4e | |||
| deff784df8 | |||
| 54e82846c3 | |||
| 67fce427eb | |||
| 8a2cb61435 | |||
| 50d03e28fc | |||
| 763581c73b | |||
| cdb2613e5d | |||
| 733df87c08 | |||
| bfd506f497 | |||
| 18b59c9e84 | |||
| d3eb31d964 | |||
| 07f4a98b95 | |||
| 0949fd6ff9 | |||
| 03e5096f10 | |||
| bb81b8c474 | |||
| cf6db57787 |
@@ -0,0 +1,10 @@
|
|||||||
|
---
|
||||||
|
name: Question
|
||||||
|
about: Ask a Question
|
||||||
|
labels: question
|
||||||
|
---
|
||||||
|
|
||||||
|
### How can I help?
|
||||||
|
|
||||||
|
I'm always here to help with any inquiries you have regarding Toy and its related projects.
|
||||||
|
|
||||||
@@ -28,6 +28,7 @@ bin/
|
|||||||
*.stackdump
|
*.stackdump
|
||||||
*.tb
|
*.tb
|
||||||
*.filters
|
*.filters
|
||||||
|
[Dd]ocs/
|
||||||
|
|
||||||
#Shell files
|
#Shell files
|
||||||
*.bat
|
*.bat
|
||||||
|
|||||||
@@ -4,9 +4,11 @@
|
|||||||
|
|
||||||
# Toy
|
# Toy
|
||||||
|
|
||||||
This is the Toy programming language interpreter, written in C.
|
The Toy programming language is an imperative bytecode-intermediate embedded scripting language. It isn't intended to operate on its own, but rather as part of another program, the "host". This process is intended to allow a decent amount of easy customisation by the host's end user, by exposing logic in script files. Alternatively, binary files in a custom format can be used as well.
|
||||||
|
|
||||||
Special thanks to http://craftinginterpreters.com/ for their fantastic book that set me on this path.
|
The host will provide all of the extensions needed on a case-by-case basis. Script files have the `.toy` file extension, while binary files have the `.tb` file extension.
|
||||||
|
|
||||||
|
This is the Toy programming language interpreter, written in C.
|
||||||
|
|
||||||
# Nifty Features
|
# Nifty Features
|
||||||
|
|
||||||
@@ -14,7 +16,7 @@ Special thanks to http://craftinginterpreters.com/ for their fantastic book that
|
|||||||
* Bytecode intermediate compilation
|
* Bytecode intermediate compilation
|
||||||
* Optional, but robust type system (including `opaque` for arbitrary data)
|
* Optional, but robust type system (including `opaque` for arbitrary data)
|
||||||
* Functions and types are first-class citizens
|
* Functions and types are first-class citizens
|
||||||
* Import external libraries
|
* Import native libraries from the host
|
||||||
* Fancy slice notation for strings, arrays and dictionaries
|
* Fancy slice notation for strings, arrays and dictionaries
|
||||||
* Can re-direct output, error and assertion failure messages
|
* Can re-direct output, error and assertion failure messages
|
||||||
* Open source under the zlib license
|
* Open source under the zlib license
|
||||||
@@ -71,3 +73,4 @@ This source code is covered by the zlib license (see [LICENSE.md](LICENSE.md)).
|
|||||||
|
|
||||||
* Seth A. Robinson
|
* Seth A. Robinson
|
||||||
|
|
||||||
|
Special thanks to http://craftinginterpreters.com/ for their fantastic book that set me on this path.
|
||||||
+25
-23
@@ -71,12 +71,12 @@
|
|||||||
</PropertyGroup>
|
</PropertyGroup>
|
||||||
<PropertyGroup Condition="'$(Configuration)|$(Platform)'=='Debug|x64'">
|
<PropertyGroup Condition="'$(Configuration)|$(Platform)'=='Debug|x64'">
|
||||||
<IgnoreImportLibrary>false</IgnoreImportLibrary>
|
<IgnoreImportLibrary>false</IgnoreImportLibrary>
|
||||||
<OutDir>$(SolutionDir)out\$(Configuration)\</OutDir>
|
<OutDir>$(SolutionDir)out\</OutDir>
|
||||||
<IntDir>$(Platform)\$(ProjectName)\$(Configuration)\</IntDir>
|
<IntDir>$(Platform)\$(Configuration)\$(ProjectName)\</IntDir>
|
||||||
</PropertyGroup>
|
</PropertyGroup>
|
||||||
<PropertyGroup Condition="'$(Configuration)|$(Platform)'=='Release|x64'">
|
<PropertyGroup Condition="'$(Configuration)|$(Platform)'=='Release|x64'">
|
||||||
<OutDir>$(SolutionDir)out\$(Configuration)\</OutDir>
|
<OutDir>$(SolutionDir)out\</OutDir>
|
||||||
<IntDir>$(Platform)\$(ProjectName)\$(Configuration)\</IntDir>
|
<IntDir>$(Platform)\$(Configuration)\$(ProjectName)\</IntDir>
|
||||||
</PropertyGroup>
|
</PropertyGroup>
|
||||||
<ItemDefinitionGroup Condition="'$(Configuration)|$(Platform)'=='Debug|Win32'">
|
<ItemDefinitionGroup Condition="'$(Configuration)|$(Platform)'=='Debug|Win32'">
|
||||||
<ClCompile>
|
<ClCompile>
|
||||||
@@ -110,45 +110,47 @@
|
|||||||
<ItemDefinitionGroup Condition="'$(Configuration)|$(Platform)'=='Debug|x64'">
|
<ItemDefinitionGroup Condition="'$(Configuration)|$(Platform)'=='Debug|x64'">
|
||||||
<Link>
|
<Link>
|
||||||
<AdditionalDependencies>Toy.lib;%(AdditionalDependencies)</AdditionalDependencies>
|
<AdditionalDependencies>Toy.lib;%(AdditionalDependencies)</AdditionalDependencies>
|
||||||
<AdditionalLibraryDirectories>$(SolutionDir)out\$(Configuration)</AdditionalLibraryDirectories>
|
<AdditionalLibraryDirectories>$(SolutionDir)out</AdditionalLibraryDirectories>
|
||||||
</Link>
|
</Link>
|
||||||
<ClCompile>
|
<ClCompile>
|
||||||
<AdditionalIncludeDirectories>$(SolutionDir)/source;%(AdditionalIncludeDirectories)</AdditionalIncludeDirectories>
|
<AdditionalIncludeDirectories>$(SolutionDir)/source;%(AdditionalIncludeDirectories)</AdditionalIncludeDirectories>
|
||||||
<LanguageStandard_C>stdc17</LanguageStandard_C>
|
<LanguageStandard_C>stdc17</LanguageStandard_C>
|
||||||
<PreprocessorDefinitions>LIB_RUNNER_EXPORT</PreprocessorDefinitions>
|
<PreprocessorDefinitions>%(PreprocessorDefinitions)</PreprocessorDefinitions>
|
||||||
</ClCompile>
|
</ClCompile>
|
||||||
</ItemDefinitionGroup>
|
</ItemDefinitionGroup>
|
||||||
<ItemDefinitionGroup Condition="'$(Configuration)|$(Platform)'=='Release|x64'">
|
<ItemDefinitionGroup Condition="'$(Configuration)|$(Platform)'=='Release|x64'">
|
||||||
<ClCompile>
|
<ClCompile>
|
||||||
<LanguageStandard_C>stdc17</LanguageStandard_C>
|
<LanguageStandard_C>stdc17</LanguageStandard_C>
|
||||||
<PreprocessorDefinitions>LIB_RUNNER_EXPORT</PreprocessorDefinitions>
|
<PreprocessorDefinitions>%(PreprocessorDefinitions)</PreprocessorDefinitions>
|
||||||
<AdditionalIncludeDirectories>$(SolutionDir)/source;%(AdditionalIncludeDirectories)</AdditionalIncludeDirectories>
|
<AdditionalIncludeDirectories>$(SolutionDir)/source;%(AdditionalIncludeDirectories)</AdditionalIncludeDirectories>
|
||||||
</ClCompile>
|
</ClCompile>
|
||||||
<Link>
|
<Link>
|
||||||
<AdditionalLibraryDirectories>$(SolutionDir)out\$(Configuration)</AdditionalLibraryDirectories>
|
<AdditionalLibraryDirectories>$(SolutionDir)out</AdditionalLibraryDirectories>
|
||||||
<AdditionalDependencies>Toy.lib;%(AdditionalDependencies)</AdditionalDependencies>
|
<AdditionalDependencies>Toy.lib;%(AdditionalDependencies)</AdditionalDependencies>
|
||||||
</Link>
|
</Link>
|
||||||
</ItemDefinitionGroup>
|
</ItemDefinitionGroup>
|
||||||
<ItemGroup>
|
|
||||||
<ClCompile Include="repl\lib_about.c" />
|
|
||||||
<ClCompile Include="repl\lib_random.c" />
|
|
||||||
<ClCompile Include="repl\lib_runner.c" />
|
|
||||||
<ClCompile Include="repl\lib_standard.c" />
|
|
||||||
<ClCompile Include="repl\repl_main.c" />
|
|
||||||
<ClCompile Include="repl\repl_tools.c" />
|
|
||||||
</ItemGroup>
|
|
||||||
<ItemGroup>
|
|
||||||
<ClInclude Include="repl\lib_about.h" />
|
|
||||||
<ClInclude Include="repl\lib_random.h" />
|
|
||||||
<ClInclude Include="repl\lib_runner.h" />
|
|
||||||
<ClInclude Include="repl\lib_standard.h" />
|
|
||||||
<ClInclude Include="repl\repl_tools.h" />
|
|
||||||
</ItemGroup>
|
|
||||||
<ItemGroup>
|
<ItemGroup>
|
||||||
<ProjectReference Include="Toy.vcxproj">
|
<ProjectReference Include="Toy.vcxproj">
|
||||||
<Project>{26360002-cc2a-469a-9b28-ba0c1af41657}</Project>
|
<Project>{26360002-cc2a-469a-9b28-ba0c1af41657}</Project>
|
||||||
</ProjectReference>
|
</ProjectReference>
|
||||||
</ItemGroup>
|
</ItemGroup>
|
||||||
|
<ItemGroup>
|
||||||
|
<ClCompile Include="repl\drive_system.c" />
|
||||||
|
<ClCompile Include="repl\lib_random.c" />
|
||||||
|
<ClCompile Include="repl\lib_runner.c" />
|
||||||
|
<ClCompile Include="repl\lib_standard.c" />
|
||||||
|
<ClCompile Include="repl\lib_toy_version_info.c" />
|
||||||
|
<ClCompile Include="repl\repl_main.c" />
|
||||||
|
<ClCompile Include="repl\repl_tools.c" />
|
||||||
|
</ItemGroup>
|
||||||
|
<ItemGroup>
|
||||||
|
<ClInclude Include="repl\drive_system.h" />
|
||||||
|
<ClInclude Include="repl\lib_random.h" />
|
||||||
|
<ClInclude Include="repl\lib_runner.h" />
|
||||||
|
<ClInclude Include="repl\lib_standard.h" />
|
||||||
|
<ClInclude Include="repl\lib_toy_version_info.h" />
|
||||||
|
<ClInclude Include="repl\repl_tools.h" />
|
||||||
|
</ItemGroup>
|
||||||
<Import Project="$(VCTargetsPath)\Microsoft.Cpp.targets" />
|
<Import Project="$(VCTargetsPath)\Microsoft.Cpp.targets" />
|
||||||
<ImportGroup Label="ExtensionTargets">
|
<ImportGroup Label="ExtensionTargets">
|
||||||
</ImportGroup>
|
</ImportGroup>
|
||||||
|
|||||||
+28
-6
@@ -70,12 +70,14 @@
|
|||||||
<LinkIncremental>true</LinkIncremental>
|
<LinkIncremental>true</LinkIncremental>
|
||||||
</PropertyGroup>
|
</PropertyGroup>
|
||||||
<PropertyGroup Condition="'$(Configuration)|$(Platform)'=='Debug|x64'">
|
<PropertyGroup Condition="'$(Configuration)|$(Platform)'=='Debug|x64'">
|
||||||
<OutDir>$(SolutionDir)out\$(Configuration)\</OutDir>
|
<OutDir>$(SolutionDir)out\</OutDir>
|
||||||
<IntDir>$(Platform)\$(ProjectName)\$(Configuration)\</IntDir>
|
<IntDir>$(Platform)\$(Configuration)\$(ProjectName)\</IntDir>
|
||||||
|
<LinkIncremental>false</LinkIncremental>
|
||||||
</PropertyGroup>
|
</PropertyGroup>
|
||||||
<PropertyGroup Condition="'$(Configuration)|$(Platform)'=='Release|x64'">
|
<PropertyGroup Condition="'$(Configuration)|$(Platform)'=='Release|x64'">
|
||||||
<OutDir>$(SolutionDir)out\$(Configuration)\</OutDir>
|
<OutDir>$(SolutionDir)out\</OutDir>
|
||||||
<IntDir>$(Platform)\$(ProjectName)\$(Configuration)\</IntDir>
|
<IntDir>$(Platform)\$(Configuration)\$(ProjectName)\</IntDir>
|
||||||
|
<LinkIncremental>false</LinkIncremental>
|
||||||
</PropertyGroup>
|
</PropertyGroup>
|
||||||
<ItemDefinitionGroup Condition="'$(Configuration)|$(Platform)'=='Debug|Win32'">
|
<ItemDefinitionGroup Condition="'$(Configuration)|$(Platform)'=='Debug|Win32'">
|
||||||
<ClCompile>
|
<ClCompile>
|
||||||
@@ -111,19 +113,38 @@
|
|||||||
<PreprocessorDefinitions>TOY_EXPORT;%(PreprocessorDefinitions)</PreprocessorDefinitions>
|
<PreprocessorDefinitions>TOY_EXPORT;%(PreprocessorDefinitions)</PreprocessorDefinitions>
|
||||||
<LanguageStandard_C>stdc17</LanguageStandard_C>
|
<LanguageStandard_C>stdc17</LanguageStandard_C>
|
||||||
</ClCompile>
|
</ClCompile>
|
||||||
|
<PostBuildEvent>
|
||||||
|
<Command>
|
||||||
|
</Command>
|
||||||
|
</PostBuildEvent>
|
||||||
|
<Link>
|
||||||
|
<OutputFile>$(Outdir)$(TargetName)$(TargetExt)</OutputFile>
|
||||||
|
</Link>
|
||||||
|
<Bscmake>
|
||||||
|
<OutputFile>$(Platform)\$(Configuration)\$(TargetName).bsc</OutputFile>
|
||||||
|
</Bscmake>
|
||||||
</ItemDefinitionGroup>
|
</ItemDefinitionGroup>
|
||||||
<ItemDefinitionGroup Condition="'$(Configuration)|$(Platform)'=='Release|x64'">
|
<ItemDefinitionGroup Condition="'$(Configuration)|$(Platform)'=='Release|x64'">
|
||||||
<ClCompile>
|
<ClCompile>
|
||||||
<LanguageStandard_C>stdc17</LanguageStandard_C>
|
<LanguageStandard_C>stdc17</LanguageStandard_C>
|
||||||
<PreprocessorDefinitions>TOY_EXPORT;%(PreprocessorDefinitions)</PreprocessorDefinitions>
|
<PreprocessorDefinitions>TOY_EXPORT;%(PreprocessorDefinitions)</PreprocessorDefinitions>
|
||||||
</ClCompile>
|
</ClCompile>
|
||||||
|
<PostBuildEvent>
|
||||||
|
<Command>
|
||||||
|
</Command>
|
||||||
|
</PostBuildEvent>
|
||||||
|
<Link>
|
||||||
|
<OutputFile>$(Outdir)$(TargetName)$(TargetExt)</OutputFile>
|
||||||
|
</Link>
|
||||||
|
<Bscmake>
|
||||||
|
<OutputFile>$(Platform)\$(Configuration)\$(TargetName).bsc</OutputFile>
|
||||||
|
</Bscmake>
|
||||||
</ItemDefinitionGroup>
|
</ItemDefinitionGroup>
|
||||||
<ItemGroup>
|
<ItemGroup>
|
||||||
<ClCompile Include="source\toy_ast_node.c" />
|
<ClCompile Include="source\toy_ast_node.c" />
|
||||||
<ClCompile Include="source\toy_builtin.c" />
|
<ClCompile Include="source\toy_builtin.c" />
|
||||||
<ClCompile Include="source\toy_common.c" />
|
<ClCompile Include="source\toy_common.c" />
|
||||||
<ClCompile Include="source\toy_compiler.c" />
|
<ClCompile Include="source\toy_compiler.c" />
|
||||||
<ClCompile Include="source\toy_drive_system.c" />
|
|
||||||
<ClCompile Include="source\toy_interpreter.c" />
|
<ClCompile Include="source\toy_interpreter.c" />
|
||||||
<ClCompile Include="source\toy_keyword_types.c" />
|
<ClCompile Include="source\toy_keyword_types.c" />
|
||||||
<ClCompile Include="source\toy_lexer.c" />
|
<ClCompile Include="source\toy_lexer.c" />
|
||||||
@@ -132,6 +153,7 @@
|
|||||||
<ClCompile Include="source\toy_literal_dictionary.c" />
|
<ClCompile Include="source\toy_literal_dictionary.c" />
|
||||||
<ClCompile Include="source\toy_memory.c" />
|
<ClCompile Include="source\toy_memory.c" />
|
||||||
<ClCompile Include="source\toy_parser.c" />
|
<ClCompile Include="source\toy_parser.c" />
|
||||||
|
<ClCompile Include="source\toy_reffunction.c" />
|
||||||
<ClCompile Include="source\toy_refstring.c" />
|
<ClCompile Include="source\toy_refstring.c" />
|
||||||
<ClCompile Include="source\toy_scope.c" />
|
<ClCompile Include="source\toy_scope.c" />
|
||||||
</ItemGroup>
|
</ItemGroup>
|
||||||
@@ -142,7 +164,6 @@
|
|||||||
<ClInclude Include="source\toy_common.h" />
|
<ClInclude Include="source\toy_common.h" />
|
||||||
<ClInclude Include="source\toy_compiler.h" />
|
<ClInclude Include="source\toy_compiler.h" />
|
||||||
<ClInclude Include="source\toy_console_colors.h" />
|
<ClInclude Include="source\toy_console_colors.h" />
|
||||||
<ClInclude Include="source\toy_drive_system.h" />
|
|
||||||
<ClInclude Include="source\toy_interpreter.h" />
|
<ClInclude Include="source\toy_interpreter.h" />
|
||||||
<ClInclude Include="source\toy_keyword_types.h" />
|
<ClInclude Include="source\toy_keyword_types.h" />
|
||||||
<ClInclude Include="source\toy_lexer.h" />
|
<ClInclude Include="source\toy_lexer.h" />
|
||||||
@@ -152,6 +173,7 @@
|
|||||||
<ClInclude Include="source\toy_memory.h" />
|
<ClInclude Include="source\toy_memory.h" />
|
||||||
<ClInclude Include="source\toy_opcodes.h" />
|
<ClInclude Include="source\toy_opcodes.h" />
|
||||||
<ClInclude Include="source\toy_parser.h" />
|
<ClInclude Include="source\toy_parser.h" />
|
||||||
|
<ClInclude Include="source\toy_reffunction.h" />
|
||||||
<ClInclude Include="source\toy_refstring.h" />
|
<ClInclude Include="source\toy_refstring.h" />
|
||||||
<ClInclude Include="source\toy_scope.h" />
|
<ClInclude Include="source\toy_scope.h" />
|
||||||
<ClInclude Include="source\toy_token_types.h" />
|
<ClInclude Include="source\toy_token_types.h" />
|
||||||
|
|||||||
@@ -1,7 +1,3 @@
|
|||||||
# Optimisation Options
|
|
||||||
# export CFLAGS+=-O2 -mtune=native -march=native
|
|
||||||
# export CFLAGS+=-fsanitize=address,undefined
|
|
||||||
|
|
||||||
export CFLAGS+=-std=c18 -pedantic -Werror
|
export CFLAGS+=-std=c18 -pedantic -Werror
|
||||||
|
|
||||||
export TOY_OUTDIR = out
|
export TOY_OUTDIR = out
|
||||||
@@ -34,6 +30,10 @@ library-release: clean $(TOY_OUTDIR)
|
|||||||
static-release: clean $(TOY_OUTDIR)
|
static-release: clean $(TOY_OUTDIR)
|
||||||
$(MAKE) -j8 -C source static-release
|
$(MAKE) -j8 -C source static-release
|
||||||
|
|
||||||
|
#distribution
|
||||||
|
dist: export CFLAGS+=-O2 -mtune=native -march=native
|
||||||
|
dist: repl-release
|
||||||
|
|
||||||
#utils
|
#utils
|
||||||
test: clean $(TOY_OUTDIR)
|
test: clean $(TOY_OUTDIR)
|
||||||
$(MAKE) -C test
|
$(MAKE) -C test
|
||||||
@@ -51,6 +51,25 @@ $(TOY_OUTDIR):
|
|||||||
install-tools:
|
install-tools:
|
||||||
cp -rf tools/toylang.vscode-highlighting ~/.vscode/extensions
|
cp -rf tools/toylang.vscode-highlighting ~/.vscode/extensions
|
||||||
|
|
||||||
|
#utils
|
||||||
|
build-mecha: $(TOY_OUTDIR)
|
||||||
|
g++ -o $(TOY_OUTDIR)/mecha tools/mecha.cpp
|
||||||
|
|
||||||
|
build-docs: build-mecha
|
||||||
|
$(TOY_OUTDIR)/mecha $(wildcard source/*.h)
|
||||||
|
$(TOY_OUTDIR)/mecha $(wildcard repl/*.h)
|
||||||
|
|
||||||
|
docs:
|
||||||
|
mkdir docs
|
||||||
|
|
||||||
|
move-docs: docs
|
||||||
|
mv -u $(wildcard source/*.md) docs
|
||||||
|
mv -u $(wildcard repl/*.md) docs
|
||||||
|
|
||||||
|
documentation:
|
||||||
|
$(MAKE) build-docs
|
||||||
|
$(MAKE) move-docs
|
||||||
|
|
||||||
.PHONY: clean
|
.PHONY: clean
|
||||||
|
|
||||||
clean:
|
clean:
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
#include "toy_drive_system.h"
|
#include "drive_system.h"
|
||||||
|
|
||||||
#include "toy_memory.h"
|
#include "toy_memory.h"
|
||||||
#include "toy_literal_dictionary.h"
|
#include "toy_literal_dictionary.h"
|
||||||
@@ -18,7 +18,7 @@ void Toy_freeDriveSystem() {
|
|||||||
Toy_freeLiteralDictionary(&driveDictionary);
|
Toy_freeLiteralDictionary(&driveDictionary);
|
||||||
}
|
}
|
||||||
|
|
||||||
TOY_API void Toy_setDrivePath(char* drive, char* path) {
|
void Toy_setDrivePath(char* drive, char* path) {
|
||||||
Toy_Literal driveLiteral = TOY_TO_STRING_LITERAL(Toy_createRefString(drive));
|
Toy_Literal driveLiteral = TOY_TO_STRING_LITERAL(Toy_createRefString(drive));
|
||||||
Toy_Literal pathLiteral = TOY_TO_STRING_LITERAL(Toy_createRefString(path));
|
Toy_Literal pathLiteral = TOY_TO_STRING_LITERAL(Toy_createRefString(path));
|
||||||
|
|
||||||
@@ -0,0 +1,76 @@
|
|||||||
|
#pragma once
|
||||||
|
|
||||||
|
/*!
|
||||||
|
# drive_system.h
|
||||||
|
|
||||||
|
When accessing the file system through Toy (such as with the runner library), it's best practice to utilize the drive system - this system (tries to) prevent malicious accessing of files outside of the designated folders. It does this by causing an error when a script tries to access a parent directory.
|
||||||
|
|
||||||
|
To use the drive system, first you must designate specific folders which can be accessed, like so:
|
||||||
|
|
||||||
|
```c
|
||||||
|
#include "drive_system.h"
|
||||||
|
|
||||||
|
int main(int argc, char* argv[]) {
|
||||||
|
//the drive system uses a LiteralDictionary, which must be initialized with this
|
||||||
|
Toy_initDriveSystem();
|
||||||
|
|
||||||
|
Toy_setDrivePath("scripts", "assets/scripts");
|
||||||
|
Toy_setDrivePath("sprites", "assets/sprites");
|
||||||
|
Toy_setDrivePath("fonts", "assets/fonts");
|
||||||
|
|
||||||
|
//TODO: do you stuff here
|
||||||
|
|
||||||
|
//clean up the drive dictionary when you're done
|
||||||
|
Toy_freeDriveSystem();
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
This utility is intended mainly for libraries to use - as such, the core of Toy does not utilize it.
|
||||||
|
|
||||||
|
### Implementation Details
|
||||||
|
|
||||||
|
The drive system uses a Toy's Dictionary structure to store the mappings between keys and values - this dictionary object is a static global which persists for the lifetime of the program.
|
||||||
|
!*/
|
||||||
|
|
||||||
|
#include "toy_common.h"
|
||||||
|
|
||||||
|
#include "toy_literal.h"
|
||||||
|
#include "toy_interpreter.h"
|
||||||
|
|
||||||
|
/*!
|
||||||
|
## Defined Functions
|
||||||
|
!*/
|
||||||
|
|
||||||
|
/*!
|
||||||
|
### void Toy_initDriveSystem()
|
||||||
|
|
||||||
|
This function initializes the drive system.
|
||||||
|
!*/
|
||||||
|
TOY_API void Toy_initDriveSystem();
|
||||||
|
|
||||||
|
/*!
|
||||||
|
### void Toy_freeDriveSystem()
|
||||||
|
|
||||||
|
This function cleans up after the drive system is no longer needed.
|
||||||
|
!*/
|
||||||
|
TOY_API void Toy_freeDriveSystem();
|
||||||
|
|
||||||
|
/*!
|
||||||
|
### void Toy_setDrivePath(char* drive, char* path)
|
||||||
|
|
||||||
|
This function sets a key-value pair in the drive system. It uses C strings, since its intended to be called directly from `main()`.
|
||||||
|
!*/
|
||||||
|
TOY_API void Toy_setDrivePath(char* drive, char* path);
|
||||||
|
|
||||||
|
/*!
|
||||||
|
### Toy_Literal Toy_getDrivePathLiteral(Toy_Interpreter* interpreter, Toy_Literal* drivePathLiteral)
|
||||||
|
|
||||||
|
This function, when given a string literal of the correct format, will return a new string literal containing the relative filepath to a specified file.
|
||||||
|
|
||||||
|
The correct format is `drive:/path/to/filename`, where `drive` is a drive that was specified with `Toy_setDrivePath()`.
|
||||||
|
|
||||||
|
On failure, this function returns a null literal.
|
||||||
|
!*/
|
||||||
|
TOY_API Toy_Literal Toy_getDrivePathLiteral(Toy_Interpreter* interpreter, Toy_Literal* drivePathLiteral);
|
||||||
@@ -1,6 +0,0 @@
|
|||||||
#pragma once
|
|
||||||
|
|
||||||
#include "toy_interpreter.h"
|
|
||||||
|
|
||||||
int Toy_hookAbout(Toy_Interpreter* interpreter, Toy_Literal identifier, Toy_Literal alias);
|
|
||||||
|
|
||||||
+1
-16
@@ -30,11 +30,6 @@ static int nativeCreateRandomGenerator(Toy_Interpreter* interpreter, Toy_Literal
|
|||||||
Toy_freeLiteral(seedLiteralIdn);
|
Toy_freeLiteral(seedLiteralIdn);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (TOY_IS_IDENTIFIER(seedLiteral)) {
|
|
||||||
Toy_freeLiteral(seedLiteral);
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!TOY_IS_INTEGER(seedLiteral)) {
|
if (!TOY_IS_INTEGER(seedLiteral)) {
|
||||||
interpreter->errorOutput("Incorrect literal type passed to createRandomGenerator");
|
interpreter->errorOutput("Incorrect literal type passed to createRandomGenerator");
|
||||||
Toy_freeLiteral(seedLiteral);
|
Toy_freeLiteral(seedLiteral);
|
||||||
@@ -70,11 +65,6 @@ static int nativeGenerateRandomNumber(Toy_Interpreter* interpreter, Toy_LiteralA
|
|||||||
Toy_freeLiteral(generatorLiteralIdn);
|
Toy_freeLiteral(generatorLiteralIdn);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (TOY_IS_IDENTIFIER(generatorLiteral)) {
|
|
||||||
Toy_freeLiteral(generatorLiteral);
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (TOY_GET_OPAQUE_TAG(generatorLiteral) != TOY_OPAQUE_TAG_RANDOM) {
|
if (TOY_GET_OPAQUE_TAG(generatorLiteral) != TOY_OPAQUE_TAG_RANDOM) {
|
||||||
interpreter->errorOutput("Unrecognized opaque literal in generateRandomNumber\n");
|
interpreter->errorOutput("Unrecognized opaque literal in generateRandomNumber\n");
|
||||||
return -1;
|
return -1;
|
||||||
@@ -111,11 +101,6 @@ static int nativeFreeRandomGenerator(Toy_Interpreter* interpreter, Toy_LiteralAr
|
|||||||
Toy_freeLiteral(generatorLiteralIdn);
|
Toy_freeLiteral(generatorLiteralIdn);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (TOY_IS_IDENTIFIER(generatorLiteral)) {
|
|
||||||
Toy_freeLiteral(generatorLiteral);
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (TOY_GET_OPAQUE_TAG(generatorLiteral) != TOY_OPAQUE_TAG_RANDOM) {
|
if (TOY_GET_OPAQUE_TAG(generatorLiteral) != TOY_OPAQUE_TAG_RANDOM) {
|
||||||
interpreter->errorOutput("Unrecognized opaque literal in freeRandomGenerator\n");
|
interpreter->errorOutput("Unrecognized opaque literal in freeRandomGenerator\n");
|
||||||
return -1;
|
return -1;
|
||||||
@@ -148,7 +133,7 @@ int Toy_hookRandom(Toy_Interpreter* interpreter, Toy_Literal identifier, Toy_Lit
|
|||||||
//store the library in an aliased dictionary
|
//store the library in an aliased dictionary
|
||||||
if (!TOY_IS_NULL(alias)) {
|
if (!TOY_IS_NULL(alias)) {
|
||||||
//make sure the name isn't taken
|
//make sure the name isn't taken
|
||||||
if (Toy_isDelcaredScopeVariable(interpreter->scope, alias)) {
|
if (Toy_isDeclaredScopeVariable(interpreter->scope, alias)) {
|
||||||
interpreter->errorOutput("Can't override an existing variable\n");
|
interpreter->errorOutput("Can't override an existing variable\n");
|
||||||
Toy_freeLiteral(alias);
|
Toy_freeLiteral(alias);
|
||||||
return -1;
|
return -1;
|
||||||
|
|||||||
+2
-2
@@ -2,6 +2,6 @@
|
|||||||
|
|
||||||
#include "toy_interpreter.h"
|
#include "toy_interpreter.h"
|
||||||
|
|
||||||
int Toy_hookRandom(Toy_Interpreter* interpreter, Toy_Literal identifier, Toy_Literal alias);
|
|
||||||
|
|
||||||
#define TOY_OPAQUE_TAG_RANDOM 200
|
#define TOY_OPAQUE_TAG_RANDOM 200
|
||||||
|
|
||||||
|
int Toy_hookRandom(Toy_Interpreter* interpreter, Toy_Literal identifier, Toy_Literal alias);
|
||||||
|
|||||||
+2
-44
@@ -1,10 +1,10 @@
|
|||||||
#include "lib_runner.h"
|
#include "lib_runner.h"
|
||||||
|
|
||||||
#include "toy_memory.h"
|
#include "toy_memory.h"
|
||||||
#include "toy_drive_system.h"
|
|
||||||
#include "toy_interpreter.h"
|
#include "toy_interpreter.h"
|
||||||
|
|
||||||
#include "repl_tools.h"
|
#include "repl_tools.h"
|
||||||
|
#include "drive_system.h"
|
||||||
|
|
||||||
#include <stdlib.h>
|
#include <stdlib.h>
|
||||||
|
|
||||||
@@ -32,11 +32,6 @@ static int nativeLoadScript(Toy_Interpreter* interpreter, Toy_LiteralArray* argu
|
|||||||
Toy_freeLiteral(drivePathLiteralIdn);
|
Toy_freeLiteral(drivePathLiteralIdn);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (TOY_IS_IDENTIFIER(drivePathLiteral)) {
|
|
||||||
Toy_freeLiteral(drivePathLiteral);
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
|
|
||||||
Toy_Literal filePathLiteral = Toy_getDrivePathLiteral(interpreter, &drivePathLiteral);
|
Toy_Literal filePathLiteral = Toy_getDrivePathLiteral(interpreter, &drivePathLiteral);
|
||||||
|
|
||||||
if (TOY_IS_NULL(filePathLiteral)) {
|
if (TOY_IS_NULL(filePathLiteral)) {
|
||||||
@@ -107,11 +102,6 @@ static int nativeLoadScriptBytecode(Toy_Interpreter* interpreter, Toy_LiteralArr
|
|||||||
Toy_freeLiteral(drivePathLiteralIdn);
|
Toy_freeLiteral(drivePathLiteralIdn);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (TOY_IS_IDENTIFIER(drivePathLiteral)) {
|
|
||||||
Toy_freeLiteral(drivePathLiteral);
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
|
|
||||||
Toy_Literal filePathLiteral = Toy_getDrivePathLiteral(interpreter, &drivePathLiteral);
|
Toy_Literal filePathLiteral = Toy_getDrivePathLiteral(interpreter, &drivePathLiteral);
|
||||||
|
|
||||||
if (TOY_IS_NULL(filePathLiteral)) {
|
if (TOY_IS_NULL(filePathLiteral)) {
|
||||||
@@ -172,11 +162,6 @@ static int nativeRunScript(Toy_Interpreter* interpreter, Toy_LiteralArray* argum
|
|||||||
Toy_freeLiteral(runnerIdn);
|
Toy_freeLiteral(runnerIdn);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (TOY_IS_IDENTIFIER(runnerLiteral)) {
|
|
||||||
Toy_freeLiteral(runnerLiteral);
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (TOY_GET_OPAQUE_TAG(runnerLiteral) != TOY_OPAQUE_TAG_RUNNER) {
|
if (TOY_GET_OPAQUE_TAG(runnerLiteral) != TOY_OPAQUE_TAG_RUNNER) {
|
||||||
interpreter->errorOutput("Unrecognized opaque literal in runScript\n");
|
interpreter->errorOutput("Unrecognized opaque literal in runScript\n");
|
||||||
return -1;
|
return -1;
|
||||||
@@ -224,12 +209,6 @@ static int nativeGetScriptVar(Toy_Interpreter* interpreter, Toy_LiteralArray* ar
|
|||||||
Toy_freeLiteral(runnerIdn);
|
Toy_freeLiteral(runnerIdn);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (TOY_IS_IDENTIFIER(varName) || TOY_IS_IDENTIFIER(runnerLiteral)) {
|
|
||||||
Toy_freeLiteral(varName);
|
|
||||||
Toy_freeLiteral(runnerLiteral);
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (TOY_GET_OPAQUE_TAG(runnerLiteral) != TOY_OPAQUE_TAG_RUNNER) {
|
if (TOY_GET_OPAQUE_TAG(runnerLiteral) != TOY_OPAQUE_TAG_RUNNER) {
|
||||||
interpreter->errorOutput("Unrecognized opaque literal in getScriptVar\n");
|
interpreter->errorOutput("Unrecognized opaque literal in getScriptVar\n");
|
||||||
return -1;
|
return -1;
|
||||||
@@ -302,12 +281,6 @@ static int nativeCallScriptFn(Toy_Interpreter* interpreter, Toy_LiteralArray* ar
|
|||||||
Toy_freeLiteral(runnerIdn);
|
Toy_freeLiteral(runnerIdn);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (TOY_IS_IDENTIFIER(varName) || TOY_IS_IDENTIFIER(runnerLiteral)) {
|
|
||||||
Toy_freeLiteral(varName);
|
|
||||||
Toy_freeLiteral(runnerLiteral);
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (TOY_GET_OPAQUE_TAG(runnerLiteral) != TOY_OPAQUE_TAG_RUNNER) {
|
if (TOY_GET_OPAQUE_TAG(runnerLiteral) != TOY_OPAQUE_TAG_RUNNER) {
|
||||||
interpreter->errorOutput("Unrecognized opaque literal in callScriptFn\n");
|
interpreter->errorOutput("Unrecognized opaque literal in callScriptFn\n");
|
||||||
return -1;
|
return -1;
|
||||||
@@ -377,11 +350,6 @@ static int nativeResetScript(Toy_Interpreter* interpreter, Toy_LiteralArray* arg
|
|||||||
Toy_freeLiteral(runnerIdn);
|
Toy_freeLiteral(runnerIdn);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (TOY_IS_IDENTIFIER(runnerLiteral)) {
|
|
||||||
Toy_freeLiteral(runnerLiteral);
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (TOY_GET_OPAQUE_TAG(runnerLiteral) != TOY_OPAQUE_TAG_RUNNER) {
|
if (TOY_GET_OPAQUE_TAG(runnerLiteral) != TOY_OPAQUE_TAG_RUNNER) {
|
||||||
interpreter->errorOutput("Unrecognized opaque literal in resetScript\n");
|
interpreter->errorOutput("Unrecognized opaque literal in resetScript\n");
|
||||||
return -1;
|
return -1;
|
||||||
@@ -418,11 +386,6 @@ static int nativeFreeScript(Toy_Interpreter* interpreter, Toy_LiteralArray* argu
|
|||||||
Toy_freeLiteral(runnerIdn);
|
Toy_freeLiteral(runnerIdn);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (TOY_IS_IDENTIFIER(runnerLiteral)) {
|
|
||||||
Toy_freeLiteral(runnerLiteral);
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (TOY_GET_OPAQUE_TAG(runnerLiteral) != TOY_OPAQUE_TAG_RUNNER) {
|
if (TOY_GET_OPAQUE_TAG(runnerLiteral) != TOY_OPAQUE_TAG_RUNNER) {
|
||||||
interpreter->errorOutput("Unrecognized opaque literal in freeScript\n");
|
interpreter->errorOutput("Unrecognized opaque literal in freeScript\n");
|
||||||
return -1;
|
return -1;
|
||||||
@@ -457,11 +420,6 @@ static int nativeCheckScriptDirty(Toy_Interpreter* interpreter, Toy_LiteralArray
|
|||||||
Toy_freeLiteral(runnerIdn);
|
Toy_freeLiteral(runnerIdn);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (TOY_IS_IDENTIFIER(runnerLiteral)) {
|
|
||||||
Toy_freeLiteral(runnerLiteral);
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (TOY_GET_OPAQUE_TAG(runnerLiteral) != TOY_OPAQUE_TAG_RUNNER) {
|
if (TOY_GET_OPAQUE_TAG(runnerLiteral) != TOY_OPAQUE_TAG_RUNNER) {
|
||||||
interpreter->errorOutput("Unrecognized opaque literal in checkScriptDirty\n");
|
interpreter->errorOutput("Unrecognized opaque literal in checkScriptDirty\n");
|
||||||
return -1;
|
return -1;
|
||||||
@@ -504,7 +462,7 @@ int Toy_hookRunner(Toy_Interpreter* interpreter, Toy_Literal identifier, Toy_Lit
|
|||||||
//store the library in an aliased dictionary
|
//store the library in an aliased dictionary
|
||||||
if (!TOY_IS_NULL(alias)) {
|
if (!TOY_IS_NULL(alias)) {
|
||||||
//make sure the name isn't taken
|
//make sure the name isn't taken
|
||||||
if (Toy_isDelcaredScopeVariable(interpreter->scope, alias)) {
|
if (Toy_isDeclaredScopeVariable(interpreter->scope, alias)) {
|
||||||
interpreter->errorOutput("Can't override an existing variable\n");
|
interpreter->errorOutput("Can't override an existing variable\n");
|
||||||
Toy_freeLiteral(alias);
|
Toy_freeLiteral(alias);
|
||||||
return -1;
|
return -1;
|
||||||
|
|||||||
+1
-2
@@ -2,7 +2,6 @@
|
|||||||
|
|
||||||
#include "toy_interpreter.h"
|
#include "toy_interpreter.h"
|
||||||
|
|
||||||
int Toy_hookRunner(Toy_Interpreter* interpreter, Toy_Literal identifier, Toy_Literal alias);
|
|
||||||
|
|
||||||
#define TOY_OPAQUE_TAG_RUNNER 100
|
#define TOY_OPAQUE_TAG_RUNNER 100
|
||||||
|
|
||||||
|
int Toy_hookRunner(Toy_Interpreter* interpreter, Toy_Literal identifier, Toy_Literal alias);
|
||||||
|
|||||||
+296
-159
@@ -47,11 +47,6 @@ static int nativeHash(Toy_Interpreter* interpreter, Toy_LiteralArray* arguments)
|
|||||||
Toy_freeLiteral(selfLiteralIdn);
|
Toy_freeLiteral(selfLiteralIdn);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (TOY_IS_IDENTIFIER(selfLiteral)) {
|
|
||||||
Toy_freeLiteral(selfLiteral);
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
|
|
||||||
Toy_Literal result = TOY_TO_INTEGER_LITERAL(Toy_hashLiteral(selfLiteral));
|
Toy_Literal result = TOY_TO_INTEGER_LITERAL(Toy_hashLiteral(selfLiteral));
|
||||||
|
|
||||||
Toy_pushLiteralArray(&interpreter->stack, result);
|
Toy_pushLiteralArray(&interpreter->stack, result);
|
||||||
@@ -77,11 +72,6 @@ static int nativeAbs(Toy_Interpreter* interpreter, Toy_LiteralArray* arguments)
|
|||||||
Toy_freeLiteral(selfLiteralIdn);
|
Toy_freeLiteral(selfLiteralIdn);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (TOY_IS_IDENTIFIER(selfLiteral)) {
|
|
||||||
Toy_freeLiteral(selfLiteral);
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!(TOY_IS_INTEGER(selfLiteral) || TOY_IS_FLOAT(selfLiteral))) {
|
if (!(TOY_IS_INTEGER(selfLiteral) || TOY_IS_FLOAT(selfLiteral))) {
|
||||||
interpreter->errorOutput("Incorrect argument type passed to abs\n");
|
interpreter->errorOutput("Incorrect argument type passed to abs\n");
|
||||||
Toy_freeLiteral(selfLiteral);
|
Toy_freeLiteral(selfLiteral);
|
||||||
@@ -120,11 +110,6 @@ static int nativeCeil(Toy_Interpreter* interpreter, Toy_LiteralArray* arguments)
|
|||||||
Toy_freeLiteral(selfLiteralIdn);
|
Toy_freeLiteral(selfLiteralIdn);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (TOY_IS_IDENTIFIER(selfLiteral)) {
|
|
||||||
Toy_freeLiteral(selfLiteral);
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!(TOY_IS_INTEGER(selfLiteral) || TOY_IS_FLOAT(selfLiteral))) {
|
if (!(TOY_IS_INTEGER(selfLiteral) || TOY_IS_FLOAT(selfLiteral))) {
|
||||||
interpreter->errorOutput("Incorrect argument type passed to ceil\n");
|
interpreter->errorOutput("Incorrect argument type passed to ceil\n");
|
||||||
Toy_freeLiteral(selfLiteral);
|
Toy_freeLiteral(selfLiteral);
|
||||||
@@ -164,11 +149,6 @@ static int nativeFloor(Toy_Interpreter* interpreter, Toy_LiteralArray* arguments
|
|||||||
Toy_freeLiteral(selfLiteralIdn);
|
Toy_freeLiteral(selfLiteralIdn);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (TOY_IS_IDENTIFIER(selfLiteral)) {
|
|
||||||
Toy_freeLiteral(selfLiteral);
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!(TOY_IS_INTEGER(selfLiteral) || TOY_IS_FLOAT(selfLiteral))) {
|
if (!(TOY_IS_INTEGER(selfLiteral) || TOY_IS_FLOAT(selfLiteral))) {
|
||||||
interpreter->errorOutput("Incorrect argument type passed to floor\n");
|
interpreter->errorOutput("Incorrect argument type passed to floor\n");
|
||||||
Toy_freeLiteral(selfLiteral);
|
Toy_freeLiteral(selfLiteral);
|
||||||
@@ -208,11 +188,6 @@ static int nativeMax(Toy_Interpreter* interpreter, Toy_LiteralArray* arguments)
|
|||||||
Toy_freeLiteral(selfLiteralIdn);
|
Toy_freeLiteral(selfLiteralIdn);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (TOY_IS_IDENTIFIER(selfLiteral)) {
|
|
||||||
Toy_freeLiteral(selfLiteral);
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!(TOY_IS_INTEGER(selfLiteral) || TOY_IS_FLOAT(selfLiteral))) {
|
if (!(TOY_IS_INTEGER(selfLiteral) || TOY_IS_FLOAT(selfLiteral))) {
|
||||||
interpreter->errorOutput("Incorrect argument type passed to max\n");
|
interpreter->errorOutput("Incorrect argument type passed to max\n");
|
||||||
Toy_freeLiteral(selfLiteral);
|
Toy_freeLiteral(selfLiteral);
|
||||||
@@ -227,11 +202,11 @@ static int nativeMax(Toy_Interpreter* interpreter, Toy_LiteralArray* arguments)
|
|||||||
|
|
||||||
//cooerce if needed
|
//cooerce if needed
|
||||||
if (TOY_IS_INTEGER(resultLiteral) && TOY_IS_FLOAT(selfLiteral)) {
|
if (TOY_IS_INTEGER(resultLiteral) && TOY_IS_FLOAT(selfLiteral)) {
|
||||||
resultLiteral = TOY_TO_FLOAT_LITERAL( TOY_AS_INTEGER(resultLiteral) );
|
resultLiteral = TOY_TO_FLOAT_LITERAL( (float)TOY_AS_INTEGER(resultLiteral) );
|
||||||
}
|
}
|
||||||
|
|
||||||
if (TOY_IS_FLOAT(resultLiteral) && TOY_IS_INTEGER(selfLiteral)) {
|
if (TOY_IS_FLOAT(resultLiteral) && TOY_IS_INTEGER(selfLiteral)) {
|
||||||
selfLiteral = TOY_TO_FLOAT_LITERAL( TOY_AS_INTEGER(selfLiteral) );
|
selfLiteral = TOY_TO_FLOAT_LITERAL( (float)TOY_AS_INTEGER(selfLiteral) );
|
||||||
}
|
}
|
||||||
|
|
||||||
//compare
|
//compare
|
||||||
@@ -269,11 +244,6 @@ static int nativeMin(Toy_Interpreter* interpreter, Toy_LiteralArray* arguments)
|
|||||||
Toy_freeLiteral(selfLiteralIdn);
|
Toy_freeLiteral(selfLiteralIdn);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (TOY_IS_IDENTIFIER(selfLiteral)) {
|
|
||||||
Toy_freeLiteral(selfLiteral);
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!(TOY_IS_INTEGER(selfLiteral) || TOY_IS_FLOAT(selfLiteral))) {
|
if (!(TOY_IS_INTEGER(selfLiteral) || TOY_IS_FLOAT(selfLiteral))) {
|
||||||
interpreter->errorOutput("Incorrect argument type passed to min\n");
|
interpreter->errorOutput("Incorrect argument type passed to min\n");
|
||||||
Toy_freeLiteral(selfLiteral);
|
Toy_freeLiteral(selfLiteral);
|
||||||
@@ -288,11 +258,11 @@ static int nativeMin(Toy_Interpreter* interpreter, Toy_LiteralArray* arguments)
|
|||||||
|
|
||||||
//cooerce if needed
|
//cooerce if needed
|
||||||
if (TOY_IS_INTEGER(resultLiteral) && TOY_IS_FLOAT(selfLiteral)) {
|
if (TOY_IS_INTEGER(resultLiteral) && TOY_IS_FLOAT(selfLiteral)) {
|
||||||
resultLiteral = TOY_TO_FLOAT_LITERAL( TOY_AS_INTEGER(resultLiteral) );
|
resultLiteral = TOY_TO_FLOAT_LITERAL( (float)TOY_AS_INTEGER(resultLiteral) );
|
||||||
}
|
}
|
||||||
|
|
||||||
if (TOY_IS_FLOAT(resultLiteral) && TOY_IS_INTEGER(selfLiteral)) {
|
if (TOY_IS_FLOAT(resultLiteral) && TOY_IS_INTEGER(selfLiteral)) {
|
||||||
selfLiteral = TOY_TO_FLOAT_LITERAL( TOY_AS_INTEGER(selfLiteral) );
|
selfLiteral = TOY_TO_FLOAT_LITERAL( (float)TOY_AS_INTEGER(selfLiteral) );
|
||||||
}
|
}
|
||||||
|
|
||||||
//compare
|
//compare
|
||||||
@@ -330,11 +300,6 @@ static int nativeRound(Toy_Interpreter* interpreter, Toy_LiteralArray* arguments
|
|||||||
Toy_freeLiteral(selfLiteralIdn);
|
Toy_freeLiteral(selfLiteralIdn);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (TOY_IS_IDENTIFIER(selfLiteral)) {
|
|
||||||
Toy_freeLiteral(selfLiteral);
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!(TOY_IS_INTEGER(selfLiteral) || TOY_IS_FLOAT(selfLiteral))) {
|
if (!(TOY_IS_INTEGER(selfLiteral) || TOY_IS_FLOAT(selfLiteral))) {
|
||||||
interpreter->errorOutput("Incorrect argument type passed to round\n");
|
interpreter->errorOutput("Incorrect argument type passed to round\n");
|
||||||
Toy_freeLiteral(selfLiteral);
|
Toy_freeLiteral(selfLiteral);
|
||||||
@@ -365,6 +330,259 @@ static int nativeRound(Toy_Interpreter* interpreter, Toy_LiteralArray* arguments
|
|||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static int nativeSign(Toy_Interpreter* interpreter, Toy_LiteralArray* arguments) {
|
||||||
|
if (arguments->count != 1) {
|
||||||
|
interpreter->errorOutput("Incorrect number of arguments to sign\n");
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
//get the self
|
||||||
|
Toy_Literal selfLiteral = Toy_popLiteralArray(arguments);
|
||||||
|
|
||||||
|
//parse to value if needed
|
||||||
|
Toy_Literal selfLiteralIdn = selfLiteral;
|
||||||
|
if (TOY_IS_IDENTIFIER(selfLiteral) && Toy_parseIdentifierToValue(interpreter, &selfLiteral)) {
|
||||||
|
Toy_freeLiteral(selfLiteralIdn);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!(TOY_IS_INTEGER(selfLiteral) || TOY_IS_FLOAT(selfLiteral))) {
|
||||||
|
interpreter->errorOutput("Incorrect argument type passed to sign\n");
|
||||||
|
Toy_freeLiteral(selfLiteral);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
Toy_Literal resultLiteral = TOY_TO_NULL_LITERAL;
|
||||||
|
|
||||||
|
if (TOY_IS_INTEGER(selfLiteral)) {
|
||||||
|
if (TOY_AS_INTEGER(selfLiteral) < 0) {
|
||||||
|
resultLiteral = TOY_TO_INTEGER_LITERAL(-1);
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
resultLiteral = TOY_TO_INTEGER_LITERAL(1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (TOY_IS_FLOAT(selfLiteral)) {
|
||||||
|
if (TOY_AS_FLOAT(selfLiteral) < 0) {
|
||||||
|
resultLiteral = TOY_TO_INTEGER_LITERAL(-1);
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
resultLiteral = TOY_TO_INTEGER_LITERAL(1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Toy_pushLiteralArray(&interpreter->stack, resultLiteral);
|
||||||
|
|
||||||
|
Toy_freeLiteral(resultLiteral);
|
||||||
|
Toy_freeLiteral(selfLiteral);
|
||||||
|
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int nativeNormalize(Toy_Interpreter* interpreter, Toy_LiteralArray* arguments) {
|
||||||
|
//NOTE: this is identical to `sign`, except it returns 0 when the argument is 0.
|
||||||
|
if (arguments->count != 1) {
|
||||||
|
interpreter->errorOutput("Incorrect number of arguments to normalize\n");
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
//get the self
|
||||||
|
Toy_Literal selfLiteral = Toy_popLiteralArray(arguments);
|
||||||
|
|
||||||
|
//parse to value if needed
|
||||||
|
Toy_Literal selfLiteralIdn = selfLiteral;
|
||||||
|
if (TOY_IS_IDENTIFIER(selfLiteral) && Toy_parseIdentifierToValue(interpreter, &selfLiteral)) {
|
||||||
|
Toy_freeLiteral(selfLiteralIdn);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!(TOY_IS_INTEGER(selfLiteral) || TOY_IS_FLOAT(selfLiteral))) {
|
||||||
|
interpreter->errorOutput("Incorrect argument type passed to normalize\n");
|
||||||
|
Toy_freeLiteral(selfLiteral);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
Toy_Literal resultLiteral = TOY_TO_NULL_LITERAL;
|
||||||
|
|
||||||
|
if (TOY_IS_INTEGER(selfLiteral)) {
|
||||||
|
if (TOY_AS_INTEGER(selfLiteral) < 0) {
|
||||||
|
resultLiteral = TOY_TO_INTEGER_LITERAL(-1);
|
||||||
|
}
|
||||||
|
else if (TOY_AS_INTEGER(selfLiteral) > 0) {
|
||||||
|
resultLiteral = TOY_TO_INTEGER_LITERAL(1);
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
resultLiteral = TOY_TO_INTEGER_LITERAL(0);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (TOY_IS_FLOAT(selfLiteral)) {
|
||||||
|
if (TOY_AS_FLOAT(selfLiteral) < 0) {
|
||||||
|
resultLiteral = TOY_TO_INTEGER_LITERAL(-1);
|
||||||
|
}
|
||||||
|
else if (TOY_AS_FLOAT(selfLiteral) > 0) {
|
||||||
|
resultLiteral = TOY_TO_INTEGER_LITERAL(1);
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
resultLiteral = TOY_TO_INTEGER_LITERAL(0);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Toy_pushLiteralArray(&interpreter->stack, resultLiteral);
|
||||||
|
|
||||||
|
Toy_freeLiteral(resultLiteral);
|
||||||
|
Toy_freeLiteral(selfLiteral);
|
||||||
|
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int nativeClamp(Toy_Interpreter* interpreter, Toy_LiteralArray* arguments) {
|
||||||
|
if (arguments->count != 3) {
|
||||||
|
interpreter->errorOutput("Incorrect number of arguments to clamp\n");
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
//get the arguments
|
||||||
|
Toy_Literal maxLiteral = Toy_popLiteralArray(arguments);
|
||||||
|
Toy_Literal minLiteral = Toy_popLiteralArray(arguments);
|
||||||
|
Toy_Literal valueLiteral = Toy_popLiteralArray(arguments);
|
||||||
|
|
||||||
|
//parse the arguments (if they're identifiers)
|
||||||
|
Toy_Literal valueLiteralIdn = valueLiteral;
|
||||||
|
if (TOY_IS_IDENTIFIER(valueLiteral) && Toy_parseIdentifierToValue(interpreter, &valueLiteral)) {
|
||||||
|
Toy_freeLiteral(valueLiteralIdn);
|
||||||
|
}
|
||||||
|
|
||||||
|
Toy_Literal minLiteralIdn = minLiteral;
|
||||||
|
if (TOY_IS_IDENTIFIER(minLiteral) && Toy_parseIdentifierToValue(interpreter, &minLiteral)) {
|
||||||
|
Toy_freeLiteral(minLiteralIdn);
|
||||||
|
}
|
||||||
|
|
||||||
|
Toy_Literal maxLiteralIdn = maxLiteral;
|
||||||
|
if (TOY_IS_IDENTIFIER(maxLiteral) && Toy_parseIdentifierToValue(interpreter, &maxLiteral)) {
|
||||||
|
Toy_freeLiteral(maxLiteralIdn);
|
||||||
|
}
|
||||||
|
|
||||||
|
//check the types
|
||||||
|
if (!(TOY_IS_INTEGER(valueLiteral) || TOY_IS_FLOAT(valueLiteral))) {
|
||||||
|
interpreter->errorOutput("Incorrect argument type passed to clamp\n");
|
||||||
|
Toy_freeLiteral(valueLiteral);
|
||||||
|
Toy_freeLiteral(minLiteral);
|
||||||
|
Toy_freeLiteral(maxLiteral);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!(TOY_IS_INTEGER(minLiteral) || TOY_IS_FLOAT(minLiteral))) {
|
||||||
|
interpreter->errorOutput("Incorrect argument type passed to clamp\n");
|
||||||
|
Toy_freeLiteral(valueLiteral);
|
||||||
|
Toy_freeLiteral(minLiteral);
|
||||||
|
Toy_freeLiteral(maxLiteral);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!(TOY_IS_INTEGER(maxLiteral) || TOY_IS_FLOAT(maxLiteral))) {
|
||||||
|
interpreter->errorOutput("Incorrect argument type passed to clamp\n");
|
||||||
|
Toy_freeLiteral(valueLiteral);
|
||||||
|
Toy_freeLiteral(minLiteral);
|
||||||
|
Toy_freeLiteral(maxLiteral);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
// cast ints to floats to handle all types of numbers
|
||||||
|
float value = TOY_IS_INTEGER(valueLiteral)? TOY_AS_INTEGER(valueLiteral) : TOY_AS_FLOAT(valueLiteral);
|
||||||
|
float min = TOY_IS_INTEGER(minLiteral)? TOY_AS_INTEGER(minLiteral) : TOY_AS_FLOAT(minLiteral);
|
||||||
|
float max = TOY_IS_INTEGER(maxLiteral)? TOY_AS_INTEGER(maxLiteral) : TOY_AS_FLOAT(maxLiteral);
|
||||||
|
|
||||||
|
//determine which literal to return (this way, we retain the original type)
|
||||||
|
if (min > value) {
|
||||||
|
Toy_pushLiteralArray(&interpreter->stack, minLiteral);
|
||||||
|
}
|
||||||
|
|
||||||
|
else if (max < value) {
|
||||||
|
Toy_pushLiteralArray(&interpreter->stack, maxLiteral);
|
||||||
|
}
|
||||||
|
|
||||||
|
else {
|
||||||
|
Toy_pushLiteralArray(&interpreter->stack, valueLiteral);
|
||||||
|
}
|
||||||
|
|
||||||
|
Toy_freeLiteral(valueLiteral);
|
||||||
|
Toy_freeLiteral(minLiteral);
|
||||||
|
Toy_freeLiteral(maxLiteral);
|
||||||
|
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int nativeLerp(Toy_Interpreter* interpreter, Toy_LiteralArray* arguments) {
|
||||||
|
if (arguments->count != 3) {
|
||||||
|
interpreter->errorOutput("Incorrect number of arguments to lerp\n");
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
//get the arguments
|
||||||
|
Toy_Literal amountLiteral = Toy_popLiteralArray(arguments);
|
||||||
|
Toy_Literal endLiteral = Toy_popLiteralArray(arguments);
|
||||||
|
Toy_Literal startLiteral = Toy_popLiteralArray(arguments);
|
||||||
|
|
||||||
|
//parse the arguments (if they're identifiers)
|
||||||
|
Toy_Literal startLiteralIdn = startLiteral;
|
||||||
|
if (TOY_IS_IDENTIFIER(startLiteral) && Toy_parseIdentifierToValue(interpreter, &startLiteral)) {
|
||||||
|
Toy_freeLiteral(startLiteralIdn);
|
||||||
|
}
|
||||||
|
|
||||||
|
Toy_Literal endLiteralIdn = endLiteral;
|
||||||
|
if (TOY_IS_IDENTIFIER(endLiteral) && Toy_parseIdentifierToValue(interpreter, &endLiteral)) {
|
||||||
|
Toy_freeLiteral(endLiteralIdn);
|
||||||
|
}
|
||||||
|
|
||||||
|
Toy_Literal amountLiteralIdn = amountLiteral;
|
||||||
|
if (TOY_IS_IDENTIFIER(amountLiteral) && Toy_parseIdentifierToValue(interpreter, &amountLiteral)) {
|
||||||
|
Toy_freeLiteral(amountLiteralIdn);
|
||||||
|
}
|
||||||
|
|
||||||
|
//check the argument types
|
||||||
|
if (!(TOY_IS_INTEGER(startLiteral) || TOY_IS_FLOAT(startLiteral))) {
|
||||||
|
interpreter->errorOutput("Incorrect argument type passed to lerp\n");
|
||||||
|
Toy_freeLiteral(startLiteral);
|
||||||
|
Toy_freeLiteral(endLiteral);
|
||||||
|
Toy_freeLiteral(amountLiteral);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!(TOY_IS_INTEGER(endLiteral) || TOY_IS_FLOAT(endLiteral))) {
|
||||||
|
interpreter->errorOutput("Incorrect argument type passed to lerp\n");
|
||||||
|
Toy_freeLiteral(startLiteral);
|
||||||
|
Toy_freeLiteral(endLiteral);
|
||||||
|
Toy_freeLiteral(amountLiteral);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!(TOY_IS_INTEGER(amountLiteral) || TOY_IS_FLOAT(amountLiteral))) {
|
||||||
|
interpreter->errorOutput("Incorrect argument type passed to lerp\n");
|
||||||
|
Toy_freeLiteral(startLiteral);
|
||||||
|
Toy_freeLiteral(endLiteral);
|
||||||
|
Toy_freeLiteral(amountLiteral);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
// cast ints to floats to handle all types of numbers
|
||||||
|
float start = TOY_IS_INTEGER(startLiteral)? TOY_AS_INTEGER(startLiteral) : TOY_AS_FLOAT(startLiteral);
|
||||||
|
float end = TOY_IS_INTEGER(endLiteral)? TOY_AS_INTEGER(endLiteral) : TOY_AS_FLOAT(endLiteral);
|
||||||
|
float amount = TOY_IS_INTEGER(amountLiteral)? TOY_AS_INTEGER(amountLiteral) : TOY_AS_FLOAT(amountLiteral);
|
||||||
|
|
||||||
|
//calculate the result
|
||||||
|
float result = start + amount * (end - start);
|
||||||
|
|
||||||
|
//return the result
|
||||||
|
Toy_Literal resultLiteral = TOY_TO_FLOAT_LITERAL(result);
|
||||||
|
Toy_pushLiteralArray(&interpreter->stack, resultLiteral);
|
||||||
|
|
||||||
|
//cleanup
|
||||||
|
Toy_freeLiteral(resultLiteral);
|
||||||
|
Toy_freeLiteral(startLiteral);
|
||||||
|
Toy_freeLiteral(endLiteral);
|
||||||
|
Toy_freeLiteral(amountLiteral);
|
||||||
|
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
static int nativeConcat(Toy_Interpreter* interpreter, Toy_LiteralArray* arguments) {
|
static int nativeConcat(Toy_Interpreter* interpreter, Toy_LiteralArray* arguments) {
|
||||||
//no arguments
|
//no arguments
|
||||||
if (arguments->count != 2) {
|
if (arguments->count != 2) {
|
||||||
@@ -387,12 +605,6 @@ static int nativeConcat(Toy_Interpreter* interpreter, Toy_LiteralArray* argument
|
|||||||
Toy_freeLiteral(otherLiteralIdn);
|
Toy_freeLiteral(otherLiteralIdn);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (TOY_IS_IDENTIFIER(selfLiteral) || TOY_IS_IDENTIFIER(otherLiteral)) {
|
|
||||||
Toy_freeLiteral(selfLiteral);
|
|
||||||
Toy_freeLiteral(otherLiteral);
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
|
|
||||||
//for each self type
|
//for each self type
|
||||||
if (TOY_IS_ARRAY(selfLiteral)) {
|
if (TOY_IS_ARRAY(selfLiteral)) {
|
||||||
if (!TOY_IS_ARRAY(otherLiteral)) {
|
if (!TOY_IS_ARRAY(otherLiteral)) {
|
||||||
@@ -503,12 +715,6 @@ static int nativeContainsKey(Toy_Interpreter* interpreter, Toy_LiteralArray* arg
|
|||||||
Toy_freeLiteral(keyLiteralIdn);
|
Toy_freeLiteral(keyLiteralIdn);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (TOY_IS_IDENTIFIER(selfLiteral) || TOY_IS_IDENTIFIER(keyLiteral)) {
|
|
||||||
Toy_freeLiteral(selfLiteral);
|
|
||||||
Toy_freeLiteral(keyLiteral);
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
|
|
||||||
//check type
|
//check type
|
||||||
if (!(/* TOY_IS_ARRAY(selfLiteral) || */ TOY_IS_DICTIONARY(selfLiteral) )) {
|
if (!(/* TOY_IS_ARRAY(selfLiteral) || */ TOY_IS_DICTIONARY(selfLiteral) )) {
|
||||||
interpreter->errorOutput("Incorrect argument type passed to containsKey\n");
|
interpreter->errorOutput("Incorrect argument type passed to containsKey\n");
|
||||||
@@ -554,12 +760,6 @@ static int nativeContainsValue(Toy_Interpreter* interpreter, Toy_LiteralArray* a
|
|||||||
Toy_freeLiteral(valueLiteralIdn);
|
Toy_freeLiteral(valueLiteralIdn);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (TOY_IS_IDENTIFIER(selfLiteral) || TOY_IS_IDENTIFIER(valueLiteral)) {
|
|
||||||
Toy_freeLiteral(selfLiteral);
|
|
||||||
Toy_freeLiteral(valueLiteral);
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
|
|
||||||
//check type
|
//check type
|
||||||
if (!( TOY_IS_ARRAY(selfLiteral) || TOY_IS_DICTIONARY(selfLiteral) )) {
|
if (!( TOY_IS_ARRAY(selfLiteral) || TOY_IS_DICTIONARY(selfLiteral) )) {
|
||||||
interpreter->errorOutput("Incorrect argument type passed to containsValue\n");
|
interpreter->errorOutput("Incorrect argument type passed to containsValue\n");
|
||||||
@@ -630,12 +830,6 @@ static int nativeEvery(Toy_Interpreter* interpreter, Toy_LiteralArray* arguments
|
|||||||
Toy_freeLiteral(fnLiteralIdn);
|
Toy_freeLiteral(fnLiteralIdn);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (TOY_IS_IDENTIFIER(selfLiteral) || TOY_IS_IDENTIFIER(fnLiteral)) {
|
|
||||||
Toy_freeLiteral(selfLiteral);
|
|
||||||
Toy_freeLiteral(fnLiteral);
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
|
|
||||||
//check type
|
//check type
|
||||||
if (!( TOY_IS_ARRAY(selfLiteral) || TOY_IS_DICTIONARY(selfLiteral) ) || !( TOY_IS_FUNCTION(fnLiteral) || TOY_IS_FUNCTION_NATIVE(fnLiteral) )) {
|
if (!( TOY_IS_ARRAY(selfLiteral) || TOY_IS_DICTIONARY(selfLiteral) ) || !( TOY_IS_FUNCTION(fnLiteral) || TOY_IS_FUNCTION_NATIVE(fnLiteral) )) {
|
||||||
interpreter->errorOutput("Incorrect argument type passed to every\n");
|
interpreter->errorOutput("Incorrect argument type passed to every\n");
|
||||||
@@ -752,12 +946,6 @@ static int nativeFilter(Toy_Interpreter* interpreter, Toy_LiteralArray* argument
|
|||||||
Toy_freeLiteral(fnLiteralIdn);
|
Toy_freeLiteral(fnLiteralIdn);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (TOY_IS_IDENTIFIER(selfLiteral) || TOY_IS_IDENTIFIER(fnLiteral)) {
|
|
||||||
Toy_freeLiteral(selfLiteral);
|
|
||||||
Toy_freeLiteral(fnLiteral);
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
|
|
||||||
//check type
|
//check type
|
||||||
if (!( TOY_IS_ARRAY(selfLiteral) || TOY_IS_DICTIONARY(selfLiteral) ) || !( TOY_IS_FUNCTION(fnLiteral) || TOY_IS_FUNCTION_NATIVE(fnLiteral) )) {
|
if (!( TOY_IS_ARRAY(selfLiteral) || TOY_IS_DICTIONARY(selfLiteral) ) || !( TOY_IS_FUNCTION(fnLiteral) || TOY_IS_FUNCTION_NATIVE(fnLiteral) )) {
|
||||||
interpreter->errorOutput("Incorrect argument type passed to filter\n");
|
interpreter->errorOutput("Incorrect argument type passed to filter\n");
|
||||||
@@ -872,12 +1060,6 @@ static int nativeForEach(Toy_Interpreter* interpreter, Toy_LiteralArray* argumen
|
|||||||
Toy_freeLiteral(fnLiteralIdn);
|
Toy_freeLiteral(fnLiteralIdn);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (TOY_IS_IDENTIFIER(selfLiteral) || TOY_IS_IDENTIFIER(fnLiteral)) {
|
|
||||||
Toy_freeLiteral(selfLiteral);
|
|
||||||
Toy_freeLiteral(fnLiteral);
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
|
|
||||||
//check type
|
//check type
|
||||||
if (!( TOY_IS_ARRAY(selfLiteral) || TOY_IS_DICTIONARY(selfLiteral) ) || !( TOY_IS_FUNCTION(fnLiteral) || TOY_IS_FUNCTION_NATIVE(fnLiteral) )) {
|
if (!( TOY_IS_ARRAY(selfLiteral) || TOY_IS_DICTIONARY(selfLiteral) ) || !( TOY_IS_FUNCTION(fnLiteral) || TOY_IS_FUNCTION_NATIVE(fnLiteral) )) {
|
||||||
interpreter->errorOutput("Incorrect argument type passed to forEach\n");
|
interpreter->errorOutput("Incorrect argument type passed to forEach\n");
|
||||||
@@ -950,11 +1132,6 @@ static int nativeGetKeys(Toy_Interpreter* interpreter, Toy_LiteralArray* argumen
|
|||||||
Toy_freeLiteral(selfLiteralIdn);
|
Toy_freeLiteral(selfLiteralIdn);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (TOY_IS_IDENTIFIER(selfLiteral)) {
|
|
||||||
Toy_freeLiteral(selfLiteral);
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
|
|
||||||
//check type
|
//check type
|
||||||
if (!TOY_IS_DICTIONARY(selfLiteral)) {
|
if (!TOY_IS_DICTIONARY(selfLiteral)) {
|
||||||
interpreter->errorOutput("Incorrect argument type passed to getKeys\n");
|
interpreter->errorOutput("Incorrect argument type passed to getKeys\n");
|
||||||
@@ -1000,11 +1177,6 @@ static int nativeGetValues(Toy_Interpreter* interpreter, Toy_LiteralArray* argum
|
|||||||
Toy_freeLiteral(selfLiteralIdn);
|
Toy_freeLiteral(selfLiteralIdn);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (TOY_IS_IDENTIFIER(selfLiteral)) {
|
|
||||||
Toy_freeLiteral(selfLiteral);
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
|
|
||||||
//check type
|
//check type
|
||||||
if (!TOY_IS_DICTIONARY(selfLiteral)) {
|
if (!TOY_IS_DICTIONARY(selfLiteral)) {
|
||||||
interpreter->errorOutput("Incorrect argument type passed to getValues\n");
|
interpreter->errorOutput("Incorrect argument type passed to getValues\n");
|
||||||
@@ -1057,12 +1229,6 @@ static int nativeIndexOf(Toy_Interpreter* interpreter, Toy_LiteralArray* argumen
|
|||||||
Toy_freeLiteral(valueLiteralIdn);
|
Toy_freeLiteral(valueLiteralIdn);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (TOY_IS_IDENTIFIER(selfLiteral) || TOY_IS_IDENTIFIER(valueLiteral)) {
|
|
||||||
Toy_freeLiteral(selfLiteral);
|
|
||||||
Toy_freeLiteral(valueLiteral);
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
|
|
||||||
//check type
|
//check type
|
||||||
if (!TOY_IS_ARRAY(selfLiteral)) {
|
if (!TOY_IS_ARRAY(selfLiteral)) {
|
||||||
interpreter->errorOutput("Incorrect argument type passed to indexOf\n");
|
interpreter->errorOutput("Incorrect argument type passed to indexOf\n");
|
||||||
@@ -1110,12 +1276,6 @@ static int nativeMap(Toy_Interpreter* interpreter, Toy_LiteralArray* arguments)
|
|||||||
Toy_freeLiteral(fnLiteralIdn);
|
Toy_freeLiteral(fnLiteralIdn);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (TOY_IS_IDENTIFIER(selfLiteral) || TOY_IS_IDENTIFIER(fnLiteral)) {
|
|
||||||
Toy_freeLiteral(selfLiteral);
|
|
||||||
Toy_freeLiteral(fnLiteral);
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
|
|
||||||
//check type
|
//check type
|
||||||
if (!( TOY_IS_ARRAY(selfLiteral) || TOY_IS_DICTIONARY(selfLiteral) ) || !( TOY_IS_FUNCTION(fnLiteral) || TOY_IS_FUNCTION_NATIVE(fnLiteral) )) {
|
if (!( TOY_IS_ARRAY(selfLiteral) || TOY_IS_DICTIONARY(selfLiteral) ) || !( TOY_IS_FUNCTION(fnLiteral) || TOY_IS_FUNCTION_NATIVE(fnLiteral) )) {
|
||||||
interpreter->errorOutput("Incorrect argument type passed to map\n");
|
interpreter->errorOutput("Incorrect argument type passed to map\n");
|
||||||
@@ -1225,13 +1385,6 @@ static int nativeReduce(Toy_Interpreter* interpreter, Toy_LiteralArray* argument
|
|||||||
Toy_freeLiteral(fnLiteralIdn);
|
Toy_freeLiteral(fnLiteralIdn);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (TOY_IS_IDENTIFIER(selfLiteral) || TOY_IS_IDENTIFIER(defaultLiteral) || TOY_IS_IDENTIFIER(fnLiteral)) {
|
|
||||||
Toy_freeLiteral(selfLiteral);
|
|
||||||
Toy_freeLiteral(defaultLiteral);
|
|
||||||
Toy_freeLiteral(fnLiteral);
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
|
|
||||||
//check type
|
//check type
|
||||||
if (!( TOY_IS_ARRAY(selfLiteral) || TOY_IS_DICTIONARY(selfLiteral) ) || !( TOY_IS_FUNCTION(fnLiteral) || TOY_IS_FUNCTION_NATIVE(fnLiteral) )) {
|
if (!( TOY_IS_ARRAY(selfLiteral) || TOY_IS_DICTIONARY(selfLiteral) ) || !( TOY_IS_FUNCTION(fnLiteral) || TOY_IS_FUNCTION_NATIVE(fnLiteral) )) {
|
||||||
interpreter->errorOutput("Incorrect argument type passed to reduce\n");
|
interpreter->errorOutput("Incorrect argument type passed to reduce\n");
|
||||||
@@ -1327,12 +1480,6 @@ static int nativeSome(Toy_Interpreter* interpreter, Toy_LiteralArray* arguments)
|
|||||||
Toy_freeLiteral(fnLiteralIdn);
|
Toy_freeLiteral(fnLiteralIdn);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (TOY_IS_IDENTIFIER(selfLiteral) || TOY_IS_IDENTIFIER(fnLiteral)) {
|
|
||||||
Toy_freeLiteral(selfLiteral);
|
|
||||||
Toy_freeLiteral(fnLiteral);
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
|
|
||||||
//check type
|
//check type
|
||||||
if (!( TOY_IS_ARRAY(selfLiteral) || TOY_IS_DICTIONARY(selfLiteral) ) || !( TOY_IS_FUNCTION(fnLiteral) || TOY_IS_FUNCTION_NATIVE(fnLiteral) )) {
|
if (!( TOY_IS_ARRAY(selfLiteral) || TOY_IS_DICTIONARY(selfLiteral) ) || !( TOY_IS_FUNCTION(fnLiteral) || TOY_IS_FUNCTION_NATIVE(fnLiteral) )) {
|
||||||
interpreter->errorOutput("Incorrect argument type passed to some\n");
|
interpreter->errorOutput("Incorrect argument type passed to some\n");
|
||||||
@@ -1471,8 +1618,13 @@ static void recursiveLiteralQuicksortUtil(Toy_Interpreter* interpreter, Toy_Lite
|
|||||||
swapLiteralsUtil(&ptr[runner], &ptr[literalCount - 1]);
|
swapLiteralsUtil(&ptr[runner], &ptr[literalCount - 1]);
|
||||||
|
|
||||||
//recurse on each end
|
//recurse on each end
|
||||||
recursiveLiteralQuicksortUtil(interpreter, &ptr[0], runner, fnCompareLiteral);
|
if (runner > 0) {
|
||||||
recursiveLiteralQuicksortUtil(interpreter, &ptr[runner + 1], literalCount - runner - 1, fnCompareLiteral);
|
recursiveLiteralQuicksortUtil(interpreter, &ptr[0], runner, fnCompareLiteral);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (runner < literalCount) {
|
||||||
|
recursiveLiteralQuicksortUtil(interpreter, &ptr[runner + 1], literalCount - runner - 1, fnCompareLiteral);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static int nativeSort(Toy_Interpreter* interpreter, Toy_LiteralArray* arguments) {
|
static int nativeSort(Toy_Interpreter* interpreter, Toy_LiteralArray* arguments) {
|
||||||
@@ -1497,12 +1649,6 @@ static int nativeSort(Toy_Interpreter* interpreter, Toy_LiteralArray* arguments)
|
|||||||
Toy_freeLiteral(fnLiteralIdn);
|
Toy_freeLiteral(fnLiteralIdn);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (TOY_IS_IDENTIFIER(selfLiteral) || TOY_IS_IDENTIFIER(fnLiteral)) {
|
|
||||||
Toy_freeLiteral(selfLiteral);
|
|
||||||
Toy_freeLiteral(fnLiteral);
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
|
|
||||||
//check type
|
//check type
|
||||||
if (!TOY_IS_ARRAY(selfLiteral) || !( TOY_IS_FUNCTION(fnLiteral) || TOY_IS_FUNCTION_NATIVE(fnLiteral) )) {
|
if (!TOY_IS_ARRAY(selfLiteral) || !( TOY_IS_FUNCTION(fnLiteral) || TOY_IS_FUNCTION_NATIVE(fnLiteral) )) {
|
||||||
interpreter->errorOutput("Incorrect argument type passed to sort\n");
|
interpreter->errorOutput("Incorrect argument type passed to sort\n");
|
||||||
@@ -1511,8 +1657,34 @@ static int nativeSort(Toy_Interpreter* interpreter, Toy_LiteralArray* arguments)
|
|||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
//BUGFIX: check if the array is already sorted
|
||||||
|
bool sorted = true;
|
||||||
|
for (int checker = 0; checker < TOY_AS_ARRAY(selfLiteral)->count - 1 && sorted; checker++) {
|
||||||
|
Toy_LiteralArray arguments;
|
||||||
|
Toy_LiteralArray returns;
|
||||||
|
|
||||||
|
Toy_initLiteralArray(&arguments);
|
||||||
|
Toy_initLiteralArray(&returns);
|
||||||
|
|
||||||
|
Toy_pushLiteralArray(&arguments, TOY_AS_ARRAY(selfLiteral)->literals[checker]);
|
||||||
|
Toy_pushLiteralArray(&arguments, TOY_AS_ARRAY(selfLiteral)->literals[checker + 1]);
|
||||||
|
|
||||||
|
Toy_callLiteralFn(interpreter, fnLiteral, &arguments, &returns);
|
||||||
|
|
||||||
|
Toy_Literal lessThan = Toy_popLiteralArray(&returns);
|
||||||
|
|
||||||
|
Toy_freeLiteralArray(&arguments);
|
||||||
|
Toy_freeLiteralArray(&returns);
|
||||||
|
|
||||||
|
if (!TOY_IS_TRUTHY(lessThan)) {
|
||||||
|
sorted = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
Toy_freeLiteral(lessThan);
|
||||||
|
}
|
||||||
|
|
||||||
//call the quicksort util
|
//call the quicksort util
|
||||||
if (TOY_IS_ARRAY(selfLiteral)) {
|
if (!sorted) {
|
||||||
recursiveLiteralQuicksortUtil(interpreter, TOY_AS_ARRAY(selfLiteral)->literals, TOY_AS_ARRAY(selfLiteral)->count, fnLiteral);
|
recursiveLiteralQuicksortUtil(interpreter, TOY_AS_ARRAY(selfLiteral)->literals, TOY_AS_ARRAY(selfLiteral)->count, fnLiteral);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -1539,11 +1711,6 @@ static int nativeToLower(Toy_Interpreter* interpreter, Toy_LiteralArray* argumen
|
|||||||
Toy_freeLiteral(selfLiteralIdn);
|
Toy_freeLiteral(selfLiteralIdn);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (TOY_IS_IDENTIFIER(selfLiteral)) {
|
|
||||||
Toy_freeLiteral(selfLiteral);
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!TOY_IS_STRING(selfLiteral)) {
|
if (!TOY_IS_STRING(selfLiteral)) {
|
||||||
interpreter->errorOutput("Incorrect argument type passed to toLower\n");
|
interpreter->errorOutput("Incorrect argument type passed to toLower\n");
|
||||||
Toy_freeLiteral(selfLiteral);
|
Toy_freeLiteral(selfLiteral);
|
||||||
@@ -1605,17 +1772,6 @@ static int nativeToString(Toy_Interpreter* interpreter, Toy_LiteralArray* argume
|
|||||||
Toy_freeLiteral(selfLiteralIdn);
|
Toy_freeLiteral(selfLiteralIdn);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (TOY_IS_IDENTIFIER(selfLiteral)) {
|
|
||||||
Toy_freeLiteral(selfLiteral);
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
|
|
||||||
//BUGFIX: probably an undefined variable
|
|
||||||
if (TOY_IS_IDENTIFIER(selfLiteral)) {
|
|
||||||
Toy_freeLiteral(selfLiteral);
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
|
|
||||||
//print it to a custom function
|
//print it to a custom function
|
||||||
Toy_printLiteralCustom(selfLiteral, toStringUtil);
|
Toy_printLiteralCustom(selfLiteral, toStringUtil);
|
||||||
|
|
||||||
@@ -1649,11 +1805,6 @@ static int nativeToUpper(Toy_Interpreter* interpreter, Toy_LiteralArray* argumen
|
|||||||
Toy_freeLiteral(selfLiteralIdn);
|
Toy_freeLiteral(selfLiteralIdn);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (TOY_IS_IDENTIFIER(selfLiteral)) {
|
|
||||||
Toy_freeLiteral(selfLiteral);
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!TOY_IS_STRING(selfLiteral)) {
|
if (!TOY_IS_STRING(selfLiteral)) {
|
||||||
interpreter->errorOutput("Incorrect argument type passed to toUpper\n");
|
interpreter->errorOutput("Incorrect argument type passed to toUpper\n");
|
||||||
Toy_freeLiteral(selfLiteral);
|
Toy_freeLiteral(selfLiteral);
|
||||||
@@ -1714,12 +1865,6 @@ static int nativeTrim(Toy_Interpreter* interpreter, Toy_LiteralArray* arguments)
|
|||||||
Toy_freeLiteral(selfLiteralIdn);
|
Toy_freeLiteral(selfLiteralIdn);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (TOY_IS_IDENTIFIER(selfLiteral) || TOY_IS_IDENTIFIER(trimCharsLiteral)) {
|
|
||||||
Toy_freeLiteral(selfLiteral);
|
|
||||||
Toy_freeLiteral(trimCharsLiteral);
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!TOY_IS_STRING(selfLiteral) || !TOY_IS_STRING(trimCharsLiteral)) {
|
if (!TOY_IS_STRING(selfLiteral) || !TOY_IS_STRING(trimCharsLiteral)) {
|
||||||
interpreter->errorOutput("Incorrect argument type passed to trim\n");
|
interpreter->errorOutput("Incorrect argument type passed to trim\n");
|
||||||
Toy_freeLiteral(trimCharsLiteral);
|
Toy_freeLiteral(trimCharsLiteral);
|
||||||
@@ -1831,12 +1976,6 @@ static int nativeTrimBegin(Toy_Interpreter* interpreter, Toy_LiteralArray* argum
|
|||||||
Toy_freeLiteral(selfLiteralIdn);
|
Toy_freeLiteral(selfLiteralIdn);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (TOY_IS_IDENTIFIER(selfLiteral) || TOY_IS_IDENTIFIER(trimCharsLiteral)) {
|
|
||||||
Toy_freeLiteral(selfLiteral);
|
|
||||||
Toy_freeLiteral(trimCharsLiteral);
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!TOY_IS_STRING(selfLiteral) || !TOY_IS_STRING(trimCharsLiteral)) {
|
if (!TOY_IS_STRING(selfLiteral) || !TOY_IS_STRING(trimCharsLiteral)) {
|
||||||
interpreter->errorOutput("Incorrect argument type passed to trimBegin\n");
|
interpreter->errorOutput("Incorrect argument type passed to trimBegin\n");
|
||||||
Toy_freeLiteral(trimCharsLiteral);
|
Toy_freeLiteral(trimCharsLiteral);
|
||||||
@@ -1925,12 +2064,6 @@ static int nativeTrimEnd(Toy_Interpreter* interpreter, Toy_LiteralArray* argumen
|
|||||||
Toy_freeLiteral(selfLiteralIdn);
|
Toy_freeLiteral(selfLiteralIdn);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (TOY_IS_IDENTIFIER(selfLiteral) || TOY_IS_IDENTIFIER(trimCharsLiteral)) {
|
|
||||||
Toy_freeLiteral(selfLiteral);
|
|
||||||
Toy_freeLiteral(trimCharsLiteral);
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!TOY_IS_STRING(selfLiteral) || !TOY_IS_STRING(trimCharsLiteral)) {
|
if (!TOY_IS_STRING(selfLiteral) || !TOY_IS_STRING(trimCharsLiteral)) {
|
||||||
interpreter->errorOutput("Incorrect argument type passed to trimEnd\n");
|
interpreter->errorOutput("Incorrect argument type passed to trimEnd\n");
|
||||||
Toy_freeLiteral(trimCharsLiteral);
|
Toy_freeLiteral(trimCharsLiteral);
|
||||||
@@ -2011,6 +2144,10 @@ int Toy_hookStandard(Toy_Interpreter* interpreter, Toy_Literal identifier, Toy_L
|
|||||||
{"max", nativeMax},
|
{"max", nativeMax},
|
||||||
{"min", nativeMin},
|
{"min", nativeMin},
|
||||||
{"round", nativeRound},
|
{"round", nativeRound},
|
||||||
|
{"sign", nativeSign},
|
||||||
|
{"normalize", nativeNormalize},
|
||||||
|
{"clamp", nativeClamp},
|
||||||
|
{"lerp", nativeLerp},
|
||||||
|
|
||||||
//compound utils
|
//compound utils
|
||||||
{"concat", nativeConcat}, //array, dictionary, string
|
{"concat", nativeConcat}, //array, dictionary, string
|
||||||
@@ -2038,7 +2175,7 @@ int Toy_hookStandard(Toy_Interpreter* interpreter, Toy_Literal identifier, Toy_L
|
|||||||
//store the library in an aliased dictionary
|
//store the library in an aliased dictionary
|
||||||
if (!TOY_IS_NULL(alias)) {
|
if (!TOY_IS_NULL(alias)) {
|
||||||
//make sure the name isn't taken
|
//make sure the name isn't taken
|
||||||
if (Toy_isDelcaredScopeVariable(interpreter->scope, alias)) {
|
if (Toy_isDeclaredScopeVariable(interpreter->scope, alias)) {
|
||||||
interpreter->errorOutput("Can't override an existing variable\n");
|
interpreter->errorOutput("Can't override an existing variable\n");
|
||||||
Toy_freeLiteral(alias);
|
Toy_freeLiteral(alias);
|
||||||
return -1;
|
return -1;
|
||||||
|
|||||||
@@ -3,4 +3,3 @@
|
|||||||
#include "toy_interpreter.h"
|
#include "toy_interpreter.h"
|
||||||
|
|
||||||
int Toy_hookStandard(Toy_Interpreter* interpreter, Toy_Literal identifier, Toy_Literal alias);
|
int Toy_hookStandard(Toy_Interpreter* interpreter, Toy_Literal identifier, Toy_Literal alias);
|
||||||
|
|
||||||
|
|||||||
@@ -1,23 +1,23 @@
|
|||||||
#include "lib_about.h"
|
#include "lib_toy_version_info.h"
|
||||||
|
|
||||||
#include "toy_memory.h"
|
#include "toy_memory.h"
|
||||||
|
|
||||||
int Toy_hookAbout(Toy_Interpreter* interpreter, Toy_Literal identifier, Toy_Literal alias) {
|
int Toy_hookToyVersionInfo(Toy_Interpreter* interpreter, Toy_Literal identifier, Toy_Literal alias) {
|
||||||
//the about keys
|
//the info keys
|
||||||
Toy_Literal majorKeyLiteral = TOY_TO_STRING_LITERAL(Toy_createRefString("major"));
|
Toy_Literal majorKeyLiteral = TOY_TO_STRING_LITERAL(Toy_createRefString("major"));
|
||||||
Toy_Literal minorKeyLiteral = TOY_TO_STRING_LITERAL(Toy_createRefString("minor"));
|
Toy_Literal minorKeyLiteral = TOY_TO_STRING_LITERAL(Toy_createRefString("minor"));
|
||||||
Toy_Literal patchKeyLiteral = TOY_TO_STRING_LITERAL(Toy_createRefString("patch"));
|
Toy_Literal patchKeyLiteral = TOY_TO_STRING_LITERAL(Toy_createRefString("patch"));
|
||||||
Toy_Literal buildKeyLiteral = TOY_TO_STRING_LITERAL(Toy_createRefString("build"));
|
Toy_Literal buildKeyLiteral = TOY_TO_STRING_LITERAL(Toy_createRefString("build"));
|
||||||
Toy_Literal authorKeyLiteral = TOY_TO_STRING_LITERAL(Toy_createRefString("author"));
|
Toy_Literal authorKeyLiteral = TOY_TO_STRING_LITERAL(Toy_createRefString("author"));
|
||||||
|
|
||||||
//the about identifiers
|
//the info identifiers
|
||||||
Toy_Literal majorIdentifierLiteral = TOY_TO_IDENTIFIER_LITERAL(Toy_createRefString("major"));
|
Toy_Literal majorIdentifierLiteral = TOY_TO_IDENTIFIER_LITERAL(Toy_createRefString("major"));
|
||||||
Toy_Literal minorIdentifierLiteral = TOY_TO_IDENTIFIER_LITERAL(Toy_createRefString("minor"));
|
Toy_Literal minorIdentifierLiteral = TOY_TO_IDENTIFIER_LITERAL(Toy_createRefString("minor"));
|
||||||
Toy_Literal patchIdentifierLiteral = TOY_TO_IDENTIFIER_LITERAL(Toy_createRefString("patch"));
|
Toy_Literal patchIdentifierLiteral = TOY_TO_IDENTIFIER_LITERAL(Toy_createRefString("patch"));
|
||||||
Toy_Literal buildIdentifierLiteral = TOY_TO_IDENTIFIER_LITERAL(Toy_createRefString("build"));
|
Toy_Literal buildIdentifierLiteral = TOY_TO_IDENTIFIER_LITERAL(Toy_createRefString("build"));
|
||||||
Toy_Literal authorIdentifierLiteral = TOY_TO_IDENTIFIER_LITERAL(Toy_createRefString("author"));
|
Toy_Literal authorIdentifierLiteral = TOY_TO_IDENTIFIER_LITERAL(Toy_createRefString("author"));
|
||||||
|
|
||||||
//the about values
|
//the info values
|
||||||
Toy_Literal majorLiteral = TOY_TO_INTEGER_LITERAL(TOY_VERSION_MAJOR);
|
Toy_Literal majorLiteral = TOY_TO_INTEGER_LITERAL(TOY_VERSION_MAJOR);
|
||||||
Toy_Literal minorLiteral = TOY_TO_INTEGER_LITERAL(TOY_VERSION_MINOR);
|
Toy_Literal minorLiteral = TOY_TO_INTEGER_LITERAL(TOY_VERSION_MINOR);
|
||||||
Toy_Literal patchLiteral = TOY_TO_INTEGER_LITERAL(TOY_VERSION_PATCH);
|
Toy_Literal patchLiteral = TOY_TO_INTEGER_LITERAL(TOY_VERSION_PATCH);
|
||||||
@@ -27,7 +27,7 @@ int Toy_hookAbout(Toy_Interpreter* interpreter, Toy_Literal identifier, Toy_Lite
|
|||||||
//store as an aliased dictionary
|
//store as an aliased dictionary
|
||||||
if (!TOY_IS_NULL(alias)) {
|
if (!TOY_IS_NULL(alias)) {
|
||||||
//make sure the name isn't taken
|
//make sure the name isn't taken
|
||||||
if (Toy_isDelcaredScopeVariable(interpreter->scope, alias)) {
|
if (Toy_isDeclaredScopeVariable(interpreter->scope, alias)) {
|
||||||
interpreter->errorOutput("Can't override an existing variable\n");
|
interpreter->errorOutput("Can't override an existing variable\n");
|
||||||
Toy_freeLiteral(alias);
|
Toy_freeLiteral(alias);
|
||||||
|
|
||||||
@@ -83,11 +83,11 @@ int Toy_hookAbout(Toy_Interpreter* interpreter, Toy_Literal identifier, Toy_Lite
|
|||||||
//store globally
|
//store globally
|
||||||
else {
|
else {
|
||||||
//make sure the names aren't taken
|
//make sure the names aren't taken
|
||||||
if (Toy_isDelcaredScopeVariable(interpreter->scope, majorKeyLiteral) ||
|
if (Toy_isDeclaredScopeVariable(interpreter->scope, majorKeyLiteral) ||
|
||||||
Toy_isDelcaredScopeVariable(interpreter->scope, minorKeyLiteral) ||
|
Toy_isDeclaredScopeVariable(interpreter->scope, minorKeyLiteral) ||
|
||||||
Toy_isDelcaredScopeVariable(interpreter->scope, patchKeyLiteral) ||
|
Toy_isDeclaredScopeVariable(interpreter->scope, patchKeyLiteral) ||
|
||||||
Toy_isDelcaredScopeVariable(interpreter->scope, buildKeyLiteral) ||
|
Toy_isDeclaredScopeVariable(interpreter->scope, buildKeyLiteral) ||
|
||||||
Toy_isDelcaredScopeVariable(interpreter->scope, authorKeyLiteral)) {
|
Toy_isDeclaredScopeVariable(interpreter->scope, authorKeyLiteral)) {
|
||||||
interpreter->errorOutput("Can't override an existing variable\n");
|
interpreter->errorOutput("Can't override an existing variable\n");
|
||||||
Toy_freeLiteral(alias);
|
Toy_freeLiteral(alias);
|
||||||
|
|
||||||
@@ -0,0 +1,5 @@
|
|||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include "toy_interpreter.h"
|
||||||
|
|
||||||
|
int Toy_hookToyVersionInfo(Toy_Interpreter* interpreter, Toy_Literal identifier, Toy_Literal alias);
|
||||||
+11
-4
@@ -1,5 +1,6 @@
|
|||||||
#include "repl_tools.h"
|
#include "repl_tools.h"
|
||||||
#include "lib_about.h"
|
#include "drive_system.h"
|
||||||
|
#include "lib_toy_version_info.h"
|
||||||
#include "lib_standard.h"
|
#include "lib_standard.h"
|
||||||
#include "lib_random.h"
|
#include "lib_random.h"
|
||||||
#include "lib_runner.h"
|
#include "lib_runner.h"
|
||||||
@@ -25,7 +26,7 @@ void repl(const char* initialInput) {
|
|||||||
Toy_initInterpreter(&interpreter);
|
Toy_initInterpreter(&interpreter);
|
||||||
|
|
||||||
//inject the libs
|
//inject the libs
|
||||||
Toy_injectNativeHook(&interpreter, "about", Toy_hookAbout);
|
Toy_injectNativeHook(&interpreter, "toy_version_info", Toy_hookToyVersionInfo);
|
||||||
Toy_injectNativeHook(&interpreter, "standard", Toy_hookStandard);
|
Toy_injectNativeHook(&interpreter, "standard", Toy_hookStandard);
|
||||||
Toy_injectNativeHook(&interpreter, "random", Toy_hookRandom);
|
Toy_injectNativeHook(&interpreter, "random", Toy_hookRandom);
|
||||||
Toy_injectNativeHook(&interpreter, "runner", Toy_hookRunner);
|
Toy_injectNativeHook(&interpreter, "runner", Toy_hookRunner);
|
||||||
@@ -193,8 +194,14 @@ int main(int argc, const char* argv[]) {
|
|||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
//run the binary file
|
if (Toy_commandLine.parseBytecodeHeader) {
|
||||||
Toy_runBinaryFile(Toy_commandLine.binaryfile);
|
//only parse the bytecode header
|
||||||
|
Toy_parseBinaryFileHeader(Toy_commandLine.binaryfile);
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
//run the binary file
|
||||||
|
Toy_runBinaryFile(Toy_commandLine.binaryfile);
|
||||||
|
}
|
||||||
|
|
||||||
//lib cleanup
|
//lib cleanup
|
||||||
Toy_freeDriveSystem();
|
Toy_freeDriveSystem();
|
||||||
|
|||||||
+58
-2
@@ -1,5 +1,5 @@
|
|||||||
#include "repl_tools.h"
|
#include "repl_tools.h"
|
||||||
#include "lib_about.h"
|
#include "lib_toy_version_info.h"
|
||||||
#include "lib_standard.h"
|
#include "lib_standard.h"
|
||||||
#include "lib_random.h"
|
#include "lib_random.h"
|
||||||
#include "lib_runner.h"
|
#include "lib_runner.h"
|
||||||
@@ -111,7 +111,7 @@ void Toy_runBinary(const unsigned char* tb, size_t size) {
|
|||||||
Toy_initInterpreter(&interpreter);
|
Toy_initInterpreter(&interpreter);
|
||||||
|
|
||||||
//inject the libs
|
//inject the libs
|
||||||
Toy_injectNativeHook(&interpreter, "about", Toy_hookAbout);
|
Toy_injectNativeHook(&interpreter, "toy_version_info", Toy_hookToyVersionInfo);
|
||||||
Toy_injectNativeHook(&interpreter, "standard", Toy_hookStandard);
|
Toy_injectNativeHook(&interpreter, "standard", Toy_hookStandard);
|
||||||
Toy_injectNativeHook(&interpreter, "random", Toy_hookRandom);
|
Toy_injectNativeHook(&interpreter, "random", Toy_hookRandom);
|
||||||
Toy_injectNativeHook(&interpreter, "runner", Toy_hookRunner);
|
Toy_injectNativeHook(&interpreter, "runner", Toy_hookRunner);
|
||||||
@@ -149,3 +149,59 @@ void Toy_runSourceFile(const char* fname) {
|
|||||||
Toy_runSource(source);
|
Toy_runSource(source);
|
||||||
free((void*)source);
|
free((void*)source);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
//utils for debugging the header
|
||||||
|
static unsigned char readByte(const unsigned char* tb, int* count) {
|
||||||
|
unsigned char ret = *(unsigned char*)(tb + *count);
|
||||||
|
*count += 1;
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
static const char* readString(const unsigned char* tb, int* count) {
|
||||||
|
const unsigned char* ret = tb + *count;
|
||||||
|
*count += (int)strlen((char*)ret) + 1; //+1 for null character
|
||||||
|
return (const char*)ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
void Toy_parseBinaryFileHeader(const char* fname) {
|
||||||
|
size_t size = 0; //not used
|
||||||
|
const unsigned char* tb = Toy_readFile(fname, &size);
|
||||||
|
if (!tb || size < 4) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
int count = 0;
|
||||||
|
|
||||||
|
//header section
|
||||||
|
const unsigned char major = readByte(tb, &count);
|
||||||
|
const unsigned char minor = readByte(tb, &count);
|
||||||
|
const unsigned char patch = readByte(tb, &count);
|
||||||
|
|
||||||
|
const char* build = readString(tb, &count);
|
||||||
|
|
||||||
|
printf("Toy Programming Language Interpreter Version %d.%d.%d (interpreter built on %s)\n\n", TOY_VERSION_MAJOR, TOY_VERSION_MINOR, TOY_VERSION_PATCH, TOY_VERSION_BUILD);
|
||||||
|
|
||||||
|
printf("Toy Programming Language Bytecode Version ");
|
||||||
|
|
||||||
|
//print the output
|
||||||
|
if (major == TOY_VERSION_MAJOR && minor == TOY_VERSION_MINOR && patch == TOY_VERSION_PATCH) {
|
||||||
|
printf("%d.%d.%d", major, minor, patch);
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
printf(TOY_CC_FONT_YELLOW TOY_CC_BACK_BLACK "%d.%d.%d" TOY_CC_RESET, major, minor, patch);
|
||||||
|
}
|
||||||
|
|
||||||
|
printf(" (interpreter built on ");
|
||||||
|
|
||||||
|
if (strncmp(build, TOY_VERSION_BUILD, strlen(TOY_VERSION_BUILD)) == 0) {
|
||||||
|
printf("%s", build);
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
printf(TOY_CC_FONT_YELLOW TOY_CC_BACK_BLACK "%s" TOY_CC_RESET, build);
|
||||||
|
}
|
||||||
|
|
||||||
|
printf(")\n");
|
||||||
|
|
||||||
|
//cleanup
|
||||||
|
free((void*)tb);
|
||||||
|
}
|
||||||
@@ -1,14 +1,84 @@
|
|||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
|
/*!
|
||||||
|
# repl_tools.h
|
||||||
|
|
||||||
|
This header provides a number of tools for compiling and running Toy, and is used primarily by the repl. However, it can also be modified and used by any host program with a little effort.
|
||||||
|
|
||||||
|
This is not a core part of Toy or a library, and as such `repl_tools.h` and `repl_tools.c` can both be found in the `repl/` folder.
|
||||||
|
!*/
|
||||||
|
|
||||||
#include "toy_common.h"
|
#include "toy_common.h"
|
||||||
|
|
||||||
|
/*!
|
||||||
|
## Defined Functions
|
||||||
|
!*/
|
||||||
|
|
||||||
|
/*!
|
||||||
|
### const char* Toy_readFile(const char* path, size_t* fileSize)
|
||||||
|
|
||||||
|
This function reads in a file, and returns it as a constant buffer. It also sets the variable pointed to by `fileSize` to the size of the given buffer.
|
||||||
|
|
||||||
|
On error, this function returns `NULL`.
|
||||||
|
!*/
|
||||||
const unsigned char* Toy_readFile(const char* path, size_t* fileSize);
|
const unsigned char* Toy_readFile(const char* path, size_t* fileSize);
|
||||||
|
|
||||||
|
/*!
|
||||||
|
### int Toy_writeFile(const char* path, const unsigned char* bytes, size_t size)
|
||||||
|
|
||||||
|
This function writes the buffer pointed to by `bytes` to a file specified by `path`. The buffer's size should be specified by `size`.
|
||||||
|
|
||||||
|
On error, this function returns a non-zero value.
|
||||||
|
!*/
|
||||||
int Toy_writeFile(const char* path, const unsigned char* bytes, size_t size);
|
int Toy_writeFile(const char* path, const unsigned char* bytes, size_t size);
|
||||||
|
|
||||||
|
/*!
|
||||||
|
### const unsigned char* Toy_compileString(const char* source, size_t* size)
|
||||||
|
|
||||||
|
This function takes a cstring of Toy source code, and returns a compiled buffer based on that source code. The variable pointed to by `size` is set to the size of the bytecode.
|
||||||
|
|
||||||
|
On error, this function returns `NULL`.
|
||||||
|
!*/
|
||||||
const unsigned char* Toy_compileString(const char* source, size_t* size);
|
const unsigned char* Toy_compileString(const char* source, size_t* size);
|
||||||
|
|
||||||
|
/*!
|
||||||
|
### void Toy_runBinary(const unsigned char* tb, size_t size)
|
||||||
|
|
||||||
|
This function takes a bytecode array of `size` size, and executes it. The libraries available to the code are currently:
|
||||||
|
|
||||||
|
* lib_toy_version_info
|
||||||
|
* lib_standard
|
||||||
|
* lib_random
|
||||||
|
* lib_runner
|
||||||
|
!*/
|
||||||
void Toy_runBinary(const unsigned char* tb, size_t size);
|
void Toy_runBinary(const unsigned char* tb, size_t size);
|
||||||
|
|
||||||
|
/*!
|
||||||
|
### void Toy_runBinaryFile(const char* fname)
|
||||||
|
|
||||||
|
This function loads in the binary file specified by `fname`, and passes it to `Toy_runBinary()`.
|
||||||
|
!*/
|
||||||
void Toy_runBinaryFile(const char* fname);
|
void Toy_runBinaryFile(const char* fname);
|
||||||
|
|
||||||
|
/*!
|
||||||
|
### void Toy_runSource(const char* source)
|
||||||
|
|
||||||
|
This function compiles the source with `Toy_compileString()`, and passes it to `Toy_runBinary()`.
|
||||||
|
!*/
|
||||||
void Toy_runSource(const char* source);
|
void Toy_runSource(const char* source);
|
||||||
|
|
||||||
|
/*!
|
||||||
|
### void Toy_runSourceFile(const char* fname)
|
||||||
|
|
||||||
|
This function loads in the file specified by `fname`, compiles it, and passes it to `Toy_runBinary()`.
|
||||||
|
!*/
|
||||||
void Toy_runSourceFile(const char* fname);
|
void Toy_runSourceFile(const char* fname);
|
||||||
|
|
||||||
|
/*!
|
||||||
|
### void Toy_parseBinaryFileHeader(const char* fname)
|
||||||
|
|
||||||
|
This function parses the header information stored within the bytecode file `fname`.
|
||||||
|
|
||||||
|
This is only used for debugging and validation purposes.
|
||||||
|
!*/
|
||||||
|
void Toy_parseBinaryFileHeader(const char* fname);
|
||||||
@@ -1,4 +0,0 @@
|
|||||||
var s = "42";
|
|
||||||
var t = "69";
|
|
||||||
|
|
||||||
print int (s + t) - 1;
|
|
||||||
+1
-1
@@ -4,7 +4,7 @@ fn fib(n : int) {
|
|||||||
return fib(n-1) + fib(n-2);
|
return fib(n-1) + fib(n-2);
|
||||||
}
|
}
|
||||||
|
|
||||||
for (var i = 0; i < 20; i++) {
|
for (var i = 0; i <= 35; i++) {
|
||||||
var res = fib(i);
|
var res = fib(i);
|
||||||
print string i + ": " + string res;
|
print string i + ": " + string res;
|
||||||
}
|
}
|
||||||
+3
-3
@@ -11,14 +11,14 @@ OBJ = $(addprefix $(ODIR)/,$(SRC:.c=.o))
|
|||||||
OUTNAME=toy
|
OUTNAME=toy
|
||||||
|
|
||||||
ifeq ($(findstring CYGWIN, $(shell uname)),CYGWIN)
|
ifeq ($(findstring CYGWIN, $(shell uname)),CYGWIN)
|
||||||
LIBLINE =-Wl,--out-implib=../$(TOY_OUTDIR)/lib$(OUTNAME).dll.a -Wl,--export-all-symbols -Wl,--enable-auto-import -Wl,--whole-archive $(OBJ) -Wl,--no-whole-archive
|
LIBLINE=-Wl,-rpath,. -Wl,--out-implib=../$(TOY_OUTDIR)/lib$(OUTNAME).dll.a -Wl,--export-all-symbols -Wl,--enable-auto-import -Wl,--whole-archive $(OBJ) -Wl,--no-whole-archive
|
||||||
OUT=../$(TOY_OUTDIR)/$(OUTNAME).dll
|
OUT=../$(TOY_OUTDIR)/$(OUTNAME).dll
|
||||||
else ifeq ($(shell uname),Linux)
|
else ifeq ($(shell uname),Linux)
|
||||||
LIBLINE=-Wl,--out-implib=../$(TOY_OUTDIR)/lib$(OUTNAME).a -Wl,--whole-archive $(OBJ) -Wl,--no-whole-archive
|
LIBLINE=-Wl,-rpath,. -Wl,--out-implib=../$(TOY_OUTDIR)/lib$(OUTNAME).a -Wl,--whole-archive $(OBJ) -Wl,--no-whole-archive
|
||||||
OUT=../$(TOY_OUTDIR)/lib$(OUTNAME).so
|
OUT=../$(TOY_OUTDIR)/lib$(OUTNAME).so
|
||||||
CFLAGS += -fPIC
|
CFLAGS += -fPIC
|
||||||
else ifeq ($(OS),Windows_NT)
|
else ifeq ($(OS),Windows_NT)
|
||||||
LIBLINE =-Wl,--out-implib=../$(TOY_OUTDIR)/lib$(OUTNAME).dll.a -Wl,--export-all-symbols -Wl,--enable-auto-import -Wl,--whole-archive $(OBJ) -Wl,--no-whole-archive
|
LIBLINE=-Wl,-rpath,. -Wl,--out-implib=../$(TOY_OUTDIR)/lib$(OUTNAME).dll.a -Wl,--export-all-symbols -Wl,--enable-auto-import -Wl,--whole-archive $(OBJ) -Wl,--no-whole-archive
|
||||||
OUT=../$(TOY_OUTDIR)/$(OUTNAME).dll
|
OUT=../$(TOY_OUTDIR)/$(OUTNAME).dll
|
||||||
else ifeq ($(shell uname),Darwin)
|
else ifeq ($(shell uname),Darwin)
|
||||||
LIBLINE = $(OBJ)
|
LIBLINE = $(OBJ)
|
||||||
|
|||||||
+46
-27
@@ -1,68 +1,87 @@
|
|||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
/* toy.h - A Toy Programming Language
|
/*!
|
||||||
|
# toy.h - A Toy Programming Language
|
||||||
|
|
||||||
If you're looking how to use Toy directly, try https://toylang.com/
|
If you're looking how to use Toy directly, try https://toylang.com/
|
||||||
Otherwise, these headers may help learn how Toy works internally.
|
Otherwise, this header may help learn how Toy works internally.
|
||||||
|
!*/
|
||||||
|
|
||||||
*/
|
/*!
|
||||||
|
## Utilities
|
||||||
|
|
||||||
/* utilities - these define a bunch of useful macros based on platform.
|
These headers define a bunch of useful macros, based on what platform you build for.
|
||||||
|
|
||||||
The most important one is `TOY_API`, which highlights functions intended for the end user.
|
The most important macro is `TOY_API`, which specifies functions intended for the end user.
|
||||||
|
|
||||||
*/
|
* [toy_common.h](toy_common_h.md)
|
||||||
|
* [toy_console_colors.h](toy_console_colors_h.md)
|
||||||
|
* [toy_memory.h](toy_memory_h.md)
|
||||||
|
!*/
|
||||||
|
|
||||||
#include "toy_common.h"
|
#include "toy_common.h"
|
||||||
#include "toy_console_colors.h"
|
#include "toy_console_colors.h"
|
||||||
#include "toy_memory.h"
|
#include "toy_memory.h"
|
||||||
#include "toy_drive_system.h"
|
|
||||||
|
|
||||||
/* core pipeline - from source to execution
|
/*!
|
||||||
|
## Core Pipeline
|
||||||
|
|
||||||
Each step is as follows:
|
From source to execution, each step is as follows:
|
||||||
|
|
||||||
|
```
|
||||||
source -> lexer -> token
|
source -> lexer -> token
|
||||||
token -> parser -> AST
|
token -> parser -> AST
|
||||||
AST -> compiler -> bytecode
|
AST -> compiler -> bytecode
|
||||||
bytecode -> interpreter -> result
|
bytecode -> interpreter -> result
|
||||||
|
```
|
||||||
|
|
||||||
I should note that the parser -> compiler phase is actually made up of two steps - the write step
|
I should note that the parser -> compiler phase is actually made up of two steps - the write step and the collate step. See `Toy_compileString()` in `repl/repl_tools.c` for an example of how to compile properly.
|
||||||
and the collate step. See `Toy_compileString()` in `repl/repl_tools.c` for an example of how to compile
|
|
||||||
properly.
|
|
||||||
|
|
||||||
*/
|
* [toy_lexer.h](toy_lexer_h.md)
|
||||||
|
* [toy_parser.h](toy_parser_h.md)
|
||||||
|
* [toy_compiler.h](toy_compiler_h.md)
|
||||||
|
* [toy_interpreter.h](toy_interpreter_h.md)
|
||||||
|
!*/
|
||||||
|
|
||||||
#include "toy_lexer.h"
|
#include "toy_lexer.h"
|
||||||
#include "toy_parser.h"
|
#include "toy_parser.h"
|
||||||
#include "toy_compiler.h"
|
#include "toy_compiler.h"
|
||||||
#include "toy_interpreter.h"
|
#include "toy_interpreter.h"
|
||||||
|
|
||||||
/* building block structures - the basic units of operation
|
/*!
|
||||||
|
## Building Block Structures
|
||||||
|
|
||||||
Literals represent any value within the language, including some internal ones that you never see.
|
Literals represent any value within the language, including some internal ones that you never see.
|
||||||
Literal Arrays are literally arrays within memory, and are the most heavily used structure in Toy.
|
|
||||||
Literal Dictionaries are unordered key-value hashmaps, that use a running strategy for collisions.
|
|
||||||
|
|
||||||
*/
|
Literal arrays are contiguous arrays within memory, and are the most heavily used structure in Toy.
|
||||||
|
|
||||||
|
Literal dictionaries are unordered key-value hashmaps, that use a running strategy for collisions.
|
||||||
|
|
||||||
|
* [toy_literal.h](toy_literal_h.md)
|
||||||
|
* [toy_literal_array.h](toy_literal_array_h.md)
|
||||||
|
* [toy_literal_dictionary.h](toy_literal_dictionary_h.md)
|
||||||
|
!*/
|
||||||
|
|
||||||
#include "toy_literal.h"
|
#include "toy_literal.h"
|
||||||
#include "toy_literal_array.h"
|
#include "toy_literal_array.h"
|
||||||
#include "toy_literal_dictionary.h"
|
#include "toy_literal_dictionary.h"
|
||||||
|
|
||||||
/* other components - you probably won't use these directly, but they're a good learning opportunity.
|
/*!
|
||||||
|
## Other Components
|
||||||
|
|
||||||
`Toy_Scope` holds the variables of a specific scope within Toy - be it a script, a function, a block, etc.
|
You probably won't use these directly, but they're a good learning opportunity.
|
||||||
Scopes are also where the type system lives at runtime. They use identifier literals as keys, exclusively.
|
|
||||||
|
|
||||||
`Toy_RefString` is a utility class that wraps traditional C strings, making them less memory intensive and
|
`Toy_Scope` holds the variables of a specific scope within Toy - be it a script, a function, a block, etc. Scopes are also where the type system lives at runtime. They use identifier literals as keys, exclusively.
|
||||||
faster to copy and move. In reality, since strings are considered immutable, multiple variables can point
|
|
||||||
to the same string to save memory, and you can just create a new one of these vars pointing to the original
|
|
||||||
rather than copying entirely for a speed boost. This module has it's own memory allocator system that is
|
|
||||||
plugged into the main memory allocator.
|
|
||||||
|
|
||||||
*/
|
`Toy_RefString` is a utility class that wraps traditional C strings, making them less memory intensive and faster to copy and move. In reality, since strings are considered immutable, multiple variables can point to the same string to save memory, and you can just create a new one of these vars pointing to the original rather than copying entirely for a speed boost. This module has it's own memory allocator system that is plugged into the main memory allocator.
|
||||||
|
|
||||||
|
`Toy_RefFunction` acts similarly to `Toy_RefString`, but instead operates on function bytecode.
|
||||||
|
|
||||||
|
* [toy_scope.h](toy_scope_h.md)
|
||||||
|
* [toy_refstring.h](toy_refstring_h.md)
|
||||||
|
* [toy_reffunction.h](toy_reffunction_h.md)
|
||||||
|
!*/
|
||||||
|
|
||||||
#include "toy_scope.h"
|
#include "toy_scope.h"
|
||||||
#include "toy_refstring.h"
|
#include "toy_refstring.h"
|
||||||
|
#include "toy_reffunction.h"
|
||||||
|
|||||||
@@ -270,4 +270,5 @@ union Toy_private_node {
|
|||||||
Toy_NodeImport import;
|
Toy_NodeImport import;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
//see toy_parser.h for more documentation on this function
|
||||||
TOY_API void Toy_freeASTNode(Toy_ASTNode* node);
|
TOY_API void Toy_freeASTNode(Toy_ASTNode* node);
|
||||||
|
|||||||
+17
-3
@@ -15,7 +15,11 @@ STATIC_ASSERT(sizeof(unsigned char) == 1);
|
|||||||
STATIC_ASSERT(sizeof(unsigned short) == 2);
|
STATIC_ASSERT(sizeof(unsigned short) == 2);
|
||||||
STATIC_ASSERT(sizeof(unsigned int) == 4);
|
STATIC_ASSERT(sizeof(unsigned int) == 4);
|
||||||
|
|
||||||
#ifndef TOY_EXPORT
|
static const char* build = __DATE__ " " __TIME__;
|
||||||
|
|
||||||
|
const char* Toy_private_version_build() {
|
||||||
|
return build;
|
||||||
|
}
|
||||||
|
|
||||||
//declare the singleton with default values
|
//declare the singleton with default values
|
||||||
Toy_CommandLine Toy_commandLine = {
|
Toy_CommandLine Toy_commandLine = {
|
||||||
@@ -29,6 +33,7 @@ Toy_CommandLine Toy_commandLine = {
|
|||||||
.source = NULL,
|
.source = NULL,
|
||||||
.initialfile = NULL,
|
.initialfile = NULL,
|
||||||
.enablePrintNewline = true,
|
.enablePrintNewline = true,
|
||||||
|
.parseBytecodeHeader = false,
|
||||||
.verbose = false
|
.verbose = false
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -89,6 +94,16 @@ void Toy_initCommandLine(int argc, const char* argv[]) {
|
|||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (!strcmp(argv[i], "-p")) {
|
||||||
|
Toy_commandLine.parseBytecodeHeader = true;
|
||||||
|
|
||||||
|
if (Toy_commandLine.binaryfile) {
|
||||||
|
Toy_commandLine.error = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
if (!strcmp(argv[i], "-n")) {
|
if (!strcmp(argv[i], "-n")) {
|
||||||
Toy_commandLine.enablePrintNewline = false;
|
Toy_commandLine.enablePrintNewline = false;
|
||||||
Toy_commandLine.error = false;
|
Toy_commandLine.error = false;
|
||||||
@@ -124,6 +139,7 @@ void Toy_helpCommandLine(int argc, const char* argv[]) {
|
|||||||
printf(" -c, --compile filename\tParse and compile the specified source file into an output file.\n");
|
printf(" -c, --compile filename\tParse and compile the specified source file into an output file.\n");
|
||||||
printf(" -o, --output outfile\t\tName of the output file built with --compile (default: out.tb).\n");
|
printf(" -o, --output outfile\t\tName of the output file built with --compile (default: out.tb).\n");
|
||||||
printf(" -t, --initial filename\tStart the repl as normal, after first running the given file.\n");
|
printf(" -t, --initial filename\tStart the repl as normal, after first running the given file.\n");
|
||||||
|
printf(" -p\t\t\t\tParse the given bytecode's header, then exit (requires file.tb).\n");
|
||||||
printf(" -n\t\t\t\tDisable the newline character at the end of the print statement.\n");
|
printf(" -n\t\t\t\tDisable the newline character at the end of the print statement.\n");
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -136,5 +152,3 @@ void Toy_copyrightCommandLine(int argc, const char* argv[]) {
|
|||||||
printf("2. Altered source versions must be plainly marked as such, and must not be misrepresented as being the original software.\n\n");
|
printf("2. Altered source versions must be plainly marked as such, and must not be misrepresented as being the original software.\n\n");
|
||||||
printf("3. This notice may not be removed or altered from any source distribution.\n\n");
|
printf("3. This notice may not be removed or altered from any source distribution.\n\n");
|
||||||
}
|
}
|
||||||
|
|
||||||
#endif
|
|
||||||
|
|||||||
+60
-8
@@ -1,15 +1,23 @@
|
|||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
|
/*!
|
||||||
|
# toy_common.h
|
||||||
|
|
||||||
|
This file is generally included in most header files within Toy, as it is where the TOY_API macro is defined. It also has some utilities intended for use only by the repl.
|
||||||
|
|
||||||
|
## Defined Macros
|
||||||
|
!*/
|
||||||
|
|
||||||
#include <stdbool.h>
|
#include <stdbool.h>
|
||||||
#include <stddef.h>
|
#include <stddef.h>
|
||||||
#include <stdint.h>
|
#include <stdint.h>
|
||||||
|
|
||||||
#define TOY_VERSION_MAJOR 1
|
/*!
|
||||||
#define TOY_VERSION_MINOR 1
|
### TOY_API
|
||||||
#define TOY_VERSION_PATCH 2
|
|
||||||
#define TOY_VERSION_BUILD __DATE__ " " __TIME__
|
This definition of this macro is platform-dependant, and used to enable cross-platform compilation of shared and static libraries.
|
||||||
|
!*/
|
||||||
|
|
||||||
//platform/compiler-specific instructions
|
|
||||||
#if defined(__linux__) || defined(__MINGW32__) || defined(__GNUC__)
|
#if defined(__linux__) || defined(__MINGW32__) || defined(__GNUC__)
|
||||||
|
|
||||||
#define TOY_API extern
|
#define TOY_API extern
|
||||||
@@ -28,7 +36,52 @@
|
|||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifndef TOY_EXPORT
|
/*!
|
||||||
|
### TOY_VERSION_MAJOR
|
||||||
|
|
||||||
|
The current major version of Toy. This value is embedded into the bytecode, and the interpreter will refuse to run bytecode with a major version that does not match it’s own version.
|
||||||
|
|
||||||
|
This value MUST fit into an unsigned char.
|
||||||
|
!*/
|
||||||
|
|
||||||
|
#define TOY_VERSION_MAJOR 1
|
||||||
|
|
||||||
|
/*!
|
||||||
|
### TOY_VERSION_MINOR
|
||||||
|
|
||||||
|
The current minor version of Toy. This value is embedded into the bytecode, and the interpreter will refuse to run bytecode with a minor version that is greater than its own minor version.
|
||||||
|
|
||||||
|
This value MUST fit into an unsigned char.
|
||||||
|
!*/
|
||||||
|
|
||||||
|
#define TOY_VERSION_MINOR 2
|
||||||
|
|
||||||
|
/*!
|
||||||
|
### TOY_VERSION_PATCH
|
||||||
|
|
||||||
|
The current patch version of Toy. This value is embedded into the bytecode.
|
||||||
|
|
||||||
|
This value MUST fit into an unsigned char.
|
||||||
|
!*/
|
||||||
|
|
||||||
|
#define TOY_VERSION_PATCH 1
|
||||||
|
|
||||||
|
/*!
|
||||||
|
### TOY_VERSION_BUILD
|
||||||
|
|
||||||
|
The current build version of Toy. This value is embedded into the bytecode.
|
||||||
|
|
||||||
|
This evaluates to a c-string, which contains build information such as compilation date and time of the interpreter. When in verbose mode, the compiler will display a warning if the build version of the bytecode does not match the build version of the interpreter.
|
||||||
|
|
||||||
|
This macro may also be used to store additonal information about forks of the Toy codebase.
|
||||||
|
!*/
|
||||||
|
|
||||||
|
#define TOY_VERSION_BUILD Toy_private_version_build()
|
||||||
|
TOY_API const char* Toy_private_version_build();
|
||||||
|
|
||||||
|
/*
|
||||||
|
The following code is intended only for use within the repl.
|
||||||
|
*/
|
||||||
|
|
||||||
//for processing the command line arguments in the repl
|
//for processing the command line arguments in the repl
|
||||||
typedef struct {
|
typedef struct {
|
||||||
@@ -42,6 +95,7 @@ typedef struct {
|
|||||||
char* source;
|
char* source;
|
||||||
char* initialfile;
|
char* initialfile;
|
||||||
bool enablePrintNewline;
|
bool enablePrintNewline;
|
||||||
|
bool parseBytecodeHeader;
|
||||||
bool verbose;
|
bool verbose;
|
||||||
} Toy_CommandLine;
|
} Toy_CommandLine;
|
||||||
|
|
||||||
@@ -53,5 +107,3 @@ TOY_API void Toy_initCommandLine(int argc, const char* argv[]);
|
|||||||
TOY_API void Toy_usageCommandLine(int argc, const char* argv[]);
|
TOY_API void Toy_usageCommandLine(int argc, const char* argv[]);
|
||||||
TOY_API void Toy_helpCommandLine(int argc, const char* argv[]);
|
TOY_API void Toy_helpCommandLine(int argc, const char* argv[]);
|
||||||
TOY_API void Toy_copyrightCommandLine(int argc, const char* argv[]);
|
TOY_API void Toy_copyrightCommandLine(int argc, const char* argv[]);
|
||||||
|
|
||||||
#endif
|
|
||||||
+88
-13
@@ -48,7 +48,7 @@ static int writeLiteralTypeToCache(Toy_LiteralArray* literalCache, Toy_Literal l
|
|||||||
}
|
}
|
||||||
|
|
||||||
//optimisation: check if exactly this literal array exists
|
//optimisation: check if exactly this literal array exists
|
||||||
int index = Toy_findLiteralIndex(literalCache, literal);
|
int index = Toy_private_findLiteralIndex(literalCache, literal);
|
||||||
if (index < 0) {
|
if (index < 0) {
|
||||||
index = Toy_pushLiteralArray(literalCache, literal);
|
index = Toy_pushLiteralArray(literalCache, literal);
|
||||||
}
|
}
|
||||||
@@ -74,7 +74,7 @@ static int writeNodeCompoundToCache(Toy_Compiler* compiler, Toy_ASTNode* node) {
|
|||||||
switch(node->compound.nodes[i].pair.left->type) {
|
switch(node->compound.nodes[i].pair.left->type) {
|
||||||
case TOY_AST_NODE_LITERAL: {
|
case TOY_AST_NODE_LITERAL: {
|
||||||
//keys are literals
|
//keys are literals
|
||||||
int key = Toy_findLiteralIndex(&compiler->literalCache, node->compound.nodes[i].pair.left->atomic.literal);
|
int key = Toy_private_findLiteralIndex(&compiler->literalCache, node->compound.nodes[i].pair.left->atomic.literal);
|
||||||
if (key < 0) {
|
if (key < 0) {
|
||||||
key = Toy_pushLiteralArray(&compiler->literalCache, node->compound.nodes[i].pair.left->atomic.literal);
|
key = Toy_pushLiteralArray(&compiler->literalCache, node->compound.nodes[i].pair.left->atomic.literal);
|
||||||
}
|
}
|
||||||
@@ -103,7 +103,7 @@ static int writeNodeCompoundToCache(Toy_Compiler* compiler, Toy_ASTNode* node) {
|
|||||||
switch(node->compound.nodes[i].pair.right->type) {
|
switch(node->compound.nodes[i].pair.right->type) {
|
||||||
case TOY_AST_NODE_LITERAL: {
|
case TOY_AST_NODE_LITERAL: {
|
||||||
//values are literals
|
//values are literals
|
||||||
int val = Toy_findLiteralIndex(&compiler->literalCache, node->compound.nodes[i].pair.right->atomic.literal);
|
int val = Toy_private_findLiteralIndex(&compiler->literalCache, node->compound.nodes[i].pair.right->atomic.literal);
|
||||||
if (val < 0) {
|
if (val < 0) {
|
||||||
val = Toy_pushLiteralArray(&compiler->literalCache, node->compound.nodes[i].pair.right->atomic.literal);
|
val = Toy_pushLiteralArray(&compiler->literalCache, node->compound.nodes[i].pair.right->atomic.literal);
|
||||||
}
|
}
|
||||||
@@ -142,7 +142,7 @@ static int writeNodeCompoundToCache(Toy_Compiler* compiler, Toy_ASTNode* node) {
|
|||||||
switch(node->compound.nodes[i].type) {
|
switch(node->compound.nodes[i].type) {
|
||||||
case TOY_AST_NODE_LITERAL: {
|
case TOY_AST_NODE_LITERAL: {
|
||||||
//values
|
//values
|
||||||
int val = Toy_findLiteralIndex(&compiler->literalCache, node->compound.nodes[i].atomic.literal);
|
int val = Toy_private_findLiteralIndex(&compiler->literalCache, node->compound.nodes[i].atomic.literal);
|
||||||
if (val < 0) {
|
if (val < 0) {
|
||||||
val = Toy_pushLiteralArray(&compiler->literalCache, node->compound.nodes[i].atomic.literal);
|
val = Toy_pushLiteralArray(&compiler->literalCache, node->compound.nodes[i].atomic.literal);
|
||||||
}
|
}
|
||||||
@@ -230,7 +230,7 @@ static int writeNodeCollectionToCache(Toy_Compiler* compiler, Toy_ASTNode* node)
|
|||||||
|
|
||||||
static int writeLiteralToCompiler(Toy_Compiler* compiler, Toy_Literal literal) {
|
static int writeLiteralToCompiler(Toy_Compiler* compiler, Toy_Literal literal) {
|
||||||
//get the index
|
//get the index
|
||||||
int index = Toy_findLiteralIndex(&compiler->literalCache, literal);
|
int index = Toy_private_findLiteralIndex(&compiler->literalCache, literal);
|
||||||
|
|
||||||
if (index < 0) {
|
if (index < 0) {
|
||||||
if (TOY_IS_TYPE(literal)) {
|
if (TOY_IS_TYPE(literal)) {
|
||||||
@@ -259,6 +259,81 @@ static int writeLiteralToCompiler(Toy_Compiler* compiler, Toy_Literal literal) {
|
|||||||
return index;
|
return index;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
//BUGFIX: check to see if this node lies within this tree
|
||||||
|
bool checkNodeInTree(Toy_ASTNode* tree, Toy_ASTNode* node) {
|
||||||
|
if (tree == node) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (tree == NULL) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
switch(tree->type) {
|
||||||
|
case TOY_AST_NODE_UNARY:
|
||||||
|
return checkNodeInTree(tree->unary.child, node);
|
||||||
|
|
||||||
|
case TOY_AST_NODE_BINARY:
|
||||||
|
return checkNodeInTree(tree->binary.left, node) || checkNodeInTree(tree->binary.right, node);
|
||||||
|
|
||||||
|
case TOY_AST_NODE_TERNARY:
|
||||||
|
return checkNodeInTree(tree->ternary.condition, node) || checkNodeInTree(tree->ternary.thenPath, node) || checkNodeInTree(tree->ternary.elsePath, node);
|
||||||
|
|
||||||
|
case TOY_AST_NODE_GROUPING:
|
||||||
|
return checkNodeInTree(tree->grouping.child, node);
|
||||||
|
|
||||||
|
case TOY_AST_NODE_BLOCK:
|
||||||
|
return checkNodeInTree(tree->block.nodes, node);
|
||||||
|
|
||||||
|
case TOY_AST_NODE_COMPOUND:
|
||||||
|
return checkNodeInTree(tree->compound.nodes, node);
|
||||||
|
|
||||||
|
case TOY_AST_NODE_PAIR:
|
||||||
|
return checkNodeInTree(tree->pair.left, node) || checkNodeInTree(tree->pair.right, node);
|
||||||
|
|
||||||
|
case TOY_AST_NODE_INDEX:
|
||||||
|
return checkNodeInTree(tree->index.first, node) || checkNodeInTree(tree->index.second, node) || checkNodeInTree(tree->index.third, node);
|
||||||
|
|
||||||
|
case TOY_AST_NODE_VAR_DECL:
|
||||||
|
return checkNodeInTree(tree->varDecl.expression, node);
|
||||||
|
|
||||||
|
case TOY_AST_NODE_FN_COLLECTION:
|
||||||
|
return checkNodeInTree(tree->fnCollection.nodes, node);
|
||||||
|
|
||||||
|
case TOY_AST_NODE_FN_DECL:
|
||||||
|
return checkNodeInTree(tree->fnDecl.arguments, node) || checkNodeInTree(tree->fnDecl.returns, node) || checkNodeInTree(tree->fnDecl.block, node);
|
||||||
|
|
||||||
|
case TOY_AST_NODE_FN_CALL:
|
||||||
|
return checkNodeInTree(tree->fnCall.arguments, node);
|
||||||
|
|
||||||
|
case TOY_AST_NODE_FN_RETURN:
|
||||||
|
return checkNodeInTree(tree->returns.returns, node);
|
||||||
|
|
||||||
|
case TOY_AST_NODE_IF:
|
||||||
|
return checkNodeInTree(tree->pathIf.condition, node) || checkNodeInTree(tree->pathIf.thenPath, node) || checkNodeInTree(tree->pathIf.elsePath, node);
|
||||||
|
|
||||||
|
case TOY_AST_NODE_WHILE:
|
||||||
|
return checkNodeInTree(tree->pathWhile.condition, node) || checkNodeInTree(tree->pathWhile.thenPath, node);
|
||||||
|
|
||||||
|
case TOY_AST_NODE_FOR:
|
||||||
|
return checkNodeInTree(tree->pathFor.preClause, node) || checkNodeInTree(tree->pathFor.condition, node) || checkNodeInTree(tree->pathFor.postClause, node) || checkNodeInTree(tree->pathFor.thenPath, node);
|
||||||
|
|
||||||
|
case TOY_AST_NODE_ERROR:
|
||||||
|
case TOY_AST_NODE_LITERAL:
|
||||||
|
case TOY_AST_NODE_BREAK:
|
||||||
|
case TOY_AST_NODE_CONTINUE:
|
||||||
|
case TOY_AST_NODE_PREFIX_INCREMENT:
|
||||||
|
case TOY_AST_NODE_PREFIX_DECREMENT:
|
||||||
|
case TOY_AST_NODE_POSTFIX_INCREMENT:
|
||||||
|
case TOY_AST_NODE_POSTFIX_DECREMENT:
|
||||||
|
case TOY_AST_NODE_IMPORT:
|
||||||
|
case TOY_AST_NODE_PASS:
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
//NOTE: jumpOfsets are included, because function arg and return indexes are embedded in the code body i.e. need to include their sizes in the jump
|
//NOTE: jumpOfsets are included, because function arg and return indexes are embedded in the code body i.e. need to include their sizes in the jump
|
||||||
//NOTE: rootNode should NOT include groupings and blocks
|
//NOTE: rootNode should NOT include groupings and blocks
|
||||||
static Toy_Opcode Toy_writeCompilerWithJumps(Toy_Compiler* compiler, Toy_ASTNode* node, void* breakAddressesPtr, void* continueAddressesPtr, int jumpOffsets, Toy_ASTNode* rootNode) {
|
static Toy_Opcode Toy_writeCompilerWithJumps(Toy_Compiler* compiler, Toy_ASTNode* node, void* breakAddressesPtr, void* continueAddressesPtr, int jumpOffsets, Toy_ASTNode* rootNode) {
|
||||||
@@ -322,7 +397,8 @@ static Toy_Opcode Toy_writeCompilerWithJumps(Toy_Compiler* compiler, Toy_ASTNode
|
|||||||
//return this if...
|
//return this if...
|
||||||
Toy_Opcode ret = Toy_writeCompilerWithJumps(compiler, node->binary.right, breakAddressesPtr, continueAddressesPtr, jumpOffsets, rootNode);
|
Toy_Opcode ret = Toy_writeCompilerWithJumps(compiler, node->binary.right, breakAddressesPtr, continueAddressesPtr, jumpOffsets, rootNode);
|
||||||
|
|
||||||
if (node->binary.opcode == TOY_OP_INDEX && rootNode->type == TOY_AST_NODE_BINARY && (rootNode->binary.opcode >= TOY_OP_VAR_ASSIGN && rootNode->binary.opcode <= TOY_OP_VAR_MODULO_ASSIGN) && rootNode->binary.right != node) { //range-based check for assignment type; make sure the index is on the left of the assignment symbol
|
//range-based check for assignment type; make sure the index is on the left of the assignment symbol
|
||||||
|
if (node->binary.opcode == TOY_OP_INDEX && rootNode->type == TOY_AST_NODE_BINARY && (rootNode->binary.opcode >= TOY_OP_VAR_ASSIGN && rootNode->binary.opcode <= TOY_OP_VAR_MODULO_ASSIGN) && !checkNodeInTree(rootNode->binary.right, node)) {
|
||||||
return TOY_OP_INDEX_ASSIGN_INTERMEDIATE;
|
return TOY_OP_INDEX_ASSIGN_INTERMEDIATE;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -474,7 +550,7 @@ static Toy_Opcode Toy_writeCompilerWithJumps(Toy_Compiler* compiler, Toy_ASTNode
|
|||||||
}
|
}
|
||||||
|
|
||||||
//write each piece of the declaration to the bytecode
|
//write each piece of the declaration to the bytecode
|
||||||
int identifierIndex = Toy_findLiteralIndex(&compiler->literalCache, node->varDecl.identifier);
|
int identifierIndex = Toy_private_findLiteralIndex(&compiler->literalCache, node->varDecl.identifier);
|
||||||
if (identifierIndex < 0) {
|
if (identifierIndex < 0) {
|
||||||
identifierIndex = Toy_pushLiteralArray(&compiler->literalCache, node->varDecl.identifier);
|
identifierIndex = Toy_pushLiteralArray(&compiler->literalCache, node->varDecl.identifier);
|
||||||
}
|
}
|
||||||
@@ -518,11 +594,10 @@ static Toy_Opcode Toy_writeCompilerWithJumps(Toy_Compiler* compiler, Toy_ASTNode
|
|||||||
}
|
}
|
||||||
|
|
||||||
//create the function in the literal cache (by storing the compiler object)
|
//create the function in the literal cache (by storing the compiler object)
|
||||||
Toy_Literal fnLiteral = TOY_TO_FUNCTION_LITERAL(fnCompiler, 0);
|
Toy_Literal fnLiteral = ((Toy_Literal){ .as = { .generic = fnCompiler }, .type = TOY_LITERAL_FUNCTION_INTERMEDIATE});
|
||||||
fnLiteral.type = TOY_LITERAL_FUNCTION_INTERMEDIATE; //NOTE: changing type
|
|
||||||
|
|
||||||
//push the name
|
//push the name
|
||||||
int identifierIndex = Toy_findLiteralIndex(&compiler->literalCache, node->fnDecl.identifier);
|
int identifierIndex = Toy_private_findLiteralIndex(&compiler->literalCache, node->fnDecl.identifier);
|
||||||
if (identifierIndex < 0) {
|
if (identifierIndex < 0) {
|
||||||
identifierIndex = Toy_pushLiteralArray(&compiler->literalCache, node->fnDecl.identifier);
|
identifierIndex = Toy_pushLiteralArray(&compiler->literalCache, node->fnDecl.identifier);
|
||||||
}
|
}
|
||||||
@@ -578,7 +653,7 @@ static Toy_Opcode Toy_writeCompilerWithJumps(Toy_Compiler* compiler, Toy_ASTNode
|
|||||||
}
|
}
|
||||||
|
|
||||||
//write each argument to the bytecode
|
//write each argument to the bytecode
|
||||||
int argumentsIndex = Toy_findLiteralIndex(&compiler->literalCache, node->fnCall.arguments->fnCollection.nodes[i].atomic.literal);
|
int argumentsIndex = Toy_private_findLiteralIndex(&compiler->literalCache, node->fnCall.arguments->fnCollection.nodes[i].atomic.literal);
|
||||||
if (argumentsIndex < 0) {
|
if (argumentsIndex < 0) {
|
||||||
argumentsIndex = Toy_pushLiteralArray(&compiler->literalCache, node->fnCall.arguments->fnCollection.nodes[i].atomic.literal);
|
argumentsIndex = Toy_pushLiteralArray(&compiler->literalCache, node->fnCall.arguments->fnCollection.nodes[i].atomic.literal);
|
||||||
}
|
}
|
||||||
@@ -600,7 +675,7 @@ static Toy_Opcode Toy_writeCompilerWithJumps(Toy_Compiler* compiler, Toy_ASTNode
|
|||||||
|
|
||||||
//push the argument COUNT to the top of the stack
|
//push the argument COUNT to the top of the stack
|
||||||
Toy_Literal argumentsCountLiteral = TOY_TO_INTEGER_LITERAL(node->fnCall.argumentCount); //argumentCount is set elsewhere to support dot operator
|
Toy_Literal argumentsCountLiteral = TOY_TO_INTEGER_LITERAL(node->fnCall.argumentCount); //argumentCount is set elsewhere to support dot operator
|
||||||
int argumentsCountIndex = Toy_findLiteralIndex(&compiler->literalCache, argumentsCountLiteral);
|
int argumentsCountIndex = Toy_private_findLiteralIndex(&compiler->literalCache, argumentsCountLiteral);
|
||||||
if (argumentsCountIndex < 0) {
|
if (argumentsCountIndex < 0) {
|
||||||
argumentsCountIndex = Toy_pushLiteralArray(&compiler->literalCache, argumentsCountLiteral);
|
argumentsCountIndex = Toy_pushLiteralArray(&compiler->literalCache, argumentsCountLiteral);
|
||||||
}
|
}
|
||||||
@@ -1210,7 +1285,7 @@ static unsigned char* collateCompilerHeaderOpt(Toy_Compiler* compiler, size_t* s
|
|||||||
case TOY_LITERAL_FUNCTION_INTERMEDIATE: {
|
case TOY_LITERAL_FUNCTION_INTERMEDIATE: {
|
||||||
//extract the compiler
|
//extract the compiler
|
||||||
Toy_Literal fn = compiler->literalCache.literals[i];
|
Toy_Literal fn = compiler->literalCache.literals[i];
|
||||||
void* fnCompiler = TOY_AS_FUNCTION(fn).inner.bytecode; //store the compiler here for now
|
void* fnCompiler = fn.as.generic; //store the compiler here for now
|
||||||
|
|
||||||
//collate the function into bytecode (without header)
|
//collate the function into bytecode (without header)
|
||||||
size_t size = 0;
|
size_t size = 0;
|
||||||
|
|||||||
+43
-5
@@ -1,11 +1,20 @@
|
|||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
|
/*!
|
||||||
|
# toy_compiler.h
|
||||||
|
|
||||||
|
This header defines the compiler structure, which is used to transform abstract syntax trees into usable intermediate bytecode. There are two steps to generating bytecode - the writing step, and the collation step.
|
||||||
|
|
||||||
|
During the writing step, the core of the program is generated, along with a series of literals representing the values within the program; these values are compressed and flattened into semi-unrecognizable forms. If the same literal is used multiple times in a program, such as a variable name, the name itself is replaced by a reference to the flattened literals within the cache.
|
||||||
|
|
||||||
|
During the collation step, everything from the core program’s execution instructions, the flattened literals, the functions (which have their own sections and protocols within the bytecode) and version information (such as the macros defined in toy_common.h) are all combined into a single buffer of bytes, known as bytecode. This bytecode can then be safely saved to a file or immediately executed.
|
||||||
|
!*/
|
||||||
|
|
||||||
#include "toy_common.h"
|
#include "toy_common.h"
|
||||||
#include "toy_opcodes.h"
|
#include "toy_opcodes.h"
|
||||||
#include "toy_ast_node.h"
|
#include "toy_ast_node.h"
|
||||||
#include "toy_literal_array.h"
|
#include "toy_literal_array.h"
|
||||||
|
|
||||||
//the compiler takes the nodes, and turns them into sequential chunks of bytecode, saving literals to an external array
|
|
||||||
typedef struct Toy_Compiler {
|
typedef struct Toy_Compiler {
|
||||||
Toy_LiteralArray literalCache;
|
Toy_LiteralArray literalCache;
|
||||||
unsigned char* bytecode;
|
unsigned char* bytecode;
|
||||||
@@ -14,9 +23,38 @@ typedef struct Toy_Compiler {
|
|||||||
bool panic;
|
bool panic;
|
||||||
} Toy_Compiler;
|
} Toy_Compiler;
|
||||||
|
|
||||||
TOY_API void Toy_initCompiler(Toy_Compiler* compiler);
|
/*!
|
||||||
TOY_API void Toy_writeCompiler(Toy_Compiler* compiler, Toy_ASTNode* node);
|
## Define Functions
|
||||||
TOY_API void Toy_freeCompiler(Toy_Compiler* compiler);
|
|
||||||
|
|
||||||
//embed the header, data section, code section, function section, etc.
|
Executing the following functions out-of-order causes undefiend behaviour.
|
||||||
|
!*/
|
||||||
|
|
||||||
|
/*!
|
||||||
|
### void Toy_initCompiler(Toy_Compiler* compiler)
|
||||||
|
|
||||||
|
This function initializes the given compiler.
|
||||||
|
!*/
|
||||||
|
TOY_API void Toy_initCompiler(Toy_Compiler* compiler);
|
||||||
|
|
||||||
|
/*!
|
||||||
|
### void Toy_writeCompiler(Toy_Compiler* compiler, Toy_ASTNode* node)
|
||||||
|
|
||||||
|
This function writes the given `node` argument to the compiler. During the writing step, this function may be called repeatedly, with a stream of results from `Toy_scanParser()`, until `Toy_scanParser()` returns `NULL`.
|
||||||
|
!*/
|
||||||
|
TOY_API void Toy_writeCompiler(Toy_Compiler* compiler, Toy_ASTNode* node);
|
||||||
|
|
||||||
|
/*!
|
||||||
|
### unsigned char* Toy_collateCompiler(Toy_Compiler* compiler, size_t* size)
|
||||||
|
|
||||||
|
This function returns a buffer of bytes, known as "bytecode", created from the given compiler; it also stores the size of the bytecode in the variable pointed to by `size`.
|
||||||
|
|
||||||
|
Calling `Toy_collateCompiler()` multiple times on the same compiler will produce undefined behaviour.
|
||||||
|
!*/
|
||||||
TOY_API unsigned char* Toy_collateCompiler(Toy_Compiler* compiler, size_t* size);
|
TOY_API unsigned char* Toy_collateCompiler(Toy_Compiler* compiler, size_t* size);
|
||||||
|
|
||||||
|
/*!
|
||||||
|
### void Toy_freeCompiler(Toy_Compiler* compiler)
|
||||||
|
|
||||||
|
This function frees a compiler. Calling this on a compiler which has not been collated will free that compiler as expected - anything written to it will be lost.
|
||||||
|
!*/
|
||||||
|
TOY_API void Toy_freeCompiler(Toy_Compiler* compiler);
|
||||||
|
|||||||
@@ -1,6 +1,16 @@
|
|||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
//NOTE: you need both font AND background for these to work
|
/* toy_console_colors.h - console utility
|
||||||
|
|
||||||
|
This file provides a number of macros that can set the color of text in a console
|
||||||
|
window. These are used for convenience only. They are supposed to be dropped into
|
||||||
|
a printf()'s first argument, like so:
|
||||||
|
|
||||||
|
printf(TOY_CC_NOTICE "Hello world" TOY_CC_RESET);
|
||||||
|
|
||||||
|
NOTE: you need both font AND background for these to work
|
||||||
|
|
||||||
|
*/
|
||||||
|
|
||||||
//platform/compiler-specific instructions
|
//platform/compiler-specific instructions
|
||||||
#if defined(__linux__) || defined(__MINGW32__) || defined(__GNUC__)
|
#if defined(__linux__) || defined(__MINGW32__) || defined(__GNUC__)
|
||||||
|
|||||||
@@ -1,12 +0,0 @@
|
|||||||
#include "toy_common.h"
|
|
||||||
|
|
||||||
#include "toy_literal.h"
|
|
||||||
#include "toy_interpreter.h"
|
|
||||||
|
|
||||||
//file system API - these need to be set by the host
|
|
||||||
TOY_API void Toy_initDriveSystem();
|
|
||||||
TOY_API void Toy_freeDriveSystem();
|
|
||||||
|
|
||||||
//file system API - for use with libs
|
|
||||||
TOY_API void Toy_setDrivePath(char* drive, char* path);
|
|
||||||
TOY_API Toy_Literal Toy_getDrivePathLiteral(Toy_Interpreter* interpreter, Toy_Literal* drivePathLiteral);
|
|
||||||
+521
-617
File diff suppressed because it is too large
Load Diff
+160
-11
@@ -1,5 +1,41 @@
|
|||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
|
/*!
|
||||||
|
# toy_interpreter.h
|
||||||
|
|
||||||
|
This header defines the interpreter structure, which is the beating heart of Toy.
|
||||||
|
|
||||||
|
`Toy_Interpreter` is a stack-based, bytecode-driven interpreter with a number of customisation options, including "hooks"; native C functions wrapped in `Toy_Literal` instances, injected into the interpreter in order to give the Toy scripts access to libraries via the `import` keyword. The hooks, when invoked this way, can then inject further native functions into the interpreter's current scope. Exactly which hooks are made available varies by host program, but `standard` is the most commonly included one.
|
||||||
|
|
||||||
|
Another useful customisation feature is the ability to redicrect output from the `print` and `assert` keywords, as well as any internal errors that occur. This can allow you to add in a logging system, or even hook the `print` statement up to some kind of HUD.
|
||||||
|
|
||||||
|
## Defined Interfaces
|
||||||
|
|
||||||
|
Note: These interfaces are *actually* defined in [toy_literal.h](toy_literal_h.md) but are documented here, because this is where it matters most.
|
||||||
|
|
||||||
|
### typedef void (*Toy_PrintFn)(const char*)
|
||||||
|
|
||||||
|
This is the interface used by "print functions" - that is, functions used to print messages from the `print` and `assert` keywords, as well as internal interpreter errors.
|
||||||
|
|
||||||
|
### typedef int (*Toy_NativeFn)(struct Toy_Interpreter* interpreter, struct Toy_LiteralArray* arguments)
|
||||||
|
|
||||||
|
This is the interface used by "native functions" - that is, functions written in C which can be called directly by Toy scripts.
|
||||||
|
|
||||||
|
The arguments to the function are passed in as a `Toy_LiteralArray`.
|
||||||
|
|
||||||
|
### typedef int (*Toy_HookFn)(struct Toy_Interpreter* interpreter, struct Toy_Literal identifier, struct Toy_Literal alias)
|
||||||
|
|
||||||
|
This is the interface used by "hook functions" - that is, functions written in C which are invoked by using the `import` keyword, and are intended to inject other native functions into the current scope. While hook functions are capable of doing other things, this is greatly discouraged.
|
||||||
|
|
||||||
|
The identifier of the library (its name) is passed in as a `Toy_Literal`, as is any given alias; if no alias is given, then `alias` will be a null literal. Here, the identifier is `standard`, while the alias is `std`.
|
||||||
|
|
||||||
|
```
|
||||||
|
import standard as std;
|
||||||
|
```
|
||||||
|
|
||||||
|
Conventionally, when an alias is given, all of the functions should instead be inserted into a `Toy_LiteralDictionary` which is then inserted into the scope with the alias as its identifier.
|
||||||
|
!*/
|
||||||
|
|
||||||
#include "toy_common.h"
|
#include "toy_common.h"
|
||||||
#include "toy_literal.h"
|
#include "toy_literal.h"
|
||||||
#include "toy_literal_array.h"
|
#include "toy_literal_array.h"
|
||||||
@@ -31,21 +67,134 @@ typedef struct Toy_Interpreter {
|
|||||||
bool panic;
|
bool panic;
|
||||||
} Toy_Interpreter;
|
} Toy_Interpreter;
|
||||||
|
|
||||||
//native API
|
/*!
|
||||||
|
## Defined Functions
|
||||||
|
!*/
|
||||||
|
|
||||||
|
/*!
|
||||||
|
### void Toy_initInterpreter(Toy_Interpreter* interpreter)
|
||||||
|
|
||||||
|
This function initializes the interpreter. It allocates memory for internal systems such as the stack, and zeroes-out systems that have yet to be invoked. Internally, it also invokes `Toy_resetInterpreter` to initialize the environment.
|
||||||
|
!*/
|
||||||
|
TOY_API void Toy_initInterpreter(Toy_Interpreter* interpreter); //start of program
|
||||||
|
|
||||||
|
/*!
|
||||||
|
### void Toy_runInterpreter(Toy_Interpreter* interpreter, const unsigned char* bytecode, size_t length)
|
||||||
|
|
||||||
|
This function takes a `Toy_Interpreter` and `bytecode` (as well as the `length` of the bytecode), checks its version information, parses and un-flattens the literal cache, and executes the compiled program stored in the bytecode. This function also consumes the bytecode, so the `bytecode` argument is no longer valid after calls.
|
||||||
|
|
||||||
|
If the given bytecode's embedded version is not compatible with the current interpreter, then this function will refuse to execute.
|
||||||
|
|
||||||
|
Re-using a `Toy_Interpreter` instance without first resetting it is possible (that's how the repl works), however doing so may have unintended consequences if the scripts are not intended to be used in such a way. Any variables declared will persist.
|
||||||
|
!*/
|
||||||
|
TOY_API void Toy_runInterpreter(Toy_Interpreter* interpreter, const unsigned char* bytecode, size_t length);
|
||||||
|
|
||||||
|
/*!
|
||||||
|
### void Toy_resetInterpreter(Toy_Interpreter* interpreter)
|
||||||
|
|
||||||
|
This function frees any scopes that the scripts have built up, and generates a new one. It also injects several globally available functions:
|
||||||
|
|
||||||
|
* set
|
||||||
|
* get
|
||||||
|
* push
|
||||||
|
* pop
|
||||||
|
* length
|
||||||
|
* clear
|
||||||
|
!*/
|
||||||
|
TOY_API void Toy_resetInterpreter(Toy_Interpreter* interpreter);
|
||||||
|
|
||||||
|
/*!
|
||||||
|
### void Toy_freeInterpreter(Toy_Interpreter* interpreter)
|
||||||
|
|
||||||
|
This function frees a `Toy_Interpreter`, clearing all of the memory used within. That interpreter is no longer valid for use, and must be re-initialized.
|
||||||
|
!*/
|
||||||
|
TOY_API void Toy_freeInterpreter(Toy_Interpreter* interpreter);
|
||||||
|
|
||||||
|
/*!
|
||||||
|
### bool Toy_injectNativeFn(Toy_Interpreter* interpreter, const char* name, Toy_NativeFn func)
|
||||||
|
|
||||||
|
This function will inject the given native function `func` into the `Toy_Interpreter`'s current scope, with the identifer as `name`. Both the name and function will be converted into literals internally before being stored. It will return true on success, otherwise it will return false.
|
||||||
|
|
||||||
|
The primary use of this function is within hooks.
|
||||||
|
!*/
|
||||||
TOY_API bool Toy_injectNativeFn(Toy_Interpreter* interpreter, const char* name, Toy_NativeFn func);
|
TOY_API bool Toy_injectNativeFn(Toy_Interpreter* interpreter, const char* name, Toy_NativeFn func);
|
||||||
|
|
||||||
|
/*!
|
||||||
|
### bool Toy_injectNativeHook(Toy_Interpreter* interpreter, const char* name, Toy_HookFn hook)
|
||||||
|
|
||||||
|
This function will inject the given native function `hook` into the `Toy_Interpreter`'s hook cache, with the identifier as `name`. Both the name and the function will be converted into literals internally before being stored. It will return true on success, otherwise it will return false.
|
||||||
|
|
||||||
|
Hooks are invoked with the `import` keyword within Toy's scripts.
|
||||||
|
!*/
|
||||||
TOY_API bool Toy_injectNativeHook(Toy_Interpreter* interpreter, const char* name, Toy_HookFn hook);
|
TOY_API bool Toy_injectNativeHook(Toy_Interpreter* interpreter, const char* name, Toy_HookFn hook);
|
||||||
|
|
||||||
|
/*!
|
||||||
|
### bool Toy_callLiteralFn(Toy_Interpreter* interpreter, Toy_Literal func, Toy_LiteralArray* arguments, Toy_LiteralArray* returns)
|
||||||
|
|
||||||
|
This function calls a `Toy_Literal` which contains a function, with the arguments to that function passed in as `arguments` and the results stored in `returns`. It returns true on success, otherwise it returns false.
|
||||||
|
|
||||||
|
The literal `func` can be either a native function or a Toy function, but it won't execute a hook.
|
||||||
|
!*/
|
||||||
TOY_API bool Toy_callLiteralFn(Toy_Interpreter* interpreter, Toy_Literal func, Toy_LiteralArray* arguments, Toy_LiteralArray* returns);
|
TOY_API bool Toy_callLiteralFn(Toy_Interpreter* interpreter, Toy_Literal func, Toy_LiteralArray* arguments, Toy_LiteralArray* returns);
|
||||||
|
|
||||||
|
/*!
|
||||||
|
### bool Toy_callFn(Toy_Interpreter* interpreter, const char* name, Toy_LiteralArray* arguments, Toy_LiteralArray* returns)
|
||||||
|
|
||||||
|
This utility function will find a `Toy_literal` within the `Toy_Interpreter`'s scope with an identifier that matches `name`, and will invoke it using `Toy_callLiteralFn` (passing in `arguments` and `returns` as expected).
|
||||||
|
!*/
|
||||||
TOY_API bool Toy_callFn(Toy_Interpreter* interpreter, const char* name, Toy_LiteralArray* arguments, Toy_LiteralArray* returns);
|
TOY_API bool Toy_callFn(Toy_Interpreter* interpreter, const char* name, Toy_LiteralArray* arguments, Toy_LiteralArray* returns);
|
||||||
|
|
||||||
//utilities for the host program
|
/*!
|
||||||
TOY_API bool Toy_parseIdentifierToValue(Toy_Interpreter* interpreter, Toy_Literal* literalPtr);
|
### bool Toy_parseIdentifierToValue(Toy_Interpreter* interpreter, Toy_Literal* literalPtr)
|
||||||
TOY_API void Toy_setInterpreterPrint(Toy_Interpreter* interpreter, Toy_PrintFn printOutput);
|
|
||||||
TOY_API void Toy_setInterpreterAssert(Toy_Interpreter* interpreter, Toy_PrintFn assertOutput);
|
|
||||||
TOY_API void Toy_setInterpreterError(Toy_Interpreter* interpreter, Toy_PrintFn errorOutput);
|
|
||||||
|
|
||||||
//main access
|
Sometimes, native functions will receive `Toy_Literal` identifiers instead of the values - the correct values can be retreived from the given interpreter's scope using the following pattern:
|
||||||
TOY_API void Toy_initInterpreter(Toy_Interpreter* interpreter); //start of program
|
|
||||||
TOY_API void Toy_runInterpreter(Toy_Interpreter* interpreter, const unsigned char* bytecode, size_t length); //run the code
|
```c
|
||||||
TOY_API void Toy_resetInterpreter(Toy_Interpreter* interpreter); //use this to reset the interpreter's environment between runs
|
Toy_Literal foobarIdn = foobar;
|
||||||
TOY_API void Toy_freeInterpreter(Toy_Interpreter* interpreter); //end of program
|
if (TOY_IS_IDENTIFIER(foobar) && Toy_parseIdentifierToValue(interpreter, &foobar)) {
|
||||||
|
freeLiteral(foobarIdn); //remember to free the identifier
|
||||||
|
}
|
||||||
|
```
|
||||||
|
!*/
|
||||||
|
TOY_API bool Toy_parseIdentifierToValue(Toy_Interpreter* interpreter, Toy_Literal* literalPtr);
|
||||||
|
|
||||||
|
/*!
|
||||||
|
### void Toy_setInterpreterPrint(Toy_Interpreter* interpreter, Toy_PrintFn printOutput)
|
||||||
|
|
||||||
|
This function sets the function called by the `print` keyword. By default, the following wrapper is used:
|
||||||
|
|
||||||
|
```c
|
||||||
|
static void printWrapper(const char* output) {
|
||||||
|
printf("%s\n", output);
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
Note: The above is a very minor lie - in reality there are some preprocessor directives to allow the repl's `-n` flag to work.
|
||||||
|
!*/
|
||||||
|
TOY_API void Toy_setInterpreterPrint(Toy_Interpreter* interpreter, Toy_PrintFn printOutput);
|
||||||
|
|
||||||
|
/*!
|
||||||
|
### void Toy_setInterpreterAssert(Toy_Interpreter* interpreter, Toy_PrintFn assertOutput)
|
||||||
|
|
||||||
|
This function sets the function called by the `assert` keyword on failure. By default, the following wrapper is used:
|
||||||
|
|
||||||
|
```c
|
||||||
|
static void assertWrapper(const char* output) {
|
||||||
|
fprintf(stderr, "Assertion failure: %s\n", output);
|
||||||
|
}
|
||||||
|
```
|
||||||
|
!*/
|
||||||
|
TOY_API void Toy_setInterpreterAssert(Toy_Interpreter* interpreter, Toy_PrintFn assertOutput);
|
||||||
|
|
||||||
|
/*!
|
||||||
|
### void Toy_setInterpreterError(Toy_Interpreter* interpreter, Toy_PrintFn errorOutput)
|
||||||
|
|
||||||
|
This function sets the function called when an error occurs within the interpreter. By default, the following wrapper is used:
|
||||||
|
|
||||||
|
```c
|
||||||
|
static void errorWrapper(const char* output) {
|
||||||
|
fprintf(stderr, "%s", output); //no newline
|
||||||
|
}
|
||||||
|
```
|
||||||
|
!*/
|
||||||
|
TOY_API void Toy_setInterpreterError(Toy_Interpreter* interpreter, Toy_PrintFn errorOutput);
|
||||||
|
|||||||
+37
-1
@@ -1,5 +1,11 @@
|
|||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
|
/*!
|
||||||
|
# toy_lexer.h
|
||||||
|
|
||||||
|
This header defines the lexer and token structures, which can be bound to a piece of source code, and used to tokenize it within a parser.
|
||||||
|
!*/
|
||||||
|
|
||||||
#include "toy_common.h"
|
#include "toy_common.h"
|
||||||
#include "toy_token_types.h"
|
#include "toy_token_types.h"
|
||||||
|
|
||||||
@@ -20,10 +26,40 @@ typedef struct {
|
|||||||
int line;
|
int line;
|
||||||
} Toy_Token;
|
} Toy_Token;
|
||||||
|
|
||||||
|
/*!
|
||||||
|
## Defined Functions
|
||||||
|
!*/
|
||||||
|
|
||||||
|
/*!
|
||||||
|
### void Toy_initLexer(Toy_Lexer* lexer, const char* source)
|
||||||
|
|
||||||
|
This function initializes a lexer, binding it to the `source` parameter; the lexer is now ready to be passed to the parser.
|
||||||
|
!*/
|
||||||
TOY_API void Toy_initLexer(Toy_Lexer* lexer, const char* source);
|
TOY_API void Toy_initLexer(Toy_Lexer* lexer, const char* source);
|
||||||
|
|
||||||
|
/*!
|
||||||
|
### Toy_Token Toy_private_scanLexer(Toy_Lexer* lexer)
|
||||||
|
|
||||||
|
This function "scans" the lexer, returning a token to the parser.
|
||||||
|
|
||||||
|
Private functions are not intended for general use.
|
||||||
|
!*/
|
||||||
TOY_API Toy_Token Toy_private_scanLexer(Toy_Lexer* lexer);
|
TOY_API Toy_Token Toy_private_scanLexer(Toy_Lexer* lexer);
|
||||||
|
|
||||||
//for debugging
|
/*!
|
||||||
|
### void Toy_private_printToken(Toy_Token* token)
|
||||||
|
|
||||||
|
This function prints a given token to stdout.
|
||||||
|
|
||||||
|
Private functions are not intended for general use.
|
||||||
|
!*/
|
||||||
TOY_API void Toy_private_printToken(Toy_Token* token);
|
TOY_API void Toy_private_printToken(Toy_Token* token);
|
||||||
|
|
||||||
|
/*!
|
||||||
|
### void Toy_private_setComments(Toy_Lexer* lexer, bool enabled)
|
||||||
|
|
||||||
|
This function sets whether comments are allowed within source code. By default, comments are allowed, and are only disabled in the repl.
|
||||||
|
|
||||||
|
Private functions are not intended for general use.
|
||||||
|
!*/
|
||||||
TOY_API void Toy_private_setComments(Toy_Lexer* lexer, bool enabled);
|
TOY_API void Toy_private_setComments(Toy_Lexer* lexer, bool enabled);
|
||||||
|
|||||||
+24
-15
@@ -59,7 +59,7 @@ void Toy_freeLiteral(Toy_Literal literal) {
|
|||||||
if (TOY_IS_FUNCTION(literal)) {
|
if (TOY_IS_FUNCTION(literal)) {
|
||||||
Toy_popScope(TOY_AS_FUNCTION(literal).scope);
|
Toy_popScope(TOY_AS_FUNCTION(literal).scope);
|
||||||
TOY_AS_FUNCTION(literal).scope = NULL;
|
TOY_AS_FUNCTION(literal).scope = NULL;
|
||||||
TOY_FREE_ARRAY(unsigned char, TOY_AS_FUNCTION(literal).inner.bytecode, TOY_AS_FUNCTION_BYTECODE_LENGTH(literal));
|
Toy_deleteRefFunction((Toy_RefFunction*)(TOY_AS_FUNCTION(literal).inner.ptr));
|
||||||
}
|
}
|
||||||
|
|
||||||
if (TOY_IS_TYPE(literal) && TOY_AS_TYPE(literal).capacity > 0) {
|
if (TOY_IS_TYPE(literal) && TOY_AS_TYPE(literal).capacity > 0) {
|
||||||
@@ -84,12 +84,8 @@ bool Toy_private_isTruthy(Toy_Literal x) {
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
Toy_Literal Toy_private_toStringLiteral(Toy_RefString* ptr) {
|
|
||||||
return ((Toy_Literal){{ .string = { .ptr = ptr }},TOY_LITERAL_STRING, 0});
|
|
||||||
}
|
|
||||||
|
|
||||||
Toy_Literal Toy_private_toIdentifierLiteral(Toy_RefString* ptr) {
|
Toy_Literal Toy_private_toIdentifierLiteral(Toy_RefString* ptr) {
|
||||||
return ((Toy_Literal){{ .identifier = { .ptr = ptr, .hash = hashString(Toy_toCString(ptr), Toy_lengthRefString(ptr)) }},TOY_LITERAL_IDENTIFIER, 0});
|
return ((Toy_Literal){{ .identifier = { .ptr = ptr, .hash = hashString(Toy_toCString(ptr), Toy_lengthRefString(ptr)) }},TOY_LITERAL_IDENTIFIER});
|
||||||
}
|
}
|
||||||
|
|
||||||
Toy_Literal* Toy_private_typePushSubtype(Toy_Literal* lit, Toy_Literal subtype) {
|
Toy_Literal* Toy_private_typePushSubtype(Toy_Literal* lit, Toy_Literal subtype) {
|
||||||
@@ -123,6 +119,10 @@ Toy_Literal Toy_copyLiteral(Toy_Literal original) {
|
|||||||
Toy_LiteralArray* array = TOY_ALLOCATE(Toy_LiteralArray, 1);
|
Toy_LiteralArray* array = TOY_ALLOCATE(Toy_LiteralArray, 1);
|
||||||
Toy_initLiteralArray(array);
|
Toy_initLiteralArray(array);
|
||||||
|
|
||||||
|
//preallocate enough space
|
||||||
|
array->capacity = TOY_AS_ARRAY(original)->capacity;
|
||||||
|
array->literals = TOY_GROW_ARRAY(Toy_Literal, array->literals, 0, array->capacity);
|
||||||
|
|
||||||
//copy each element
|
//copy each element
|
||||||
for (int i = 0; i < TOY_AS_ARRAY(original)->count; i++) {
|
for (int i = 0; i < TOY_AS_ARRAY(original)->count; i++) {
|
||||||
Toy_pushLiteralArray(array, TOY_AS_ARRAY(original)->literals[i]);
|
Toy_pushLiteralArray(array, TOY_AS_ARRAY(original)->literals[i]);
|
||||||
@@ -135,6 +135,15 @@ Toy_Literal Toy_copyLiteral(Toy_Literal original) {
|
|||||||
Toy_LiteralDictionary* dictionary = TOY_ALLOCATE(Toy_LiteralDictionary, 1);
|
Toy_LiteralDictionary* dictionary = TOY_ALLOCATE(Toy_LiteralDictionary, 1);
|
||||||
Toy_initLiteralDictionary(dictionary);
|
Toy_initLiteralDictionary(dictionary);
|
||||||
|
|
||||||
|
//preallocate enough space
|
||||||
|
dictionary->capacity = TOY_AS_DICTIONARY(original)->capacity;
|
||||||
|
dictionary->entries = TOY_ALLOCATE(Toy_private_dictionary_entry, dictionary->capacity);
|
||||||
|
|
||||||
|
for (int i = 0; i < dictionary->capacity; i++) {
|
||||||
|
dictionary->entries[i].key = TOY_TO_NULL_LITERAL;
|
||||||
|
dictionary->entries[i].value = TOY_TO_NULL_LITERAL;
|
||||||
|
}
|
||||||
|
|
||||||
//copy each entry
|
//copy each entry
|
||||||
for (int i = 0; i < TOY_AS_DICTIONARY(original)->capacity; i++) {
|
for (int i = 0; i < TOY_AS_DICTIONARY(original)->capacity; i++) {
|
||||||
if ( !TOY_IS_NULL(TOY_AS_DICTIONARY(original)->entries[i].key) ) {
|
if ( !TOY_IS_NULL(TOY_AS_DICTIONARY(original)->entries[i].key) ) {
|
||||||
@@ -146,10 +155,8 @@ Toy_Literal Toy_copyLiteral(Toy_Literal original) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
case TOY_LITERAL_FUNCTION: {
|
case TOY_LITERAL_FUNCTION: {
|
||||||
unsigned char* buffer = TOY_ALLOCATE(unsigned char, TOY_AS_FUNCTION_BYTECODE_LENGTH(original));
|
Toy_Literal literal = TOY_TO_FUNCTION_LITERAL(Toy_copyRefFunction( TOY_AS_FUNCTION(original).inner.ptr ));
|
||||||
memcpy(buffer, TOY_AS_FUNCTION(original).inner.bytecode, TOY_AS_FUNCTION_BYTECODE_LENGTH(original));
|
|
||||||
|
|
||||||
Toy_Literal literal = TOY_TO_FUNCTION_LITERAL(buffer, TOY_AS_FUNCTION_BYTECODE_LENGTH(original));
|
|
||||||
TOY_AS_FUNCTION(literal).scope = Toy_copyScope(TOY_AS_FUNCTION(original).scope);
|
TOY_AS_FUNCTION(literal).scope = Toy_copyScope(TOY_AS_FUNCTION(original).scope);
|
||||||
|
|
||||||
return literal;
|
return literal;
|
||||||
@@ -174,7 +181,7 @@ Toy_Literal Toy_copyLiteral(Toy_Literal original) {
|
|||||||
return original; //literally a shallow copy
|
return original; //literally a shallow copy
|
||||||
}
|
}
|
||||||
|
|
||||||
case TOY_LITERAL_ARRAY_INTERMEDIATE: {
|
case TOY_LITERAL_ARRAY_INTERMEDIATE: { //TODO: efficient preallocation?
|
||||||
Toy_LiteralArray* array = TOY_ALLOCATE(Toy_LiteralArray, 1);
|
Toy_LiteralArray* array = TOY_ALLOCATE(Toy_LiteralArray, 1);
|
||||||
Toy_initLiteralArray(array);
|
Toy_initLiteralArray(array);
|
||||||
|
|
||||||
@@ -190,7 +197,7 @@ Toy_Literal Toy_copyLiteral(Toy_Literal original) {
|
|||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
case TOY_LITERAL_DICTIONARY_INTERMEDIATE: {
|
case TOY_LITERAL_DICTIONARY_INTERMEDIATE: { //TODO: efficient preallocation?
|
||||||
Toy_LiteralArray* array = TOY_ALLOCATE(Toy_LiteralArray, 1);
|
Toy_LiteralArray* array = TOY_ALLOCATE(Toy_LiteralArray, 1);
|
||||||
Toy_initLiteralArray(array);
|
Toy_initLiteralArray(array);
|
||||||
|
|
||||||
@@ -206,7 +213,7 @@ Toy_Literal Toy_copyLiteral(Toy_Literal original) {
|
|||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
case TOY_LITERAL_TYPE_INTERMEDIATE: {
|
case TOY_LITERAL_TYPE_INTERMEDIATE: { //TODO: efficient preallocation?
|
||||||
Toy_LiteralArray* array = TOY_ALLOCATE(Toy_LiteralArray, 1);
|
Toy_LiteralArray* array = TOY_ALLOCATE(Toy_LiteralArray, 1);
|
||||||
Toy_initLiteralArray(array);
|
Toy_initLiteralArray(array);
|
||||||
|
|
||||||
@@ -378,8 +385,10 @@ int Toy_hashLiteral(Toy_Literal lit) {
|
|||||||
case TOY_LITERAL_INTEGER:
|
case TOY_LITERAL_INTEGER:
|
||||||
return hashUInt((unsigned int)TOY_AS_INTEGER(lit));
|
return hashUInt((unsigned int)TOY_AS_INTEGER(lit));
|
||||||
|
|
||||||
case TOY_LITERAL_FLOAT:
|
case TOY_LITERAL_FLOAT: {
|
||||||
return hashUInt(*(unsigned int*)(&TOY_AS_FLOAT(lit)));
|
unsigned int* ptr = (unsigned int*)(&TOY_AS_FLOAT(lit));
|
||||||
|
return hashUInt(*ptr);
|
||||||
|
}
|
||||||
|
|
||||||
case TOY_LITERAL_STRING:
|
case TOY_LITERAL_STRING:
|
||||||
return hashString(Toy_toCString(TOY_AS_STRING(lit)), Toy_lengthRefString(TOY_AS_STRING(lit)));
|
return hashString(Toy_toCString(TOY_AS_STRING(lit)), Toy_lengthRefString(TOY_AS_STRING(lit)));
|
||||||
@@ -446,7 +455,7 @@ static void printToBuffer(const char* str) {
|
|||||||
globalPrintBuffer = TOY_GROW_ARRAY(char, globalPrintBuffer, oldCapacity, globalPrintCapacity);
|
globalPrintBuffer = TOY_GROW_ARRAY(char, globalPrintBuffer, oldCapacity, globalPrintCapacity);
|
||||||
}
|
}
|
||||||
|
|
||||||
snprintf(globalPrintBuffer + globalPrintCount, strlen(str) + 1, "%s", str);
|
snprintf(globalPrintBuffer + globalPrintCount, strlen(str) + 1, "%s", str ? str : "\0");
|
||||||
globalPrintCount += strlen(str);
|
globalPrintCount += strlen(str);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
+251
-29
@@ -1,8 +1,19 @@
|
|||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
|
/*!
|
||||||
|
# toy_literal.h
|
||||||
|
|
||||||
|
This header defines the literal structure, which is used extensively throughout Toy to represent values of some kind.
|
||||||
|
|
||||||
|
The main way of interacting with literals is to use a macro of some kind, as the exact implementation of `Toy_Literal` has and will change based on the needs of Toy.
|
||||||
|
|
||||||
|
User data can be passed around within Toy as an opaque type - use the tag value for determining what kind of opaque it is, or leave it as 0.
|
||||||
|
!*/
|
||||||
|
|
||||||
#include "toy_common.h"
|
#include "toy_common.h"
|
||||||
|
|
||||||
#include "toy_refstring.h"
|
#include "toy_refstring.h"
|
||||||
|
#include "toy_reffunction.h"
|
||||||
|
|
||||||
//forward delcare stuff
|
//forward delcare stuff
|
||||||
struct Toy_Literal;
|
struct Toy_Literal;
|
||||||
@@ -14,6 +25,31 @@ typedef int (*Toy_NativeFn)(struct Toy_Interpreter* interpreter, struct Toy_Lite
|
|||||||
typedef int (*Toy_HookFn)(struct Toy_Interpreter* interpreter, struct Toy_Literal identifier, struct Toy_Literal alias);
|
typedef int (*Toy_HookFn)(struct Toy_Interpreter* interpreter, struct Toy_Literal identifier, struct Toy_Literal alias);
|
||||||
typedef void (*Toy_PrintFn)(const char*);
|
typedef void (*Toy_PrintFn)(const char*);
|
||||||
|
|
||||||
|
/*!
|
||||||
|
## Defined Enums
|
||||||
|
|
||||||
|
### Toy_LiteralType
|
||||||
|
|
||||||
|
* `TOY_LITERAL_NULL`
|
||||||
|
* `TOY_LITERAL_BOOLEAN`
|
||||||
|
* `TOY_LITERAL_INTEGER`
|
||||||
|
* `TOY_LITERAL_FLOAT`
|
||||||
|
* `TOY_LITERAL_STRING`
|
||||||
|
* `TOY_LITERAL_ARRAY`
|
||||||
|
* `TOY_LITERAL_DICTIONARY`
|
||||||
|
* `TOY_LITERAL_FUNCTION`
|
||||||
|
* `TOY_LITERAL_FUNCTION_NATIVE`
|
||||||
|
* `TOY_LITERAL_FUNCTION_HOOK`
|
||||||
|
* `TOY_LITERAL_IDENTIFIER`
|
||||||
|
* `TOY_LITERAL_TYPE`
|
||||||
|
* `TOY_LITERAL_OPAQUE`
|
||||||
|
* `TOY_LITERAL_ANY`
|
||||||
|
|
||||||
|
These are the main values of `Toy_LiteralType`, each of which represents a potential state of the `Toy_Literal` structure. Do not interact with a literal without determining its type with the `IS_*` macros first.
|
||||||
|
|
||||||
|
Other type values are possible, but are only used internally.
|
||||||
|
!*/
|
||||||
|
|
||||||
typedef enum {
|
typedef enum {
|
||||||
TOY_LITERAL_NULL,
|
TOY_LITERAL_NULL,
|
||||||
TOY_LITERAL_BOOLEAN,
|
TOY_LITERAL_BOOLEAN,
|
||||||
@@ -55,7 +91,7 @@ typedef struct Toy_Literal {
|
|||||||
|
|
||||||
struct {
|
struct {
|
||||||
union {
|
union {
|
||||||
void* bytecode; //8
|
Toy_RefFunction* ptr; //8
|
||||||
Toy_NativeFn native; //8
|
Toy_NativeFn native; //8
|
||||||
Toy_HookFn hook; //8
|
Toy_HookFn hook; //8
|
||||||
} inner; //8
|
} inner; //8
|
||||||
@@ -63,12 +99,11 @@ typedef struct Toy_Literal {
|
|||||||
} function; //16
|
} function; //16
|
||||||
|
|
||||||
struct { //for variable names
|
struct { //for variable names
|
||||||
Toy_RefString* ptr; //8
|
Toy_RefString* ptr; //8
|
||||||
int hash; //4
|
int hash; //4
|
||||||
} identifier; //16
|
} identifier; //16
|
||||||
|
|
||||||
struct {
|
struct {
|
||||||
|
|
||||||
struct Toy_Literal* subtypes; //8
|
struct Toy_Literal* subtypes; //8
|
||||||
Toy_LiteralType typeOf; //4
|
Toy_LiteralType typeOf; //4
|
||||||
unsigned char capacity; //1
|
unsigned char capacity; //1
|
||||||
@@ -80,12 +115,37 @@ typedef struct Toy_Literal {
|
|||||||
void* ptr; //8
|
void* ptr; //8
|
||||||
int tag; //4
|
int tag; //4
|
||||||
} opaque; //16
|
} opaque; //16
|
||||||
|
|
||||||
|
void* generic; //8
|
||||||
} as; //16
|
} as; //16
|
||||||
|
|
||||||
Toy_LiteralType type; //4
|
Toy_LiteralType type; //4
|
||||||
int bytecodeLength; //4 - shenanigans with byte alignment reduces the size of Toy_Literal
|
//4 - unused
|
||||||
|
//shenanigans with byte alignment reduces the size of Toy_Literal
|
||||||
} Toy_Literal;
|
} Toy_Literal;
|
||||||
|
|
||||||
|
/*!
|
||||||
|
## Defined Macros
|
||||||
|
!*/
|
||||||
|
|
||||||
|
/*!
|
||||||
|
The following macros are used to determine if a given literal, passed in as `value`, is of a specific type. It should be noted that `TOY_IS_FUNCTION` will return false for native and hook functions.
|
||||||
|
|
||||||
|
* `TOY_IS_NULL(value)`
|
||||||
|
* `TOY_IS_BOOLEAN(value)`
|
||||||
|
* `TOY_IS_INTEGER(value)`
|
||||||
|
* `TOY_IS_FLOAT(value)`
|
||||||
|
* `TOY_IS_STRING(value)`
|
||||||
|
* `TOY_IS_ARRAY(value)`
|
||||||
|
* `TOY_IS_DICTIONARY(value)`
|
||||||
|
* `TOY_IS_FUNCTION(value)`
|
||||||
|
* `TOY_IS_FUNCTION_NATIVE(value)`
|
||||||
|
* `TOY_IS_FUNCTION_HOOK(value)`
|
||||||
|
* `TOY_IS_IDENTIFIER(value)`
|
||||||
|
* `TOY_IS_TYPE(value)`
|
||||||
|
* `TOY_IS_OPAQUE(value)`
|
||||||
|
!*/
|
||||||
|
|
||||||
#define TOY_IS_NULL(value) ((value).type == TOY_LITERAL_NULL)
|
#define TOY_IS_NULL(value) ((value).type == TOY_LITERAL_NULL)
|
||||||
#define TOY_IS_BOOLEAN(value) ((value).type == TOY_LITERAL_BOOLEAN)
|
#define TOY_IS_BOOLEAN(value) ((value).type == TOY_LITERAL_BOOLEAN)
|
||||||
#define TOY_IS_INTEGER(value) ((value).type == TOY_LITERAL_INTEGER)
|
#define TOY_IS_INTEGER(value) ((value).type == TOY_LITERAL_INTEGER)
|
||||||
@@ -100,6 +160,23 @@ typedef struct Toy_Literal {
|
|||||||
#define TOY_IS_TYPE(value) ((value).type == TOY_LITERAL_TYPE)
|
#define TOY_IS_TYPE(value) ((value).type == TOY_LITERAL_TYPE)
|
||||||
#define TOY_IS_OPAQUE(value) ((value).type == TOY_LITERAL_OPAQUE)
|
#define TOY_IS_OPAQUE(value) ((value).type == TOY_LITERAL_OPAQUE)
|
||||||
|
|
||||||
|
/*!
|
||||||
|
The following macros are used to cast a literal to a specific C type to be used.
|
||||||
|
|
||||||
|
* `TOY_AS_BOOLEAN(value)`
|
||||||
|
* `TOY_AS_INTEGER(value)`
|
||||||
|
* `TOY_AS_FLOAT(value)`
|
||||||
|
* `TOY_AS_STRING(value)`
|
||||||
|
* `TOY_AS_ARRAY(value)`
|
||||||
|
* `TOY_AS_DICTIONARY(value)`
|
||||||
|
* `TOY_AS_FUNCTION(value)`
|
||||||
|
* `TOY_AS_FUNCTION_NATIVE(value)`
|
||||||
|
* `TOY_AS_FUNCTION_HOOK(value)`
|
||||||
|
* `TOY_AS_IDENTIFIER(value)`
|
||||||
|
* `TOY_AS_TYPE(value)`
|
||||||
|
* `TOY_AS_OPAQUE(value)`
|
||||||
|
!*/
|
||||||
|
|
||||||
#define TOY_AS_BOOLEAN(value) ((value).as.boolean)
|
#define TOY_AS_BOOLEAN(value) ((value).as.boolean)
|
||||||
#define TOY_AS_INTEGER(value) ((value).as.integer)
|
#define TOY_AS_INTEGER(value) ((value).as.integer)
|
||||||
#define TOY_AS_FLOAT(value) ((value).as.number)
|
#define TOY_AS_FLOAT(value) ((value).as.number)
|
||||||
@@ -113,46 +190,191 @@ typedef struct Toy_Literal {
|
|||||||
#define TOY_AS_TYPE(value) ((value).as.type)
|
#define TOY_AS_TYPE(value) ((value).as.type)
|
||||||
#define TOY_AS_OPAQUE(value) ((value).as.opaque.ptr)
|
#define TOY_AS_OPAQUE(value) ((value).as.opaque.ptr)
|
||||||
|
|
||||||
#define TOY_TO_NULL_LITERAL ((Toy_Literal){{ .integer = 0 }, TOY_LITERAL_NULL, 0})
|
/*!
|
||||||
#define TOY_TO_BOOLEAN_LITERAL(value) ((Toy_Literal){{ .boolean = value }, TOY_LITERAL_BOOLEAN, 0})
|
The following macros are used to create a new literal, with the given `value` as it's internal value.
|
||||||
#define TOY_TO_INTEGER_LITERAL(value) ((Toy_Literal){{ .integer = value }, TOY_LITERAL_INTEGER, 0})
|
|
||||||
#define TOY_TO_FLOAT_LITERAL(value) ((Toy_Literal){{ .number = value }, TOY_LITERAL_FLOAT, 0})
|
* `TOY_TO_NULL_LITERAL` - does not need parantheses
|
||||||
#define TOY_TO_STRING_LITERAL(value) Toy_private_toStringLiteral(value)
|
* `TOY_TO_BOOLEAN_LITERAL(value)`
|
||||||
#define TOY_TO_ARRAY_LITERAL(value) ((Toy_Literal){{ .array = value }, TOY_LITERAL_ARRAY, 0})
|
* `TOY_TO_INTEGER_LITERAL(value)`
|
||||||
#define TOY_TO_DICTIONARY_LITERAL(value) ((Toy_Literal){{ .dictionary = value }, TOY_LITERAL_DICTIONARY, 0})
|
* `TOY_TO_FLOAT_LITERAL(value)`
|
||||||
#define TOY_TO_FUNCTION_LITERAL(value, l) ((Toy_Literal){{ .function = { .inner = { .bytecode = value }, .scope = NULL }}, TOY_LITERAL_FUNCTION, l})
|
* `TOY_TO_STRING_LITERAL(value)`
|
||||||
#define TOY_TO_FUNCTION_NATIVE_LITERAL(value) ((Toy_Literal){{ .function = { .inner = { .native = value }, .scope = NULL }}, TOY_LITERAL_FUNCTION_NATIVE, 0})
|
* `TOY_TO_ARRAY_LITERAL(value)`
|
||||||
#define TOY_TO_FUNCTION_HOOK_LITERAL(value) ((Toy_Literal){{ .function = { .inner = { .hook = value }, .scope = NULL }}, TOY_LITERAL_FUNCTION_HOOK, 0})
|
* `TOY_TO_DICTIONARY_LITERAL(value)`
|
||||||
|
* `TOY_TO_FUNCTION_LITERAL(value, l)` - `l` represents the length of the bytecode passed as `value`
|
||||||
|
* `TOY_TO_FUNCTION_NATIVE_LITERAL(value)`
|
||||||
|
* `TOY_TO_FUNCTION_HOOK_LITERAL(value)`
|
||||||
|
* `TOY_TO_IDENTIFIER_LITERAL(value)`
|
||||||
|
* `TOY_TO_TYPE_LITERAL(value, c)` - `c` is the true of the type should be const
|
||||||
|
* `TOY_TO_OPAQUE_LITERAL(value, t)` - `t` is the integer tag
|
||||||
|
!*/
|
||||||
|
|
||||||
|
#define TOY_TO_NULL_LITERAL ((Toy_Literal){{ .integer = 0 }, TOY_LITERAL_NULL})
|
||||||
|
#define TOY_TO_BOOLEAN_LITERAL(value) ((Toy_Literal){{ .boolean = value }, TOY_LITERAL_BOOLEAN})
|
||||||
|
#define TOY_TO_INTEGER_LITERAL(value) ((Toy_Literal){{ .integer = value }, TOY_LITERAL_INTEGER})
|
||||||
|
#define TOY_TO_FLOAT_LITERAL(value) ((Toy_Literal){{ .number = value }, TOY_LITERAL_FLOAT})
|
||||||
|
#define TOY_TO_STRING_LITERAL(value) ((Toy_Literal){{ .string = { .ptr = value }},TOY_LITERAL_STRING})
|
||||||
|
#define TOY_TO_ARRAY_LITERAL(value) ((Toy_Literal){{ .array = value }, TOY_LITERAL_ARRAY})
|
||||||
|
#define TOY_TO_DICTIONARY_LITERAL(value) ((Toy_Literal){{ .dictionary = value }, TOY_LITERAL_DICTIONARY})
|
||||||
|
#define TOY_TO_FUNCTION_LITERAL(value) ((Toy_Literal){{ .function = { .inner = { .ptr = value }, .scope = NULL }}, TOY_LITERAL_FUNCTION})
|
||||||
|
#define TOY_TO_FUNCTION_NATIVE_LITERAL(value) ((Toy_Literal){{ .function = { .inner = { .native = value }, .scope = NULL }}, TOY_LITERAL_FUNCTION_NATIVE})
|
||||||
|
#define TOY_TO_FUNCTION_HOOK_LITERAL(value) ((Toy_Literal){{ .function = { .inner = { .hook = value }, .scope = NULL }}, TOY_LITERAL_FUNCTION_HOOK})
|
||||||
#define TOY_TO_IDENTIFIER_LITERAL(value) Toy_private_toIdentifierLiteral(value)
|
#define TOY_TO_IDENTIFIER_LITERAL(value) Toy_private_toIdentifierLiteral(value)
|
||||||
#define TOY_TO_TYPE_LITERAL(value, c) ((Toy_Literal){{ .type = { .typeOf = value, .constant = c, .subtypes = NULL, .capacity = 0, .count = 0 }}, TOY_LITERAL_TYPE, 0})
|
#define TOY_TO_TYPE_LITERAL(value, c) ((Toy_Literal){{ .type = { .typeOf = value, .constant = c, .subtypes = NULL, .capacity = 0, .count = 0 }}, TOY_LITERAL_TYPE})
|
||||||
#define TOY_TO_OPAQUE_LITERAL(value, t) ((Toy_Literal){{ .opaque = { .ptr = value, .tag = t }}, TOY_LITERAL_OPAQUE, 0})
|
#define TOY_TO_OPAQUE_LITERAL(value, t) ((Toy_Literal){{ .opaque = { .ptr = value, .tag = t }}, TOY_LITERAL_OPAQUE})
|
||||||
|
|
||||||
//BUGFIX: For blank indexing
|
//BUGFIX: For blank indexing - not for general use
|
||||||
#define TOY_IS_INDEX_BLANK(value) ((value).type == TOY_LITERAL_INDEX_BLANK)
|
#define TOY_IS_INDEX_BLANK(value) ((value).type == TOY_LITERAL_INDEX_BLANK)
|
||||||
#define TOY_TO_INDEX_BLANK_LITERAL ((Toy_Literal){{ .integer = 0 }, TOY_LITERAL_INDEX_BLANK, 0})
|
#define TOY_TO_INDEX_BLANK_LITERAL ((Toy_Literal){{ .integer = 0 }, TOY_LITERAL_INDEX_BLANK})
|
||||||
|
|
||||||
TOY_API void Toy_freeLiteral(Toy_Literal literal);
|
/*!
|
||||||
|
## More Defined Macros
|
||||||
|
|
||||||
#define TOY_IS_TRUTHY(x) Toy_private_isTruthy(x)
|
The following macros are utilities used throughout Toy's internals, and are available for the user as well.
|
||||||
|
!*/
|
||||||
|
|
||||||
#define TOY_AS_FUNCTION_BYTECODE_LENGTH(lit) ((lit).bytecodeLength)
|
/*!
|
||||||
|
### TOY_IS_TRUTHY(x)
|
||||||
|
|
||||||
|
Returns true of the literal `x` is truthy, otherwise it returns false.
|
||||||
|
|
||||||
|
Currently, every value is considered truthy except `false`, which is falsy and `null`, which is neither true or false.
|
||||||
|
!*/
|
||||||
|
|
||||||
|
#define TOY_IS_TRUTHY(x) Toy_private_isTruthy(x)
|
||||||
|
|
||||||
|
/*!
|
||||||
|
### TOY_AS_FUNCTION_BYTECODE_LENGTH(lit)
|
||||||
|
|
||||||
|
Returns the length of a Toy function's bytecode.
|
||||||
|
|
||||||
|
This macro is only valid on `TOY_LITERAL_FUNCTION`.
|
||||||
|
!*/
|
||||||
|
#define TOY_AS_FUNCTION_BYTECODE_LENGTH(lit) (Toy_lengthRefFunction((lit).inner.ptr))
|
||||||
|
|
||||||
|
/*!
|
||||||
|
### TOY_MAX_STRING_LENGTH
|
||||||
|
|
||||||
|
The maximum length of a string in Toy, which is 4096 bytes by default. This can be changed at compile time, but the results of doing so are not officially supported.
|
||||||
|
!*/
|
||||||
#define TOY_MAX_STRING_LENGTH 4096
|
#define TOY_MAX_STRING_LENGTH 4096
|
||||||
|
|
||||||
|
/*!
|
||||||
|
### TOY_HASH_I(lit)
|
||||||
|
|
||||||
|
Identifiers are the names of values within Toy; to speed up execution, their "hash value" is computed at compile time and stored within them. Use this to access it, if needed.
|
||||||
|
|
||||||
|
This macro is only valid on `TOY_LITERAL_IDENTIFIER`.
|
||||||
|
!*/
|
||||||
#define TOY_HASH_I(lit) ((lit).as.identifier.hash)
|
#define TOY_HASH_I(lit) ((lit).as.identifier.hash)
|
||||||
|
|
||||||
|
/*!
|
||||||
|
### TOY_TYPE_PUSH_SUBTYPE(lit, subtype)
|
||||||
|
|
||||||
|
When building a complex type, such as the type of an array or dictionary, you may need to specify inner types. Use this to push a `subtype`. calling `Toy_freeLiteral()` on the outermost type should clean up all inner types, as expected.
|
||||||
|
|
||||||
|
This macro returns the index of the newly pushed value within it's parent.
|
||||||
|
|
||||||
|
This macro is only valid on `TOY_LITERAL_TYPE`, for both `type` and `subtype`.
|
||||||
|
!*/
|
||||||
#define TOY_TYPE_PUSH_SUBTYPE(lit, subtype) Toy_private_typePushSubtype(lit, subtype)
|
#define TOY_TYPE_PUSH_SUBTYPE(lit, subtype) Toy_private_typePushSubtype(lit, subtype)
|
||||||
|
|
||||||
|
/*!
|
||||||
|
### TOY_GET_OPAQUE_TAG(o)
|
||||||
|
|
||||||
|
Returns the value of the opaque `o`'s tag.
|
||||||
|
|
||||||
|
This macro is only valid on `TOY_LITERAL_OPAQUE`.
|
||||||
|
!*/
|
||||||
#define TOY_GET_OPAQUE_TAG(o) o.as.opaque.tag
|
#define TOY_GET_OPAQUE_TAG(o) o.as.opaque.tag
|
||||||
|
|
||||||
//BUGFIX: macros are not functions
|
/*!
|
||||||
TOY_API bool Toy_private_isTruthy(Toy_Literal x);
|
## Defined Functions
|
||||||
TOY_API Toy_Literal Toy_private_toStringLiteral(Toy_RefString* ptr);
|
!*/
|
||||||
TOY_API Toy_Literal Toy_private_toIdentifierLiteral(Toy_RefString* ptr);
|
|
||||||
TOY_API Toy_Literal* Toy_private_typePushSubtype(Toy_Literal* lit, Toy_Literal subtype);
|
|
||||||
|
|
||||||
//utils
|
/*!
|
||||||
|
### void Toy_freeLiteral(Toy_Literal literal)
|
||||||
|
|
||||||
|
This function frees the given literal's memory. Any internal pointers are now invalid.
|
||||||
|
|
||||||
|
This function should be called on EVERY literal when it is no longer needed, regardless of type.
|
||||||
|
!*/
|
||||||
|
TOY_API void Toy_freeLiteral(Toy_Literal literal);
|
||||||
|
|
||||||
|
/*!
|
||||||
|
### Toy_Literal Toy_copyLiteral(Toy_Literal original)
|
||||||
|
|
||||||
|
This function returns a copy of the given literal. Literals should never be copied without this function, as it handles a lot of internal memory allocations.
|
||||||
|
!*/
|
||||||
TOY_API Toy_Literal Toy_copyLiteral(Toy_Literal original);
|
TOY_API Toy_Literal Toy_copyLiteral(Toy_Literal original);
|
||||||
|
|
||||||
|
/*!
|
||||||
|
### bool Toy_literalsAreEqual(Toy_Literal lhs, Toy_Literal rhs)
|
||||||
|
|
||||||
|
This checks to see if two given literals are equal.
|
||||||
|
|
||||||
|
When an integer and a float are compared, the integer is cooerced into a float for the duration of the call.
|
||||||
|
|
||||||
|
Arrays or dictionaries are equal only if their keys and values all equal. Likewise, types only equal if all subtypes are equal, in order.
|
||||||
|
|
||||||
|
Functions and opaques are never equal to anything, while values with the type `TOY_LITERAL_ANY` are always equal.
|
||||||
|
!*/
|
||||||
TOY_API bool Toy_literalsAreEqual(Toy_Literal lhs, Toy_Literal rhs);
|
TOY_API bool Toy_literalsAreEqual(Toy_Literal lhs, Toy_Literal rhs);
|
||||||
|
|
||||||
|
/*!
|
||||||
|
### int Toy_hashLiteral(Toy_Literal lit)
|
||||||
|
|
||||||
|
This finds the hash of a literal, for various purposes. Different hashing algorithms are used for different types, and some types can't be hashed at all.
|
||||||
|
|
||||||
|
types that can't be hashed are
|
||||||
|
|
||||||
|
* all kinds of functions
|
||||||
|
* type
|
||||||
|
* opaque
|
||||||
|
* any
|
||||||
|
|
||||||
|
In the case of identifiers, their hashes are precomputed on creation and are stored within the literal.
|
||||||
|
!*/
|
||||||
TOY_API int Toy_hashLiteral(Toy_Literal lit);
|
TOY_API int Toy_hashLiteral(Toy_Literal lit);
|
||||||
|
|
||||||
//not thread-safe
|
/*!
|
||||||
|
### void Toy_printLiteral(Toy_Literal literal)
|
||||||
|
|
||||||
|
This wraps a call to `Toy_printLiteralCustom`, with a printf-stdout wrapper as `printFn`.
|
||||||
|
!*/
|
||||||
TOY_API void Toy_printLiteral(Toy_Literal literal);
|
TOY_API void Toy_printLiteral(Toy_Literal literal);
|
||||||
|
|
||||||
|
/*!
|
||||||
|
### void Toy_printLiteralCustom(Toy_Literal literal, PrintFn printFn)
|
||||||
|
|
||||||
|
This function passes the string representation of `literal` to `printFn`.
|
||||||
|
|
||||||
|
This function is not thread safe - due to the loopy and recursive nature of printing compound values, this function uses some globally persistent variables.
|
||||||
|
!*/
|
||||||
TOY_API void Toy_printLiteralCustom(Toy_Literal literal, Toy_PrintFn);
|
TOY_API void Toy_printLiteralCustom(Toy_Literal literal, Toy_PrintFn);
|
||||||
|
|
||||||
|
/*!
|
||||||
|
### bool Toy_private_isTruthy(Toy_Literal x)
|
||||||
|
|
||||||
|
Utilized by the `TOY_IS_TRUTHY` macro.
|
||||||
|
|
||||||
|
Private functions are not intended for general use.
|
||||||
|
!*/
|
||||||
|
TOY_API bool Toy_private_isTruthy(Toy_Literal x);
|
||||||
|
|
||||||
|
/*!
|
||||||
|
### bool Toy_private_toIdentifierLiteral(Toy_RefString* ptr)
|
||||||
|
|
||||||
|
Utilized by the `TOY_TO_IDENTIFIER_LITERAL` macro.
|
||||||
|
|
||||||
|
Private functions are not intended for general use.
|
||||||
|
!*/
|
||||||
|
TOY_API Toy_Literal Toy_private_toIdentifierLiteral(Toy_RefString* ptr);
|
||||||
|
|
||||||
|
/*!
|
||||||
|
### bool Toy_private_typePushSubtype(Toy_Literal* lit, Toy_Literal subtype)
|
||||||
|
|
||||||
|
Utilized by the `TOY_TYPE_PUSH_SUBTYPE` macro.
|
||||||
|
|
||||||
|
Private functions are not intended for general use.
|
||||||
|
!*/
|
||||||
|
TOY_API Toy_Literal* Toy_private_typePushSubtype(Toy_Literal* lit, Toy_Literal subtype);
|
||||||
|
|||||||
@@ -52,7 +52,7 @@ Toy_Literal Toy_popLiteralArray(Toy_LiteralArray* array) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
//find a literal in the array that matches the "literal" argument
|
//find a literal in the array that matches the "literal" argument
|
||||||
int Toy_findLiteralIndex(Toy_LiteralArray* array, Toy_Literal literal) {
|
int Toy_private_findLiteralIndex(Toy_LiteralArray* array, Toy_Literal literal) {
|
||||||
for (int i = 0; i < array->count; i++) {
|
for (int i = 0; i < array->count; i++) {
|
||||||
//not the same type
|
//not the same type
|
||||||
if (array->literals[i].type != literal.type) {
|
if (array->literals[i].type != literal.type) {
|
||||||
|
|||||||
@@ -1,5 +1,13 @@
|
|||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
|
/*!
|
||||||
|
# literal_array.h
|
||||||
|
|
||||||
|
This header defines the array structure, which manages a series of `Toy_Literal` instances in sequential memory. The array does not take ownership of given literals, instead it makes an internal copy.
|
||||||
|
|
||||||
|
The array type is one of two fundemental data structures used throughout Toy - the other is the dictionary.
|
||||||
|
!*/
|
||||||
|
|
||||||
#include "toy_common.h"
|
#include "toy_common.h"
|
||||||
|
|
||||||
#include "toy_literal.h"
|
#include "toy_literal.h"
|
||||||
@@ -10,13 +18,65 @@ typedef struct Toy_LiteralArray {
|
|||||||
int count;
|
int count;
|
||||||
} Toy_LiteralArray;
|
} Toy_LiteralArray;
|
||||||
|
|
||||||
|
/*!
|
||||||
|
## Defined Functions
|
||||||
|
!*/
|
||||||
|
|
||||||
|
/*
|
||||||
|
### void Toy_initLiteralArray(Toy_LiteralArray* array)
|
||||||
|
|
||||||
|
This function initializes a `Toy_LiteralArray` pointed to by `array`.
|
||||||
|
*/
|
||||||
TOY_API void Toy_initLiteralArray(Toy_LiteralArray* array);
|
TOY_API void Toy_initLiteralArray(Toy_LiteralArray* array);
|
||||||
|
|
||||||
|
/*!
|
||||||
|
### void Toy_freeLiteralArray(Toy_LiteralArray* array)
|
||||||
|
|
||||||
|
This function frees a `Toy_LiteralArray` pointed to by `array`. Every literal within is passed to `Toy_freeLiteral()` before its memory is released.
|
||||||
|
!*/
|
||||||
TOY_API void Toy_freeLiteralArray(Toy_LiteralArray* array);
|
TOY_API void Toy_freeLiteralArray(Toy_LiteralArray* array);
|
||||||
|
|
||||||
|
/*!
|
||||||
|
### int Toy_pushLiteralArray(Toy_LiteralArray* array, Toy_Literal literal)
|
||||||
|
|
||||||
|
This function adds a new `literal` to the end of the `array`, growing the array's internal buffer if needed.
|
||||||
|
|
||||||
|
This function returns the index of the inserted value.
|
||||||
|
!*/
|
||||||
TOY_API int Toy_pushLiteralArray(Toy_LiteralArray* array, Toy_Literal literal);
|
TOY_API int Toy_pushLiteralArray(Toy_LiteralArray* array, Toy_Literal literal);
|
||||||
|
|
||||||
|
/*!
|
||||||
|
### Toy_Literal Toy_popLiteralArray(Toy_LiteralArray* array)
|
||||||
|
|
||||||
|
This function removes the literal at the end of the `array`, and returns it.
|
||||||
|
!*/
|
||||||
TOY_API Toy_Literal Toy_popLiteralArray(Toy_LiteralArray* array);
|
TOY_API Toy_Literal Toy_popLiteralArray(Toy_LiteralArray* array);
|
||||||
|
|
||||||
|
/*!
|
||||||
|
### bool Toy_setLiteralArray(Toy_LiteralArray* array, Toy_Literal index, Toy_Literal value)
|
||||||
|
|
||||||
|
This function frees the literal at the position represented by the integer literal `index`, and stores `value` in its place.
|
||||||
|
|
||||||
|
This function returns true on success, otherwise it returns false.
|
||||||
|
!*/
|
||||||
TOY_API bool Toy_setLiteralArray(Toy_LiteralArray* array, Toy_Literal index, Toy_Literal value);
|
TOY_API bool Toy_setLiteralArray(Toy_LiteralArray* array, Toy_Literal index, Toy_Literal value);
|
||||||
|
|
||||||
|
/*!
|
||||||
|
### Toy_Literal Toy_getLiteralArray(Toy_LiteralArray* array, Toy_Literal index)
|
||||||
|
|
||||||
|
This function returns the literal at the position represented by the integer literal `index`, or returns a null literal if none is found.
|
||||||
|
|
||||||
|
If `index` is not an integer literal or is out of bounds, this function returns a null literal.
|
||||||
|
!*/
|
||||||
TOY_API Toy_Literal Toy_getLiteralArray(Toy_LiteralArray* array, Toy_Literal index);
|
TOY_API Toy_Literal Toy_getLiteralArray(Toy_LiteralArray* array, Toy_Literal index);
|
||||||
|
|
||||||
int Toy_findLiteralIndex(Toy_LiteralArray* array, Toy_Literal literal);
|
/*!
|
||||||
|
### int Toy_private_findLiteralIndex(Toy_LiteralArray* array, Toy_Literal literal)
|
||||||
|
|
||||||
//TODO: add a function to get the capacity & count
|
This function scans through the array, and returns the index of the first element that matches the given `literal`, otherwise it returns -1.
|
||||||
|
|
||||||
|
Private functions are not intended for general use.
|
||||||
|
!*/
|
||||||
|
int Toy_private_findLiteralIndex(Toy_LiteralArray* array, Toy_Literal literal);
|
||||||
|
|
||||||
|
//TODO: add a function to get the capacity & count
|
||||||
|
|||||||
@@ -131,7 +131,6 @@ static void freeEntryArray(Toy_private_dictionary_entry* array, int capacity) {
|
|||||||
|
|
||||||
//exposed functions
|
//exposed functions
|
||||||
void Toy_initLiteralDictionary(Toy_LiteralDictionary* dictionary) {
|
void Toy_initLiteralDictionary(Toy_LiteralDictionary* dictionary) {
|
||||||
//HACK: because modulo by 0 is undefined, set the capacity to a non-zero value (and allocate the arrays)
|
|
||||||
dictionary->entries = NULL;
|
dictionary->entries = NULL;
|
||||||
dictionary->capacity = 0;
|
dictionary->capacity = 0;
|
||||||
dictionary->contains = 0;
|
dictionary->contains = 0;
|
||||||
|
|||||||
@@ -1,9 +1,31 @@
|
|||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
|
/*!
|
||||||
|
# toy_literal_dictionary.h
|
||||||
|
|
||||||
|
This header defines the dictionary structure (as well as the private entry structure), which manages a series of `Toy_Literal` instances stored in a key-value hash map. The dictionary does not take ownership of given literals, instead it makes an internal copy.
|
||||||
|
|
||||||
|
The dictionary type is one of two fundemental data structures used throughout Toy - the other is the array.
|
||||||
|
!*/
|
||||||
|
|
||||||
#include "toy_common.h"
|
#include "toy_common.h"
|
||||||
|
|
||||||
#include "toy_literal.h"
|
#include "toy_literal.h"
|
||||||
|
|
||||||
|
/*!
|
||||||
|
## Defined Macros
|
||||||
|
!*/
|
||||||
|
|
||||||
|
/*!
|
||||||
|
### TOY_DICTIONARY_MAX_LOAD
|
||||||
|
|
||||||
|
If the contents of a dictionary exceeds this percentage of it's capacity, then a new buffer is created, the old contents are copied over one-by-one, and the original buffer is freed.
|
||||||
|
|
||||||
|
Since this process can be memory and time intensive, a configurable macro is used to allow for fine-grained control across the lang.
|
||||||
|
|
||||||
|
The current default value is `0.75`, representing 75% capacity.
|
||||||
|
!*/
|
||||||
|
|
||||||
//TODO: benchmark this
|
//TODO: benchmark this
|
||||||
#define TOY_DICTIONARY_MAX_LOAD 0.75
|
#define TOY_DICTIONARY_MAX_LOAD 0.75
|
||||||
|
|
||||||
@@ -19,11 +41,56 @@ typedef struct Toy_LiteralDictionary {
|
|||||||
int contains; //count + tombstones, for internal use
|
int contains; //count + tombstones, for internal use
|
||||||
} Toy_LiteralDictionary;
|
} Toy_LiteralDictionary;
|
||||||
|
|
||||||
|
/*!
|
||||||
|
## Defined Functions
|
||||||
|
!*/
|
||||||
|
|
||||||
|
/*!
|
||||||
|
### void Toy_initLiteralDictionary(Toy_LiteralDictionary* dictionary)
|
||||||
|
|
||||||
|
This function initializes the `Toy_LiteralDictionary` pointed to by `dictionary`.
|
||||||
|
!*/
|
||||||
TOY_API void Toy_initLiteralDictionary(Toy_LiteralDictionary* dictionary);
|
TOY_API void Toy_initLiteralDictionary(Toy_LiteralDictionary* dictionary);
|
||||||
|
|
||||||
|
/*!
|
||||||
|
### void Toy_freeLiteralDictionary(Toy_LiteralDictionary* dictionary)
|
||||||
|
|
||||||
|
This function frees a `Toy_LiteralDictionary` pointed to by `dictionary`. Every literal within is passed to `Toy_freeLiteral()` before its memory is released.
|
||||||
|
!*/
|
||||||
TOY_API void Toy_freeLiteralDictionary(Toy_LiteralDictionary* dictionary);
|
TOY_API void Toy_freeLiteralDictionary(Toy_LiteralDictionary* dictionary);
|
||||||
|
|
||||||
|
/*!
|
||||||
|
### void Toy_setLiteralDictionary(Toy_LiteralDictionary* dictionary, Toy_Literal key, Toy_Literal value)
|
||||||
|
|
||||||
|
This function inserts the given key-value pair of literals into `dictionary`, creating it if it doesn't exist, or freeing and overwriting it if `key` is already present. This function may also expand the memory buffer if needed.
|
||||||
|
|
||||||
|
When expanding the memory buffer, a full copy of the existing dictionary's contents is created - this can be memory intensive.
|
||||||
|
|
||||||
|
Literal functions and opaques cannot be used as keys.
|
||||||
|
!*/
|
||||||
TOY_API void Toy_setLiteralDictionary(Toy_LiteralDictionary* dictionary, Toy_Literal key, Toy_Literal value);
|
TOY_API void Toy_setLiteralDictionary(Toy_LiteralDictionary* dictionary, Toy_Literal key, Toy_Literal value);
|
||||||
|
|
||||||
|
/*!
|
||||||
|
### Toy_Literal Toy_getLiteralDictionary(Toy_LiteralDictionary* dictionary, Toy_Literal key)
|
||||||
|
|
||||||
|
This function returns the value of the literal within `dictionary` identified by `key`, or a null literal if it doesn't exist.
|
||||||
|
|
||||||
|
Literal functions and opaques cannot be used as keys.
|
||||||
|
!*/
|
||||||
TOY_API Toy_Literal Toy_getLiteralDictionary(Toy_LiteralDictionary* dictionary, Toy_Literal key);
|
TOY_API Toy_Literal Toy_getLiteralDictionary(Toy_LiteralDictionary* dictionary, Toy_Literal key);
|
||||||
|
|
||||||
|
/*!
|
||||||
|
### void Toy_removeLiteralDictionary(Toy_LiteralDictionary* dictionary, Toy_Literal key)
|
||||||
|
|
||||||
|
This function removes the key-value pair of literals from `dictionary` identified by `key`, if it exists.
|
||||||
|
|
||||||
|
Literal functions and opaques cannot be used as keys.
|
||||||
|
!*/
|
||||||
TOY_API void Toy_removeLiteralDictionary(Toy_LiteralDictionary* dictionary, Toy_Literal key);
|
TOY_API void Toy_removeLiteralDictionary(Toy_LiteralDictionary* dictionary, Toy_Literal key);
|
||||||
|
|
||||||
|
/*!
|
||||||
|
### bool Toy_existsLiteralDictionary(Toy_LiteralDictionary* dictionary, Toy_Literal key)
|
||||||
|
|
||||||
|
This function returns true if the key-value pair identified by `key` exists within `dictionary`, otherwise it returns false.
|
||||||
|
!*/
|
||||||
TOY_API bool Toy_existsLiteralDictionary(Toy_LiteralDictionary* dictionary, Toy_Literal key);
|
TOY_API bool Toy_existsLiteralDictionary(Toy_LiteralDictionary* dictionary, Toy_Literal key);
|
||||||
|
|||||||
@@ -1,5 +1,6 @@
|
|||||||
#include "toy_memory.h"
|
#include "toy_memory.h"
|
||||||
#include "toy_refstring.h"
|
#include "toy_refstring.h"
|
||||||
|
#include "toy_reffunction.h"
|
||||||
|
|
||||||
#include "toy_console_colors.h"
|
#include "toy_console_colors.h"
|
||||||
|
|
||||||
@@ -49,4 +50,5 @@ void Toy_setMemoryAllocator(Toy_MemoryAllocatorFn fn) {
|
|||||||
|
|
||||||
allocator = fn;
|
allocator = fn;
|
||||||
Toy_setRefStringAllocatorFn(fn);
|
Toy_setRefStringAllocatorFn(fn);
|
||||||
|
Toy_setRefFunctionAllocatorFn(fn);
|
||||||
}
|
}
|
||||||
|
|||||||
+94
-3
@@ -1,21 +1,112 @@
|
|||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
|
/*!
|
||||||
|
# toy_memory.h
|
||||||
|
|
||||||
|
This header defines all of the memory management utilities. Any and all heap-based memory management goes through these utilities.
|
||||||
|
|
||||||
|
A default memory allocator function is used internally, but it can be overwritten for diagnostic and platform related purposes.
|
||||||
|
!*/
|
||||||
|
|
||||||
#include "toy_common.h"
|
#include "toy_common.h"
|
||||||
|
|
||||||
|
/*!
|
||||||
|
## Defined Macros
|
||||||
|
!*/
|
||||||
|
|
||||||
|
/*!
|
||||||
|
### TOY_GROW_CAPACITY(capacity)
|
||||||
|
|
||||||
|
This macro calculates, in place, what size of memory should be allocated based on the previous size.
|
||||||
|
!*/
|
||||||
#define TOY_GROW_CAPACITY(capacity) ((capacity) < 8 ? 8 : (capacity) * 2)
|
#define TOY_GROW_CAPACITY(capacity) ((capacity) < 8 ? 8 : (capacity) * 2)
|
||||||
|
|
||||||
|
/*!
|
||||||
|
### TOY_GROW_CAPACITY_FAST(capacity)
|
||||||
|
|
||||||
|
This macro calculates, in place, what size of memory should be allocated based on the previous size. It grows faster than `TOY_GROW_CAPACITY`.
|
||||||
|
!*/
|
||||||
#define TOY_GROW_CAPACITY_FAST(capacity) ((capacity) < 32 ? 32 : (capacity) * 2)
|
#define TOY_GROW_CAPACITY_FAST(capacity) ((capacity) < 32 ? 32 : (capacity) * 2)
|
||||||
|
|
||||||
|
/*
|
||||||
|
### TOY_ALLOCATE(type, count)
|
||||||
|
|
||||||
|
This macro wraps `Toy_reallocate()`, which itself calls the allocator function. `type` is the type that will be allocated, and `count` is the number which will be needed (usually calculated with `TOY_GROW_CAPACITY`).
|
||||||
|
|
||||||
|
This returns a pointer of `type`.
|
||||||
|
*/
|
||||||
#define TOY_ALLOCATE(type, count) ((type*)Toy_reallocate(NULL, 0, sizeof(type) * (count)))
|
#define TOY_ALLOCATE(type, count) ((type*)Toy_reallocate(NULL, 0, sizeof(type) * (count)))
|
||||||
|
|
||||||
|
/*!
|
||||||
|
### TOY_FREE(type, pointer)
|
||||||
|
|
||||||
|
This macro wraps `Toy_reallocate()`, which itself calls the allocator function. `type` is the type that will be freed, and `pointer` is to what is being freed. This should only be used when a single element has been allocated, as opposed to an array.
|
||||||
|
!*/
|
||||||
#define TOY_FREE(type, pointer) Toy_reallocate(pointer, sizeof(type), 0)
|
#define TOY_FREE(type, pointer) Toy_reallocate(pointer, sizeof(type), 0)
|
||||||
|
|
||||||
|
/*!
|
||||||
|
### TOY_FREE_ARRAY(type, pointer, oldCount)
|
||||||
|
|
||||||
|
This macro wraps `Toy_reallocate()`, which itself calls the allocator function. `type` is the type that will be freed, `pointer` is a reference to what is being freed, and `oldCount` is the size of the array being freed. This should only be used when an array has been allocated, as opposed to a single element.
|
||||||
|
!*/
|
||||||
#define TOY_FREE_ARRAY(type, pointer, oldCount) Toy_reallocate((type*)pointer, sizeof(type) * (oldCount), 0)
|
#define TOY_FREE_ARRAY(type, pointer, oldCount) Toy_reallocate((type*)pointer, sizeof(type) * (oldCount), 0)
|
||||||
|
|
||||||
|
/*!
|
||||||
|
### TOY_GROW_ARRAY(type, pointer, oldCount, count)
|
||||||
|
|
||||||
|
This macro wraps `Toy_reallocate()`, which itself calls the allocator function. `type` is the type that is being operated on, `pointer` is what is being resized, `oldCount` is the previous size of the array and `count` is the new size of the array (usually calculated with `TOY_GROW_CAPACITY`).
|
||||||
|
|
||||||
|
This returns a pointer of `type`.
|
||||||
|
!*/
|
||||||
#define TOY_GROW_ARRAY(type, pointer, oldCount, count) (type*)Toy_reallocate((type*)pointer, sizeof(type) * (oldCount), sizeof(type) * (count))
|
#define TOY_GROW_ARRAY(type, pointer, oldCount, count) (type*)Toy_reallocate((type*)pointer, sizeof(type) * (oldCount), sizeof(type) * (count))
|
||||||
|
|
||||||
|
/*!
|
||||||
|
### TOY_SHRINK_ARRAY(type, pointer, oldCount, count)
|
||||||
|
|
||||||
|
This macro wraps `Toy_reallocate()`, which itself calls the allocator function. `type` is the type that is being operated on, `pointer` is what is being resized, `oldCount` is the previous size of the array and `count` is the new size of the array.
|
||||||
|
|
||||||
|
This returns a pointer of `type`.
|
||||||
|
!*/
|
||||||
#define TOY_SHRINK_ARRAY(type, pointer, oldCount, count) (type*)Toy_reallocate((type*)pointer, sizeof(type) * (oldCount), sizeof(type) * (count))
|
#define TOY_SHRINK_ARRAY(type, pointer, oldCount, count) (type*)Toy_reallocate((type*)pointer, sizeof(type) * (oldCount), sizeof(type) * (count))
|
||||||
|
|
||||||
//implementation details
|
/*!
|
||||||
|
## Defined Interfaces
|
||||||
|
!*/
|
||||||
|
|
||||||
|
/*!
|
||||||
|
### typedef void* (*Toy_MemoryAllocatorFn)(void* pointer, size_t oldSize, size_t newSize)
|
||||||
|
|
||||||
|
This function interface is used for defining any memory allocator functions.
|
||||||
|
|
||||||
|
Any and all memory allocator functions should:
|
||||||
|
|
||||||
|
* Take a `pointer` to a previously allocated block of memory, or `NULL`
|
||||||
|
* Take the `oldSize`, which is the previous size of the `pointer` allocated, in bytes (`oldSize` can be 0)
|
||||||
|
* Take the `newSize`, which is the new size of the buffer to be allocaated, in bytes (`newSize` can be 0)
|
||||||
|
* Return the newly allocated buffer, or `NULL` if `newSize` is zero
|
||||||
|
* Return `NULL` on error
|
||||||
|
!*/
|
||||||
|
|
||||||
|
typedef void* (*Toy_MemoryAllocatorFn)(void* pointer, size_t oldSize, size_t newSize);
|
||||||
|
|
||||||
|
/*!
|
||||||
|
## Defined Functions
|
||||||
|
!*/
|
||||||
|
|
||||||
|
/*!
|
||||||
|
### TOY_API void* Toy_reallocate(void* pointer, size_t oldSize, size_t newSize)
|
||||||
|
|
||||||
|
This function shouldn't be called directly. Instead, use one of the given macros.
|
||||||
|
|
||||||
|
This function wraps a call to the internal assigned memory allocator.
|
||||||
|
!*/
|
||||||
TOY_API void* Toy_reallocate(void* pointer, size_t oldSize, size_t newSize);
|
TOY_API void* Toy_reallocate(void* pointer, size_t oldSize, size_t newSize);
|
||||||
|
|
||||||
//assign the memory allocator
|
/*!
|
||||||
typedef void* (*Toy_MemoryAllocatorFn)(void* pointer, size_t oldSize, size_t newSize);
|
### void Toy_setMemoryAllocator(Toy_MemoryAllocatorFn)
|
||||||
|
|
||||||
|
This function sets the memory allocator, replacing the default memory allocator.
|
||||||
|
|
||||||
|
This function also overwrites any given refstring and reffunction memory allocators, see [toy_refstring.h](toy_refstring_h.md).
|
||||||
|
!*/
|
||||||
TOY_API void Toy_setMemoryAllocator(Toy_MemoryAllocatorFn);
|
TOY_API void Toy_setMemoryAllocator(Toy_MemoryAllocatorFn);
|
||||||
|
|||||||
@@ -86,5 +86,9 @@ typedef enum Toy_Opcode {
|
|||||||
TOY_OP_FN_END, //different from SECTION_END
|
TOY_OP_FN_END, //different from SECTION_END
|
||||||
TOY_OP_SECTION_END = 255,
|
TOY_OP_SECTION_END = 255,
|
||||||
//TODO: add more
|
//TODO: add more
|
||||||
|
|
||||||
|
//prefix & postfix signals (used internally)
|
||||||
|
TOY_OP_PREFIX,
|
||||||
|
TOY_OP_POSTFIX,
|
||||||
} Toy_Opcode;
|
} Toy_Opcode;
|
||||||
|
|
||||||
|
|||||||
+30
-21
@@ -119,6 +119,7 @@ ParseRule parseRules[];
|
|||||||
static void declaration(Toy_Parser* parser, Toy_ASTNode** nodeHandle);
|
static void declaration(Toy_Parser* parser, Toy_ASTNode** nodeHandle);
|
||||||
static void parsePrecedence(Toy_Parser* parser, Toy_ASTNode** nodeHandle, PrecedenceRule rule);
|
static void parsePrecedence(Toy_Parser* parser, Toy_ASTNode** nodeHandle, PrecedenceRule rule);
|
||||||
static Toy_Literal readTypeToLiteral(Toy_Parser* parser);
|
static Toy_Literal readTypeToLiteral(Toy_Parser* parser);
|
||||||
|
static void varDecl(Toy_Parser* parser, Toy_ASTNode** nodeHandle);
|
||||||
|
|
||||||
//TODO: resolve the messy order of these
|
//TODO: resolve the messy order of these
|
||||||
//the expression rules
|
//the expression rules
|
||||||
@@ -627,6 +628,14 @@ static Toy_Opcode castingPrefix(Toy_Parser* parser, Toy_ASTNode** nodeHandle) {
|
|||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
|
|
||||||
|
//BUGFIX: handle this here, and not in castingPrefix, so "any" can be recognized as a type properly
|
||||||
|
case TOY_TOKEN_ANY: {
|
||||||
|
Toy_Literal literal = TOY_TO_TYPE_LITERAL(TOY_LITERAL_ANY, false);
|
||||||
|
Toy_emitASTNodeLiteral(nodeHandle, literal);
|
||||||
|
Toy_freeLiteral(literal);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
|
||||||
default:
|
default:
|
||||||
error(parser, parser->previous, "Unexpected token passed to casting precedence rule");
|
error(parser, parser->previous, "Unexpected token passed to casting precedence rule");
|
||||||
return TOY_OP_EOF;
|
return TOY_OP_EOF;
|
||||||
@@ -684,7 +693,7 @@ static Toy_Opcode incrementPrefix(Toy_Parser* parser, Toy_ASTNode** nodeHandle)
|
|||||||
|
|
||||||
Toy_freeASTNode(tmpNode);
|
Toy_freeASTNode(tmpNode);
|
||||||
|
|
||||||
return TOY_OP_EOF;
|
return TOY_OP_PREFIX;
|
||||||
}
|
}
|
||||||
|
|
||||||
static Toy_Opcode incrementInfix(Toy_Parser* parser, Toy_ASTNode** nodeHandle) {
|
static Toy_Opcode incrementInfix(Toy_Parser* parser, Toy_ASTNode** nodeHandle) {
|
||||||
@@ -701,7 +710,7 @@ static Toy_Opcode incrementInfix(Toy_Parser* parser, Toy_ASTNode** nodeHandle) {
|
|||||||
|
|
||||||
Toy_freeASTNode(tmpNode);
|
Toy_freeASTNode(tmpNode);
|
||||||
|
|
||||||
return TOY_OP_EOF;
|
return TOY_OP_POSTFIX;
|
||||||
}
|
}
|
||||||
|
|
||||||
static Toy_Opcode decrementPrefix(Toy_Parser* parser, Toy_ASTNode** nodeHandle) {
|
static Toy_Opcode decrementPrefix(Toy_Parser* parser, Toy_ASTNode** nodeHandle) {
|
||||||
@@ -718,7 +727,7 @@ static Toy_Opcode decrementPrefix(Toy_Parser* parser, Toy_ASTNode** nodeHandle)
|
|||||||
|
|
||||||
Toy_freeASTNode(tmpNode);
|
Toy_freeASTNode(tmpNode);
|
||||||
|
|
||||||
return TOY_OP_EOF;
|
return TOY_OP_PREFIX;
|
||||||
}
|
}
|
||||||
|
|
||||||
static Toy_Opcode decrementInfix(Toy_Parser* parser, Toy_ASTNode** nodeHandle) {
|
static Toy_Opcode decrementInfix(Toy_Parser* parser, Toy_ASTNode** nodeHandle) {
|
||||||
@@ -735,29 +744,16 @@ static Toy_Opcode decrementInfix(Toy_Parser* parser, Toy_ASTNode** nodeHandle) {
|
|||||||
|
|
||||||
Toy_freeASTNode(tmpNode);
|
Toy_freeASTNode(tmpNode);
|
||||||
|
|
||||||
return TOY_OP_EOF;
|
return TOY_OP_POSTFIX;
|
||||||
}
|
}
|
||||||
|
|
||||||
static Toy_Opcode fnCall(Toy_Parser* parser, Toy_ASTNode** nodeHandle) {
|
static Toy_Opcode fnCall(Toy_Parser* parser, Toy_ASTNode** nodeHandle) {
|
||||||
//wait - is the previous token a type? this should be casting instead
|
//wait - is the previous token a type? this should be casting instead
|
||||||
if (parser->previous.type >= TOY_TOKEN_NULL && parser->previous.type <= TOY_TOKEN_ANY) {
|
if (parser->previous.type >= TOY_TOKEN_NULL && parser->previous.type <= TOY_TOKEN_ANY) {
|
||||||
//casting type
|
|
||||||
Toy_ASTNode* lhsNode = NULL;
|
|
||||||
castingPrefix(parser, &lhsNode);
|
|
||||||
advance(parser);
|
|
||||||
|
|
||||||
//casting value
|
//casting value
|
||||||
Toy_ASTNode* rhsNode = NULL;
|
parsePrecedence(parser, nodeHandle, PREC_CALL);
|
||||||
grouping(parser, &rhsNode);
|
|
||||||
|
|
||||||
//emit the cast node
|
return TOY_OP_TYPE_CAST; //opcode value
|
||||||
|
|
||||||
Toy_emitASTNodeBinary(&lhsNode, rhsNode, TOY_OP_TYPE_CAST);
|
|
||||||
|
|
||||||
//pass it off to the caller
|
|
||||||
*nodeHandle = lhsNode;
|
|
||||||
|
|
||||||
return TOY_OP_GROUPING_BEGIN; //dummy value
|
|
||||||
}
|
}
|
||||||
|
|
||||||
advance(parser); //skip the left paren
|
advance(parser); //skip the left paren
|
||||||
@@ -939,7 +935,7 @@ ParseRule parseRules[] = { //must match the token types
|
|||||||
{NULL, NULL, PREC_NONE},// TOKEN_DICTIONARY,
|
{NULL, NULL, PREC_NONE},// TOKEN_DICTIONARY,
|
||||||
{NULL, NULL, PREC_NONE},// TOKEN_FUNCTION,
|
{NULL, NULL, PREC_NONE},// TOKEN_FUNCTION,
|
||||||
{NULL, NULL, PREC_NONE},// TOKEN_OPAQUE,
|
{NULL, NULL, PREC_NONE},// TOKEN_OPAQUE,
|
||||||
{NULL, NULL, PREC_NONE},// TOKEN_ANY,
|
{castingPrefix, NULL, PREC_CALL},// TOKEN_ANY,
|
||||||
|
|
||||||
//keywords and reserved words
|
//keywords and reserved words
|
||||||
{NULL, NULL, PREC_NONE},// TOKEN_AS,
|
{NULL, NULL, PREC_NONE},// TOKEN_AS,
|
||||||
@@ -1278,6 +1274,13 @@ static void parsePrecedence(Toy_Parser* parser, Toy_ASTNode** nodeHandle, Preced
|
|||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
//BUGFIX: keep going, don't skip out on a postfix
|
||||||
|
if (opcode == TOY_OP_PREFIX || opcode == TOY_OP_POSTFIX) {
|
||||||
|
Toy_freeASTNode(*nodeHandle);
|
||||||
|
*nodeHandle = rhsNode;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
Toy_emitASTNodeBinary(nodeHandle, rhsNode, opcode);
|
Toy_emitASTNodeBinary(nodeHandle, rhsNode, opcode);
|
||||||
|
|
||||||
//optimise away the constants
|
//optimise away the constants
|
||||||
@@ -1397,7 +1400,13 @@ static void forStmt(Toy_Parser* parser, Toy_ASTNode** nodeHandle) {
|
|||||||
|
|
||||||
//check the pre-clause
|
//check the pre-clause
|
||||||
if (parser->current.type != TOY_TOKEN_SEMICOLON) {
|
if (parser->current.type != TOY_TOKEN_SEMICOLON) {
|
||||||
declaration(parser, &preClause); //allow defining variables in the pre-clause
|
//allow defining variables in the pre-clause
|
||||||
|
if (match(parser, TOY_TOKEN_VAR)) {
|
||||||
|
varDecl(parser, &preClause);
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
parsePrecedence(parser, &preClause, PREC_ASSIGNMENT);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
consume(parser, TOY_TOKEN_SEMICOLON, "Expected ';' after empty declaration of for clause");
|
consume(parser, TOY_TOKEN_SEMICOLON, "Expected ';' after empty declaration of for clause");
|
||||||
|
|||||||
+84
-1
@@ -1,10 +1,62 @@
|
|||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
|
/*!
|
||||||
|
# toy_parser.h
|
||||||
|
|
||||||
|
This header defines the parser structure which, after being initialized with a lexer produces a series of abstract syntax trees to be passed to the compiler. The following is a utility function provided by [repl_tools.h](repl_tools_h.md), demonstrating how to use the parser.
|
||||||
|
|
||||||
|
```c
|
||||||
|
//generate bytecode from a given string
|
||||||
|
const unsigned char* Toy_compileString(const char* source, size_t* size) {
|
||||||
|
//declare the relevant instances
|
||||||
|
Toy_Lexer lexer;
|
||||||
|
Toy_Parser parser;
|
||||||
|
Toy_Compiler compiler;
|
||||||
|
|
||||||
|
//initialize each of them
|
||||||
|
Toy_initLexer(&lexer, source);
|
||||||
|
Toy_initParser(&parser, &lexer);
|
||||||
|
Toy_initCompiler(&compiler);
|
||||||
|
|
||||||
|
//when the parser returns NULL, it is finished
|
||||||
|
Toy_ASTNode* node = Toy_scanParser(&parser);
|
||||||
|
while(node != NULL) {
|
||||||
|
//if the parser returns an error node, clean up and exit gracefully
|
||||||
|
if (node->type == TOY_AST_NODE_ERROR) {
|
||||||
|
Toy_freeASTNode(node);
|
||||||
|
Toy_freeCompiler(&compiler);
|
||||||
|
Toy_freeParser(&parser);
|
||||||
|
//no need to clean the lexer
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
//write the node to the compiler
|
||||||
|
Toy_writeCompiler(&compiler, node);
|
||||||
|
Toy_freeASTNode(node);
|
||||||
|
|
||||||
|
//grab the next node
|
||||||
|
node = Toy_scanParser(&parser);
|
||||||
|
}
|
||||||
|
|
||||||
|
//get the bytecode to be returned
|
||||||
|
const unsigned char* tb = Toy_collateCompiler(&compiler, size);
|
||||||
|
|
||||||
|
//cleanup
|
||||||
|
Toy_freeCompiler(&compiler);
|
||||||
|
Toy_freeParser(&parser);
|
||||||
|
//no need to clean the lexer
|
||||||
|
|
||||||
|
//finally
|
||||||
|
return tb;
|
||||||
|
}
|
||||||
|
```
|
||||||
|
!*/
|
||||||
|
|
||||||
#include "toy_common.h"
|
#include "toy_common.h"
|
||||||
#include "toy_lexer.h"
|
#include "toy_lexer.h"
|
||||||
#include "toy_ast_node.h"
|
#include "toy_ast_node.h"
|
||||||
|
|
||||||
//DOCS: parsers are bound to a lexer, and turn the outputted tokens into AST nodes
|
//Parsers are bound to a lexer, and turn the outputted tokens into AST nodes
|
||||||
typedef struct {
|
typedef struct {
|
||||||
Toy_Lexer* lexer;
|
Toy_Lexer* lexer;
|
||||||
bool error; //I've had an error
|
bool error; //I've had an error
|
||||||
@@ -15,6 +67,37 @@ typedef struct {
|
|||||||
Toy_Token previous;
|
Toy_Token previous;
|
||||||
} Toy_Parser;
|
} Toy_Parser;
|
||||||
|
|
||||||
|
/*!
|
||||||
|
## Defined Functions
|
||||||
|
!*/
|
||||||
|
|
||||||
|
/*!
|
||||||
|
### void Toy_initParser(Toy_Parser* parser, Toy_Lexer* lexer)
|
||||||
|
|
||||||
|
This function initializes a `Toy_Parser`, binding the given `Toy_Lexer` to it.
|
||||||
|
!*/
|
||||||
TOY_API void Toy_initParser(Toy_Parser* parser, Toy_Lexer* lexer);
|
TOY_API void Toy_initParser(Toy_Parser* parser, Toy_Lexer* lexer);
|
||||||
|
|
||||||
|
/*!
|
||||||
|
### void Toy_freeParser(Toy_Parser* parser)
|
||||||
|
|
||||||
|
This function frees a `Toy_Parser` once its task is completed.
|
||||||
|
!*/
|
||||||
TOY_API void Toy_freeParser(Toy_Parser* parser);
|
TOY_API void Toy_freeParser(Toy_Parser* parser);
|
||||||
|
|
||||||
|
/*!
|
||||||
|
### Toy_ASTNode* Toy_scanParser(Toy_Parser* parser)
|
||||||
|
|
||||||
|
This function returns an abstract syntax tree representing part of the program, or an error node. The abstract syntax tree must be passed to `Toy_writeCompiler()` and/or `Toy_freeASTNode()`.
|
||||||
|
|
||||||
|
This function should be called repeatedly until it returns `NULL`, indicating the end of the program.
|
||||||
|
!*/
|
||||||
TOY_API Toy_ASTNode* Toy_scanParser(Toy_Parser* parser);
|
TOY_API Toy_ASTNode* Toy_scanParser(Toy_Parser* parser);
|
||||||
|
|
||||||
|
/*!
|
||||||
|
### void Toy_freeASTNode(Toy_ASTNode* node)
|
||||||
|
|
||||||
|
This function cleans up any valid instance of `Toy_ASTNode` pointer passed to it. It is most commonly used to clean up the values returned by `Toy_scanParser`, after they have been passsed to `Toy_writeCompiler`, or when the node is an error node.
|
||||||
|
|
||||||
|
Note: this function is *actually* defined in toy_ast_node.h, but documented here, because this is where it matters most.
|
||||||
|
!*/
|
||||||
|
|||||||
@@ -0,0 +1,55 @@
|
|||||||
|
#include "toy_reffunction.h"
|
||||||
|
|
||||||
|
#include <string.h>
|
||||||
|
|
||||||
|
//memory allocation
|
||||||
|
extern void* Toy_private_defaultMemoryAllocator(void* pointer, size_t oldSize, size_t newSize);
|
||||||
|
static Toy_RefFunctionAllocatorFn allocate = Toy_private_defaultMemoryAllocator;
|
||||||
|
|
||||||
|
void Toy_setRefFunctionAllocatorFn(Toy_RefFunctionAllocatorFn allocator) {
|
||||||
|
allocate = allocator;
|
||||||
|
}
|
||||||
|
|
||||||
|
//API
|
||||||
|
Toy_RefFunction* Toy_createRefFunction(const void* data, size_t length) {
|
||||||
|
//allocate the memory area (including metadata space)
|
||||||
|
Toy_RefFunction* refFunction = allocate(NULL, 0, sizeof(size_t) + sizeof(int) + sizeof(char) * length);
|
||||||
|
|
||||||
|
if (refFunction == NULL) {
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
//set the data
|
||||||
|
refFunction->refCount = 1;
|
||||||
|
refFunction->length = length;
|
||||||
|
memcpy(refFunction->data, data, refFunction->length);
|
||||||
|
|
||||||
|
return refFunction;
|
||||||
|
}
|
||||||
|
|
||||||
|
void Toy_deleteRefFunction(Toy_RefFunction* refFunction) {
|
||||||
|
//decrement, then check
|
||||||
|
refFunction->refCount--;
|
||||||
|
if (refFunction->refCount <= 0) {
|
||||||
|
allocate(refFunction, sizeof(size_t) + sizeof(int) + sizeof(char) * (refFunction->length + 1), 0);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
int Toy_countRefFunction(Toy_RefFunction* refFunction) {
|
||||||
|
return refFunction->refCount;
|
||||||
|
}
|
||||||
|
|
||||||
|
size_t Toy_lengthRefFunction(Toy_RefFunction* refFunction) {
|
||||||
|
return refFunction->length;
|
||||||
|
}
|
||||||
|
|
||||||
|
Toy_RefFunction* Toy_copyRefFunction(Toy_RefFunction* refFunction) {
|
||||||
|
//Cheaty McCheater Face
|
||||||
|
refFunction->refCount++;
|
||||||
|
return refFunction;
|
||||||
|
}
|
||||||
|
|
||||||
|
Toy_RefFunction* Toy_deepCopyRefFunction(Toy_RefFunction* refFunction) {
|
||||||
|
//create a new function, with a new refCount
|
||||||
|
return Toy_createRefFunction(refFunction->data, refFunction->length);
|
||||||
|
}
|
||||||
@@ -0,0 +1,88 @@
|
|||||||
|
#pragma once
|
||||||
|
|
||||||
|
/*!
|
||||||
|
# toy_reffunction.h
|
||||||
|
|
||||||
|
This header defines the Toy_RefFunction structure, as well as all of the related utilities.
|
||||||
|
|
||||||
|
See [Toy_RefString](toy_refstring_h.md) for more information about the reference pattern.
|
||||||
|
|
||||||
|
This module reserves the right to instead preform a deep copy when it sees fit (this is for future debugging purposes).
|
||||||
|
!*/
|
||||||
|
|
||||||
|
#include "toy_common.h"
|
||||||
|
|
||||||
|
//the RefFunction structure
|
||||||
|
typedef struct Toy_RefFunction {
|
||||||
|
size_t length;
|
||||||
|
int refCount;
|
||||||
|
unsigned char data[];
|
||||||
|
} Toy_RefFunction;
|
||||||
|
|
||||||
|
/*!
|
||||||
|
## Defined Interfaces
|
||||||
|
!*/
|
||||||
|
|
||||||
|
/*!
|
||||||
|
### typedef void* (*Toy_RefFunctionAllocatorFn)(void* pointer, size_t oldSize, size_t newSize)
|
||||||
|
|
||||||
|
This interface conforms to Toy's memory API, and generally shouldn't be used without a good reason.
|
||||||
|
!*/
|
||||||
|
typedef void* (*Toy_RefFunctionAllocatorFn)(void* pointer, size_t oldSize, size_t newSize);
|
||||||
|
|
||||||
|
/*!
|
||||||
|
## Defined Functions
|
||||||
|
!*/
|
||||||
|
|
||||||
|
/*!
|
||||||
|
### void Toy_setRefFunctionAllocatorFn(Toy_RefFunctionAllocatorFn)
|
||||||
|
|
||||||
|
This function conforms to and is invoked by Toy's memory API, and generally shouldn't be used without a good reason.
|
||||||
|
!*/
|
||||||
|
TOY_API void Toy_setRefFunctionAllocatorFn(Toy_RefFunctionAllocatorFn);
|
||||||
|
|
||||||
|
/*!
|
||||||
|
### Toy_RefFunction* Toy_createRefFunction(const void* data, size_t length)
|
||||||
|
|
||||||
|
This function returns a new `Toy_RefFunction`, containing a copy of `data`, or `NULL` on error.
|
||||||
|
|
||||||
|
This function also sets the returned `refFunction`'s reference counter to 1.
|
||||||
|
!*/
|
||||||
|
TOY_API Toy_RefFunction* Toy_createRefFunction(const void* data, size_t length);
|
||||||
|
|
||||||
|
/*!
|
||||||
|
### void Toy_deleteRefFunction(Toy_RefFunction* refFunction)
|
||||||
|
|
||||||
|
This function reduces the `refFunction`'s reference counter by 1 and, if it reaches 0, frees the memory.
|
||||||
|
!*/
|
||||||
|
TOY_API void Toy_deleteRefFunction(Toy_RefFunction* refFunction);
|
||||||
|
|
||||||
|
/*!
|
||||||
|
### int Toy_countRefFunction(Toy_RefFunction* refFunction)
|
||||||
|
|
||||||
|
This function returns the total number of references to `refFunction`, for debugging.
|
||||||
|
!*/
|
||||||
|
TOY_API int Toy_countRefFunction(Toy_RefFunction* refFunction);
|
||||||
|
|
||||||
|
/*!
|
||||||
|
### size_t Toy_lengthRefFunction(Toy_RefFunction* refFunction)
|
||||||
|
|
||||||
|
This function returns the length of the underlying bytecode of `refFunction`.
|
||||||
|
!*/
|
||||||
|
TOY_API size_t Toy_lengthRefFunction(Toy_RefFunction* refFunction);
|
||||||
|
|
||||||
|
/*!
|
||||||
|
### Toy_RefFunction* Toy_copyRefFunction(Toy_RefFunction* refFunction)
|
||||||
|
|
||||||
|
This function increases the reference counter of `refFunction` by 1, before returning the given pointer.
|
||||||
|
|
||||||
|
This function reserves the right to create a deep copy where needed.
|
||||||
|
!*/
|
||||||
|
TOY_API Toy_RefFunction* Toy_copyRefFunction(Toy_RefFunction* refFunction);
|
||||||
|
|
||||||
|
/*!
|
||||||
|
### Toy_RefFunction* Toy_deepCopyRefFunction(Toy_RefFunction* refFunction)
|
||||||
|
|
||||||
|
This function behaves identically to `Toy_copyRefFunction`, except that it explicitly forces a deep copy of the internal memory. Using this function should be done carefully, as it incurs a performance penalty that negates the benefit of this module.
|
||||||
|
!*/
|
||||||
|
TOY_API Toy_RefFunction* Toy_deepCopyRefFunction(Toy_RefFunction* refFunction);
|
||||||
+100
-6
@@ -1,13 +1,24 @@
|
|||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
|
/*!
|
||||||
|
# toy_refstring.h
|
||||||
|
|
||||||
|
This header defines the structure `Toy_RefString`, as well as all of the related utilities.
|
||||||
|
|
||||||
|
[refstring](https://github.com/Ratstail91/refstring) is a stand-alone utility written to reduce the amount of memory manipulation used within Toy. It was independantly written and tested, before being incorporated into Toy proper. As such it has it's own memory management API, which by default is tied into Toy's [core memory API](toy_memory_h.md).
|
||||||
|
|
||||||
|
Instances of `Toy_RefString` are reference counted - that is, rather than copying an existing string in memory, a pointer to the refstring is returned, and the internal reference counter is increased by 1. When the pointer is no longer needed, `Toy_DeleteRefString` can be called; this will decrement the internal reference counter by 1, and only free it when it reaches 0. This has multiple benefits, when used correctly:
|
||||||
|
|
||||||
|
* Reduced memory usage
|
||||||
|
* Faster program execution
|
||||||
|
|
||||||
|
This module reserves the right to instead preform a deep copy when it sees fit (this is for future debugging purposes).
|
||||||
|
!*/
|
||||||
|
|
||||||
#include "toy_common.h"
|
#include "toy_common.h"
|
||||||
|
|
||||||
#include <string.h>
|
#include <string.h>
|
||||||
|
|
||||||
//memory allocation hook
|
|
||||||
typedef void* (*Toy_RefStringAllocatorFn)(void* pointer, size_t oldSize, size_t newSize);
|
|
||||||
TOY_API void Toy_setRefStringAllocatorFn(Toy_RefStringAllocatorFn);
|
|
||||||
|
|
||||||
//the RefString structure
|
//the RefString structure
|
||||||
typedef struct Toy_RefString {
|
typedef struct Toy_RefString {
|
||||||
size_t length;
|
size_t length;
|
||||||
@@ -15,17 +26,100 @@ typedef struct Toy_RefString {
|
|||||||
char data[];
|
char data[];
|
||||||
} Toy_RefString;
|
} Toy_RefString;
|
||||||
|
|
||||||
//API
|
/*!
|
||||||
|
## Defined Interfaces
|
||||||
|
!*/
|
||||||
|
|
||||||
|
/*!
|
||||||
|
### typedef void* (*Toy_RefStringAllocatorFn)(void* pointer, size_t oldSize, size_t newSize)
|
||||||
|
|
||||||
|
This interface conforms to Toy's memory API, and generally shouldn't be used without a good reason.
|
||||||
|
!*/
|
||||||
|
typedef void* (*Toy_RefStringAllocatorFn)(void* pointer, size_t oldSize, size_t newSize);
|
||||||
|
|
||||||
|
/*!
|
||||||
|
## Defined Functions
|
||||||
|
!*/
|
||||||
|
|
||||||
|
/*!
|
||||||
|
### void Toy_setRefStringAllocatorFn(Toy_RefStringAllocatorFn)
|
||||||
|
|
||||||
|
This function conforms to and is invoked by Toy's memory API, and generally shouldn't be used without a good reason.
|
||||||
|
!*/
|
||||||
|
TOY_API void Toy_setRefStringAllocatorFn(Toy_RefStringAllocatorFn);
|
||||||
|
|
||||||
|
/*!
|
||||||
|
### Toy_RefString* Toy_createRefString(const char* cstring)
|
||||||
|
|
||||||
|
This function wraps `Toy_CreateRefStringLength`, by determining the length of the given `cstring` and passing it to the other function.
|
||||||
|
!*/
|
||||||
TOY_API Toy_RefString* Toy_createRefString(const char* cstring);
|
TOY_API Toy_RefString* Toy_createRefString(const char* cstring);
|
||||||
|
|
||||||
|
/*!
|
||||||
|
### Toy_RefString* Toy_createRefStringLength(const char* cstring, size_t length)
|
||||||
|
|
||||||
|
This function returns a new `Toy_RefString`, containing a copy of `cstring`, or `NULL` on error.
|
||||||
|
|
||||||
|
This function also sets the returned refstring's reference counter to 1.
|
||||||
|
!*/
|
||||||
TOY_API Toy_RefString* Toy_createRefStringLength(const char* cstring, size_t length);
|
TOY_API Toy_RefString* Toy_createRefStringLength(const char* cstring, size_t length);
|
||||||
|
|
||||||
|
/*!
|
||||||
|
### void Toy_deleteRefString(Toy_RefString* refString)
|
||||||
|
|
||||||
|
This function reduces the `refString`'s reference counter by 1 and, if it reaches 0, frees the memory.
|
||||||
|
!*/
|
||||||
TOY_API void Toy_deleteRefString(Toy_RefString* refString);
|
TOY_API void Toy_deleteRefString(Toy_RefString* refString);
|
||||||
|
|
||||||
|
/*!
|
||||||
|
### int Toy_countRefString(Toy_RefString* refString)
|
||||||
|
|
||||||
|
This function returns the total number of references to `refString`, for debugging.
|
||||||
|
!*/
|
||||||
TOY_API int Toy_countRefString(Toy_RefString* refString);
|
TOY_API int Toy_countRefString(Toy_RefString* refString);
|
||||||
|
|
||||||
|
/*!
|
||||||
|
### size_t Toy_lengthRefString(Toy_RefString* refString)
|
||||||
|
|
||||||
|
This function returns the length of the underlying cstring of `refString`.
|
||||||
|
!*/
|
||||||
TOY_API size_t Toy_lengthRefString(Toy_RefString* refString);
|
TOY_API size_t Toy_lengthRefString(Toy_RefString* refString);
|
||||||
|
|
||||||
|
/*!
|
||||||
|
### Toy_RefString* Toy_copyRefString(Toy_RefString* refString)
|
||||||
|
|
||||||
|
This function increases the reference counter of `refString` by 1, before returning the given pointer.
|
||||||
|
|
||||||
|
This function reserves the right to create a deep copy where needed.
|
||||||
|
!*/
|
||||||
TOY_API Toy_RefString* Toy_copyRefString(Toy_RefString* refString);
|
TOY_API Toy_RefString* Toy_copyRefString(Toy_RefString* refString);
|
||||||
|
|
||||||
|
/*!
|
||||||
|
### Toy_RefString* Toy_deepCopyRefString(Toy_RefString* refString)
|
||||||
|
|
||||||
|
This function behaves identically to `Toy_copyRefString`, except that it explicitly forces a deep copy of the internal memory. Using this function should be done carefully, as it incurs a performance penalty that negates the benefit of this module.
|
||||||
|
!*/
|
||||||
TOY_API Toy_RefString* Toy_deepCopyRefString(Toy_RefString* refString);
|
TOY_API Toy_RefString* Toy_deepCopyRefString(Toy_RefString* refString);
|
||||||
|
|
||||||
|
/*!
|
||||||
|
### const char* Toy_toCString(Toy_RefString* refString)
|
||||||
|
|
||||||
|
This function exposes the interal cstring of `refString`. Only use this function when dealing with external APIs.
|
||||||
|
!*/
|
||||||
TOY_API const char* Toy_toCString(Toy_RefString* refString);
|
TOY_API const char* Toy_toCString(Toy_RefString* refString);
|
||||||
|
|
||||||
|
/*!
|
||||||
|
### bool Toy_equalsRefString(Toy_RefString* lhs, Toy_RefString* rhs)
|
||||||
|
|
||||||
|
This function returns true when the two refstrings are either the same refstring, or contain the same value. Otherwise it returns false.
|
||||||
|
!*/
|
||||||
TOY_API bool Toy_equalsRefString(Toy_RefString* lhs, Toy_RefString* rhs);
|
TOY_API bool Toy_equalsRefString(Toy_RefString* lhs, Toy_RefString* rhs);
|
||||||
|
|
||||||
|
/*!
|
||||||
|
### bool Toy_equalsRefStringCString(Toy_RefString* lhs, char* cstring)
|
||||||
|
|
||||||
|
This function returns true when the `refString` contains the same value as the `cstring`. Otherwise it returns false.
|
||||||
|
!*/
|
||||||
TOY_API bool Toy_equalsRefStringCString(Toy_RefString* lhs, char* cstring);
|
TOY_API bool Toy_equalsRefStringCString(Toy_RefString* lhs, char* cstring);
|
||||||
|
|
||||||
//TODO: merge refstring memory
|
//TODO: merge refstring memory
|
||||||
|
|
||||||
|
|||||||
+1
-1
@@ -256,7 +256,7 @@ bool Toy_declareScopeVariable(Toy_Scope* scope, Toy_Literal key, Toy_Literal typ
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool Toy_isDelcaredScopeVariable(Toy_Scope* scope, Toy_Literal key) {
|
bool Toy_isDeclaredScopeVariable(Toy_Scope* scope, Toy_Literal key) {
|
||||||
while (scope != NULL) {
|
while (scope != NULL) {
|
||||||
if (Toy_existsLiteralDictionary(&scope->variables, key)) {
|
if (Toy_existsLiteralDictionary(&scope->variables, key)) {
|
||||||
return true;
|
return true;
|
||||||
|
|||||||
+68
-4
@@ -1,5 +1,15 @@
|
|||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
|
/*!
|
||||||
|
# toy_scope.h
|
||||||
|
|
||||||
|
This header defines the scope structure, which stores all of the variables used within a given block of code.
|
||||||
|
|
||||||
|
Scopes are arranged into a linked list of ancestors, each of which is reference counted. When a scope is popped off the end of the chain, every ancestor scope has it's reference counter reduced by 1 and, if any reach 0, they are freed.
|
||||||
|
|
||||||
|
This is also where Toy's type system lives.
|
||||||
|
!*/
|
||||||
|
|
||||||
#include "toy_literal.h"
|
#include "toy_literal.h"
|
||||||
#include "toy_literal_array.h"
|
#include "toy_literal_array.h"
|
||||||
#include "toy_literal_dictionary.h"
|
#include "toy_literal_dictionary.h"
|
||||||
@@ -11,16 +21,70 @@ typedef struct Toy_Scope {
|
|||||||
int references; //how many scopes point here
|
int references; //how many scopes point here
|
||||||
} Toy_Scope;
|
} Toy_Scope;
|
||||||
|
|
||||||
|
/*!
|
||||||
|
## Defined Functions
|
||||||
|
!*/
|
||||||
|
|
||||||
|
/*!
|
||||||
|
### Toy_Scope* Toy_pushScope(Toy_Scope* scope)
|
||||||
|
|
||||||
|
This function creates a new `Toy_scope` with `scope` as it's ancestor, and returns it.
|
||||||
|
!*/
|
||||||
TOY_API Toy_Scope* Toy_pushScope(Toy_Scope* scope);
|
TOY_API Toy_Scope* Toy_pushScope(Toy_Scope* scope);
|
||||||
|
|
||||||
|
/*!
|
||||||
|
### Toy_Scope* Toy_popScope(Toy_Scope* scope)
|
||||||
|
|
||||||
|
This function frees the given `scope`, and returns it's ancestor.
|
||||||
|
!*/
|
||||||
TOY_API Toy_Scope* Toy_popScope(Toy_Scope* scope);
|
TOY_API Toy_Scope* Toy_popScope(Toy_Scope* scope);
|
||||||
|
|
||||||
|
/*!
|
||||||
|
### Toy_Scope* Toy_copyScope(Toy_Scope* original)
|
||||||
|
|
||||||
|
This function copies an existing scope, and returns the copy.
|
||||||
|
|
||||||
|
This copies the internal dictionaries, so it can be memory intensive.
|
||||||
|
!*/
|
||||||
TOY_API Toy_Scope* Toy_copyScope(Toy_Scope* original);
|
TOY_API Toy_Scope* Toy_copyScope(Toy_Scope* original);
|
||||||
|
|
||||||
//returns false if error
|
/*!
|
||||||
TOY_API bool Toy_declareScopeVariable(Toy_Scope* scope, Toy_Literal key, Toy_Literal type);
|
### bool Toy_declareScopeVariable(Toy_Scope* scope, Toy_Literal key, Toy_Literal type)
|
||||||
TOY_API bool Toy_isDelcaredScopeVariable(Toy_Scope* scope, Toy_Literal key);
|
|
||||||
|
|
||||||
//return false if undefined
|
This function declares a new variable `key` within `scope`, giving it the type of `type`.
|
||||||
|
|
||||||
|
This function returns true on success, otherwise it returns failure (such as if the given key already exists).
|
||||||
|
!*/
|
||||||
|
TOY_API bool Toy_declareScopeVariable(Toy_Scope* scope, Toy_Literal key, Toy_Literal type);
|
||||||
|
|
||||||
|
/*!
|
||||||
|
### bool Toy_isDeclaredScopeVariable(Toy_Scope* scope, Toy_Literal key)
|
||||||
|
|
||||||
|
This function checks to see if a given variable with the name `key` has been previously declared.
|
||||||
|
!*/
|
||||||
|
TOY_API bool Toy_isDeclaredScopeVariable(Toy_Scope* scope, Toy_Literal key);
|
||||||
|
|
||||||
|
/*!
|
||||||
|
### bool Toy_setScopeVariable(Toy_Scope* scope, Toy_Literal key, Toy_Literal value, bool constCheck)
|
||||||
|
|
||||||
|
This function sets an existing variable named `key` to the value of `value`. This function fails if `constCheck` is true and the given key's type has the constaant flag set. It also fails if the given key doesn't exist.
|
||||||
|
|
||||||
|
This function returns true on success, otherwise it returns false.
|
||||||
|
!*/
|
||||||
TOY_API bool Toy_setScopeVariable(Toy_Scope* scope, Toy_Literal key, Toy_Literal value, bool constCheck);
|
TOY_API bool Toy_setScopeVariable(Toy_Scope* scope, Toy_Literal key, Toy_Literal value, bool constCheck);
|
||||||
|
|
||||||
|
/*!
|
||||||
|
### bool Toy_getScopeVariable(Toy_Scope* scope, Toy_Literal key, Toy_Literal* value)
|
||||||
|
|
||||||
|
This function sets the literal pointed to by `value` to equal the variable named `key`.
|
||||||
|
|
||||||
|
This function returns true on success, otherwise it returns false.
|
||||||
|
!*/
|
||||||
TOY_API bool Toy_getScopeVariable(Toy_Scope* scope, Toy_Literal key, Toy_Literal* value);
|
TOY_API bool Toy_getScopeVariable(Toy_Scope* scope, Toy_Literal key, Toy_Literal* value);
|
||||||
|
|
||||||
|
/*!
|
||||||
|
### Toy_Literal Toy_getScopeType(Toy_Scope* scope, Toy_Literal key)
|
||||||
|
|
||||||
|
This function returns a new `Toy_Literal` representing the type of the variable named `key`.
|
||||||
|
!*/
|
||||||
TOY_API Toy_Literal Toy_getScopeType(Toy_Scope* scope, Toy_Literal key);
|
TOY_API Toy_Literal Toy_getScopeType(Toy_Scope* scope, Toy_Literal key);
|
||||||
|
|||||||
+1
-1
@@ -4,7 +4,7 @@ IDIR +=. ../source ../repl
|
|||||||
CFLAGS +=$(addprefix -I,$(IDIR)) -g -Wall -W -Wno-unused-parameter -Wno-unused-function -Wno-unused-variable
|
CFLAGS +=$(addprefix -I,$(IDIR)) -g -Wall -W -Wno-unused-parameter -Wno-unused-function -Wno-unused-variable
|
||||||
LIBS +=
|
LIBS +=
|
||||||
ODIR = obj
|
ODIR = obj
|
||||||
TARGETS = $(wildcard ../source/*.c) $(wildcard ../repl/lib_*.c) ../repl/repl_tools.c
|
TARGETS = $(wildcard ../source/*.c) $(wildcard ../repl/lib_*.c) ../repl/repl_tools.c ../repl/drive_system.c
|
||||||
TESTS = $(wildcard test_*.c)
|
TESTS = $(wildcard test_*.c)
|
||||||
OBJ = $(addprefix $(ODIR)/,$(TARGETS:../source/%.c=%.o)) $(addprefix $(ODIR)/,$(TESTS:.c=.o))
|
OBJ = $(addprefix $(ODIR)/,$(TARGETS:../source/%.c=%.o)) $(addprefix $(ODIR)/,$(TESTS:.c=.o))
|
||||||
|
|
||||||
|
|||||||
@@ -0,0 +1,30 @@
|
|||||||
|
{
|
||||||
|
fn a() {
|
||||||
|
fn b() {
|
||||||
|
return 42;
|
||||||
|
}
|
||||||
|
|
||||||
|
return b;
|
||||||
|
}
|
||||||
|
|
||||||
|
assert a()() == 42, "function within function failed";
|
||||||
|
}
|
||||||
|
|
||||||
|
{
|
||||||
|
fn a() {
|
||||||
|
fn b() {
|
||||||
|
fn c() {
|
||||||
|
return 42;
|
||||||
|
}
|
||||||
|
|
||||||
|
return c;
|
||||||
|
}
|
||||||
|
|
||||||
|
return b;
|
||||||
|
}
|
||||||
|
|
||||||
|
assert a()()() == 42, "function within function within function failed";
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
print "All good";
|
||||||
@@ -0,0 +1,11 @@
|
|||||||
|
//test for casting + grouping, see #67
|
||||||
|
{
|
||||||
|
assert string(10 % 4) == "2", "basic group casting failed";
|
||||||
|
assert string 4 == "4", "normal casting failed";
|
||||||
|
|
||||||
|
assert typeof string(10 % 4) == string, "group casting type failed";
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
print "All good";
|
||||||
@@ -0,0 +1,15 @@
|
|||||||
|
var a = 0;
|
||||||
|
|
||||||
|
if (a++ >= 1) {
|
||||||
|
assert false, "increment postfix bugfix failed (first check)";
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
if (a++ >= 1) {
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
assert a == 2, "increment postfix bugfix failed (second check)";
|
||||||
|
|
||||||
|
|
||||||
|
print "All good";
|
||||||
@@ -92,4 +92,14 @@
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
//test deep-combine example
|
||||||
|
{
|
||||||
|
fn combine(a, b, c, d) {
|
||||||
|
return [[a, b], [c, d]];
|
||||||
|
}
|
||||||
|
|
||||||
|
assert combine(1, 2, 3, 4) == [[1, 2], [3, 4]], "deep-combine example failed";
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
print "All good";
|
print "All good";
|
||||||
|
|||||||
@@ -0,0 +1,10 @@
|
|||||||
|
|
||||||
|
var result; //result must exist to ensure assingment, rather than declaration is invoked by the comparison below
|
||||||
|
|
||||||
|
var lhs = [0];
|
||||||
|
var rhs = [0];
|
||||||
|
|
||||||
|
result = lhs[0] < rhs[0]; //make sure this doesn't invoke TOY_OP_INDEX_ASSIGN_INTERMEDIATE
|
||||||
|
|
||||||
|
|
||||||
|
print "All good";
|
||||||
@@ -101,6 +101,63 @@ import standard;
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
//test sign
|
||||||
|
{
|
||||||
|
assert sign(4) == 1, "sign(int) failed";
|
||||||
|
assert sign(-4) == -1, "sign(-int) failed";
|
||||||
|
assert sign(4.1) == 1, "sign(float) failed";
|
||||||
|
assert sign(-4.1) == -1, "sign(-float) failed";
|
||||||
|
assert sign(0) == 1, "sign(0) failed";
|
||||||
|
|
||||||
|
var x = 4.1;
|
||||||
|
|
||||||
|
assert x.sign() == 1, "var.sign() failed";
|
||||||
|
|
||||||
|
assert typeof sign(1.0) == int, "typeof sign() == int failed";
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
//test normalize
|
||||||
|
{
|
||||||
|
assert normalize(4) == 1, "normalize(int) failed";
|
||||||
|
assert normalize(-4) == -1, "normalize(-int) failed";
|
||||||
|
assert normalize(4.1) == 1, "normalize(float) failed";
|
||||||
|
assert normalize(-4.1) == -1, "normalize(-float) failed";
|
||||||
|
assert normalize(0) == 0, "normalize(0) failed";
|
||||||
|
|
||||||
|
var x = 4.1;
|
||||||
|
|
||||||
|
assert x.normalize() == 1, "var.normalize() failed";
|
||||||
|
|
||||||
|
assert typeof normalize(1.0) == int, "typeof normalize() == int failed";
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
//test clamp
|
||||||
|
{
|
||||||
|
assert clamp(1, 0, 5) == 1, "clamp(1, 0, 5) failed";
|
||||||
|
assert clamp(0, 1, 5) == 1, "clamp(0, 1, 5) failed";
|
||||||
|
assert clamp(10, 1, 5) == 5, "clamp(10, 1, 5) failed";
|
||||||
|
|
||||||
|
assert clamp(1.0, 0.0, 5.0) == 1, "clamp(1.0, 0.0, 5.0) failed";
|
||||||
|
assert clamp(0.0, 1.0, 5.0) == 1, "clamp(0.0, 1.0, 5.0) failed";
|
||||||
|
assert clamp(10.0, 1.0, 5.0) == 5, "clamp(10.0, 1.0, 5.0) failed";
|
||||||
|
|
||||||
|
assert typeof clamp(10, 1, 5) == int, "typeof clamp(10, 1, 5) == int failed";
|
||||||
|
assert typeof clamp(10.0, 1, 5) == int, "typeof clamp(10.0, 1, 5) == int failed";
|
||||||
|
assert typeof clamp(10, 1, 5.0) == float, "typeof clamp(10, 1, 5.0) == float failed";
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
//test lerp
|
||||||
|
{
|
||||||
|
assert lerp(0, 10, 0.5) == 5, "lerp 50% failed";
|
||||||
|
assert lerp(0, 10, 1.5) == 15, "lerp 150% failed";
|
||||||
|
|
||||||
|
assert typeof lerp(0, 10, 0) == float, "typeof lerp result failed";
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
//test concat
|
//test concat
|
||||||
{
|
{
|
||||||
//test array concat
|
//test array concat
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
import about as about;
|
import toy_version_info as toy_version_info;
|
||||||
import about;
|
import toy_version_info;
|
||||||
|
|
||||||
assert author == "Kayne Ruse, KR Game Studios", "Author failed";
|
assert author == "Kayne Ruse, KR Game Studios", "Author failed";
|
||||||
|
|
||||||
@@ -22,4 +22,9 @@ var dict: complex = [
|
|||||||
"third array": [7, 8, 9]
|
"third array": [7, 8, 9]
|
||||||
];
|
];
|
||||||
|
|
||||||
|
|
||||||
|
//check the any type is recognized as a type within an array
|
||||||
|
var a: [type] = [int, bool, any];
|
||||||
|
|
||||||
|
|
||||||
print "All good";
|
print "All good";
|
||||||
|
|||||||
@@ -0,0 +1,79 @@
|
|||||||
|
#include "drive_system.h"
|
||||||
|
|
||||||
|
#include "toy_console_colors.h"
|
||||||
|
|
||||||
|
#include <stdio.h>
|
||||||
|
#include <string.h>
|
||||||
|
|
||||||
|
int main() {
|
||||||
|
{
|
||||||
|
//test init and quit
|
||||||
|
Toy_initDriveSystem();
|
||||||
|
Toy_freeDriveSystem();
|
||||||
|
}
|
||||||
|
|
||||||
|
{
|
||||||
|
//setup
|
||||||
|
Toy_initDriveSystem();
|
||||||
|
|
||||||
|
//test storing a value as a drive
|
||||||
|
Toy_setDrivePath("drive", "folder");
|
||||||
|
|
||||||
|
//cleanup
|
||||||
|
Toy_freeDriveSystem();
|
||||||
|
}
|
||||||
|
|
||||||
|
{
|
||||||
|
//setup
|
||||||
|
Toy_initDriveSystem();
|
||||||
|
|
||||||
|
//create a dummy interpreter (only needed for the error output function)
|
||||||
|
Toy_Interpreter interpreter;
|
||||||
|
Toy_initInterpreter(&interpreter);
|
||||||
|
|
||||||
|
//prerequisite
|
||||||
|
Toy_setDrivePath("drive", "path/to/drive");
|
||||||
|
|
||||||
|
//create the argument literal
|
||||||
|
Toy_Literal argumentLiteral = TOY_TO_STRING_LITERAL(Toy_createRefString("drive:/path/to/file"));
|
||||||
|
|
||||||
|
//test retrieving a relative path, as a literal, from the drive system
|
||||||
|
Toy_Literal resultLiteral = Toy_getDrivePathLiteral(&interpreter, &argumentLiteral);
|
||||||
|
|
||||||
|
//assert the correct value was returned
|
||||||
|
const char* cstring = Toy_toCString(TOY_AS_STRING(resultLiteral));
|
||||||
|
if (strcmp(cstring, "path/to/drive/path/to/file") != 0) {
|
||||||
|
fprintf(stderr, TOY_CC_ERROR "ERROR: Incorrect value retrieved from drive system: %s" TOY_CC_RESET, cstring);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
//cleanup
|
||||||
|
Toy_freeLiteral(argumentLiteral);
|
||||||
|
Toy_freeLiteral(resultLiteral);
|
||||||
|
Toy_freeInterpreter(&interpreter);
|
||||||
|
Toy_freeDriveSystem();
|
||||||
|
}
|
||||||
|
|
||||||
|
{
|
||||||
|
//setup
|
||||||
|
Toy_initDriveSystem();
|
||||||
|
|
||||||
|
//test storing enough drives to trigger internal dictionary expansion
|
||||||
|
Toy_setDrivePath("A", "folder");
|
||||||
|
Toy_setDrivePath("B", "folder");
|
||||||
|
Toy_setDrivePath("C", "folder");
|
||||||
|
Toy_setDrivePath("D", "folder");
|
||||||
|
Toy_setDrivePath("E", "folder");
|
||||||
|
Toy_setDrivePath("F", "folder");
|
||||||
|
Toy_setDrivePath("G", "folder");
|
||||||
|
Toy_setDrivePath("H", "folder");
|
||||||
|
Toy_setDrivePath("I", "folder");
|
||||||
|
Toy_setDrivePath("J", "folder");
|
||||||
|
|
||||||
|
//cleanup
|
||||||
|
Toy_freeDriveSystem();
|
||||||
|
}
|
||||||
|
|
||||||
|
printf(TOY_CC_NOTICE "All good\n" TOY_CC_RESET);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
@@ -117,9 +117,13 @@ int main() {
|
|||||||
"dot-chaining.toy",
|
"dot-chaining.toy",
|
||||||
"dot-modulo-bugfix.toy",
|
"dot-modulo-bugfix.toy",
|
||||||
"dottify-bugfix.toy",
|
"dottify-bugfix.toy",
|
||||||
|
"function-within-function-bugfix.toy",
|
||||||
"functions.toy",
|
"functions.toy",
|
||||||
|
"group-casting-bugfix.toy",
|
||||||
|
"increment-postfix-bugfix.toy",
|
||||||
"index-arrays.toy",
|
"index-arrays.toy",
|
||||||
"index-assignment-both-bugfix.toy",
|
"index-assignment-both-bugfix.toy",
|
||||||
|
"index-assignment-intermediate-bugfix.toy",
|
||||||
"index-assignment-left-bugfix.toy",
|
"index-assignment-left-bugfix.toy",
|
||||||
"index-dictionaries.toy",
|
"index-dictionaries.toy",
|
||||||
"index-strings.toy",
|
"index-strings.toy",
|
||||||
|
|||||||
@@ -6,18 +6,18 @@
|
|||||||
#include "toy_console_colors.h"
|
#include "toy_console_colors.h"
|
||||||
|
|
||||||
#include "toy_memory.h"
|
#include "toy_memory.h"
|
||||||
#include "toy_drive_system.h"
|
|
||||||
|
|
||||||
#include <stdio.h>
|
#include <stdio.h>
|
||||||
#include <stdlib.h>
|
#include <stdlib.h>
|
||||||
#include <string.h>
|
#include <string.h>
|
||||||
|
|
||||||
#include "../repl/repl_tools.h"
|
#include "../repl/repl_tools.h"
|
||||||
|
#include "../repl/drive_system.h"
|
||||||
|
|
||||||
#include "../repl/lib_about.h"
|
#include "../repl/lib_toy_version_info.h"
|
||||||
|
#include "../repl/lib_standard.h"
|
||||||
#include "../repl/lib_random.h"
|
#include "../repl/lib_random.h"
|
||||||
#include "../repl/lib_runner.h"
|
#include "../repl/lib_runner.h"
|
||||||
#include "../repl/lib_standard.h"
|
|
||||||
|
|
||||||
//supress the print output
|
//supress the print output
|
||||||
static void noPrintFn(const char* output) {
|
static void noPrintFn(const char* output) {
|
||||||
@@ -72,7 +72,7 @@ int main() {
|
|||||||
//run each file in test/scripts
|
//run each file in test/scripts
|
||||||
Payload payloads[] = {
|
Payload payloads[] = {
|
||||||
{"interactions.toy", "standard", Toy_hookStandard}, //interactions needs standard
|
{"interactions.toy", "standard", Toy_hookStandard}, //interactions needs standard
|
||||||
{"about.toy", "about", Toy_hookAbout},
|
{"toy_version_info.toy", "toy_version_info", Toy_hookToyVersionInfo},
|
||||||
{"standard.toy", "standard", Toy_hookStandard},
|
{"standard.toy", "standard", Toy_hookStandard},
|
||||||
{"runner.toy", "runner", Toy_hookRunner},
|
{"runner.toy", "runner", Toy_hookRunner},
|
||||||
{"random.toy", "random", Toy_hookRandom},
|
{"random.toy", "random", Toy_hookRandom},
|
||||||
|
|||||||
@@ -0,0 +1,73 @@
|
|||||||
|
//changing of the guard - made for @hyperiondev's microcontrollers
|
||||||
|
#include <iostream>
|
||||||
|
#include <fstream>
|
||||||
|
#include <string>
|
||||||
|
#include <algorithm>
|
||||||
|
|
||||||
|
std::string convert(std::string str) {
|
||||||
|
str = str.substr(str.find_last_of("\\/")+1);
|
||||||
|
std::transform(str.begin(), str.end(), str.begin(), ::toupper);
|
||||||
|
std::replace(str.begin(), str.end(), '.', '_');
|
||||||
|
return str;
|
||||||
|
}
|
||||||
|
|
||||||
|
std::string convertToGuardStart(std::string str) {
|
||||||
|
str = convert(str);
|
||||||
|
return "#ifndef " + str + "\n#define " + str + "\n";
|
||||||
|
}
|
||||||
|
|
||||||
|
std::string convertToGuardEnd(std::string str) {
|
||||||
|
str = convert(str);
|
||||||
|
return "\n#endif //" + str + "\n";
|
||||||
|
}
|
||||||
|
|
||||||
|
int main(int argc, char** argv) {
|
||||||
|
if (argc <= 1) {
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
for (int fileCounter = 1; fileCounter < argc; fileCounter++) {
|
||||||
|
std::ifstream is; //input stream
|
||||||
|
std::string buffer; //store output file
|
||||||
|
|
||||||
|
//open
|
||||||
|
is.open(argv[fileCounter]);
|
||||||
|
if (!is.is_open()) {
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
while (!is.eof()) {
|
||||||
|
std::string top; //I dislike C++
|
||||||
|
getline(is, top);
|
||||||
|
|
||||||
|
//check for pragma guard
|
||||||
|
if (top == "#pragma once") {
|
||||||
|
top = convertToGuardStart(argv[fileCounter]);
|
||||||
|
getline(is, buffer, '\0');
|
||||||
|
buffer += convertToGuardEnd(argv[fileCounter]);
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
top += "\n";
|
||||||
|
getline(is, buffer, '\0');
|
||||||
|
}
|
||||||
|
|
||||||
|
buffer = top + buffer;
|
||||||
|
}
|
||||||
|
|
||||||
|
//finally
|
||||||
|
is.close();
|
||||||
|
|
||||||
|
std::ofstream os;
|
||||||
|
|
||||||
|
os.open(argv[fileCounter]);
|
||||||
|
if (!os.is_open()) {
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
os << buffer;
|
||||||
|
|
||||||
|
os.close();
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
+105
@@ -0,0 +1,105 @@
|
|||||||
|
//MECHA: Markdown Embedded Comment Heuristic Analyzer
|
||||||
|
|
||||||
|
/*!
|
||||||
|
# Handy-Dandy *MECHA* Docs!
|
||||||
|
|
||||||
|
The following is the source code for MECHA - a tool for reading in source code, and splicing out the markdown embedded within. It can also spit out chunks of code, though that's a bit uglier.
|
||||||
|
|
||||||
|
In theory, this should work correctly on all languages that conform to C-style comments, including Toy.
|
||||||
|
|
||||||
|
To build it, simply run `g++ -o mecha mecha.cpp`.
|
||||||
|
|
||||||
|
To run the result, you must pass in a series of filenames via the command line.
|
||||||
|
|
||||||
|
## License
|
||||||
|
|
||||||
|
This tool is considered part of Toy's toolchain, so Toy's zlib license should cover it.
|
||||||
|
|
||||||
|
## Header Files
|
||||||
|
|
||||||
|
If you're reading this as a comment in the source code, this is what I mean by ugly. Thankfully, this section is outputted correctly in markdown.
|
||||||
|
|
||||||
|
!*/
|
||||||
|
|
||||||
|
//*!```
|
||||||
|
#include <iostream>
|
||||||
|
#include <fstream>
|
||||||
|
#include <string>
|
||||||
|
#include <algorithm>
|
||||||
|
//!*
|
||||||
|
|
||||||
|
/*!```
|
||||||
|
|
||||||
|
# Recursive Docs
|
||||||
|
|
||||||
|
If you run mecha on it's own source code, it should Just Work, producing a file called mecha_cpp.md
|
||||||
|
!*/
|
||||||
|
|
||||||
|
int main(int argc, char** argv) {
|
||||||
|
if (argc <= 1) {
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
for (int fileCounter = 1; fileCounter < argc; fileCounter++) {
|
||||||
|
std::ifstream is; //input stream
|
||||||
|
std::string buffer; //store output file
|
||||||
|
|
||||||
|
//open
|
||||||
|
is.open(argv[fileCounter]);
|
||||||
|
if (!is.is_open()) {
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
while (!is.eof()) {
|
||||||
|
//skip until correct characters found
|
||||||
|
if (is.get() != '*') {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (is.get() != '!') {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
//found the start of the block - begin reading markdown content
|
||||||
|
while (!is.eof()) {
|
||||||
|
char c = is.get();
|
||||||
|
|
||||||
|
if (c == '!' && is.peek() == '*') {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
buffer += c;
|
||||||
|
}
|
||||||
|
|
||||||
|
//bugfix
|
||||||
|
if (buffer.length() >= 2 && buffer.substr(buffer.length()-2) == "//") {
|
||||||
|
buffer.pop_back();
|
||||||
|
buffer.pop_back();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
//finally
|
||||||
|
is.close();
|
||||||
|
|
||||||
|
if (buffer.length() == 0) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
std::ofstream os;
|
||||||
|
std::string ofname = argv[fileCounter];
|
||||||
|
|
||||||
|
std::replace(ofname.begin(), ofname.end(), '.', '_');
|
||||||
|
ofname += ".md";
|
||||||
|
|
||||||
|
os.open(ofname);
|
||||||
|
if (!os.is_open()) {
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
os << buffer;
|
||||||
|
|
||||||
|
os.close();
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
Reference in New Issue
Block a user