-
-
Notifications
You must be signed in to change notification settings - Fork 2.6k
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
* Introduce the concept of packages. Closes #3 * Add support for error notes. * Introduce `@import` and `@c_import` builtin functions and remove the `import` and `c_import` top level declarations. * Introduce the `use` top level declaration. * Add `--check-unused` parameter to perform semantic analysis and codegen on all top level declarations, not just exported ones and ones referenced by exported ones. * Delete the root export node and add `--library` argument.
- Loading branch information
Showing
31 changed files
with
1,466 additions
and
1,840 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,75 @@ | ||
# How Semantic Analysis Works | ||
|
||
We start with a set of files. Typically the user only has one entry point file, | ||
which imports the other files they want to use. However, the compiler may | ||
choose to add more files to the compilation, for example bootstrap.zig which | ||
contains the code that calls main. | ||
|
||
Our goal now is to treat everything that is marked with the `export` keyword | ||
as a root node, and then then parse and semantically analyze as little as | ||
possible in order to fulfill these exports. | ||
|
||
So, some parts of the code very well may have uncaught semantic errors, but as | ||
long as the code is not referenced in any way, the compiler will not complain | ||
because the code may as well not exist. This is similar to the fact that code | ||
excluded from compilation with an `#ifdef` in C is not analyzed. Avoiding | ||
analyzing unused code will save compilation time - one of Zig's goals. | ||
|
||
So, for each file, we iterate over the top level declarations. The set of top | ||
level declarations are: | ||
|
||
* Function Definition | ||
* Global Variable Declaration | ||
* Container Declaration (struct or enum) | ||
* Type Declaration | ||
* Error Value Declaration | ||
* Use Declaration | ||
|
||
Each of these can have `export` attached to them except for error value | ||
declarations and use declarations. | ||
|
||
When we see a top level declaration during this iteration, we determine its | ||
unique name identifier within the file. For example, for a function definition, | ||
the unique name identifier is simply its name. Using this name we add the top | ||
level declaration to a map. | ||
|
||
If the top level declaration is exported, we add it to a set of exported top | ||
level identifiers. | ||
|
||
If the top level declaration is a use declaration, we add it to a set of use | ||
declarations. | ||
|
||
If the top level declaration is an error value declaration, we assign it a value | ||
and increment the count of error values. | ||
|
||
After this preliminary iteration over the top level declarations, we iterate | ||
over the use declarations and resolve them. To resolve a use declaration, we | ||
analyze the associated expression, verify that its type is the namespace type, | ||
and then add all the items from the namespace into the top level declaration | ||
map for the current file. | ||
|
||
To analyze an expression, we recurse the abstract syntax tree of the | ||
expression. Whenever we must look up a symbol, if the symbol exists already, | ||
we can use it. Otherwise, we look it up in the top level declaration map. | ||
If it exists, we can use it. Otherwise, we interrupt resolving this use | ||
declaration to resolve the next one. If a dependency loop is detected, emit | ||
an error. If all use declarations are resolved yet the symbol we need still | ||
does not exist, emit an error. | ||
|
||
To analyze an `@import` expression, find the referenced file, parse it, and | ||
add it to the set of files to perform semantic analysis on. | ||
|
||
Proceed through the rest of the use declarations the same way. | ||
|
||
If we make it through the use declarations without an error, then we have a | ||
complete map of all globals that exist in the current file. | ||
|
||
Next we iterate over the set of exported top level declarations. | ||
|
||
If it's a function definition, add it to the set of exported function | ||
definitions and resolve the function prototype only. Otherwise, resolve the | ||
top level declaration completely. This may involve recursively resolving other | ||
top level declarations that expressions depend on. | ||
|
||
Finally, iterate over the set of exported function definitions and analyze the | ||
bodies. |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,7 +1,5 @@ | ||
export executable "hello"; | ||
|
||
import "std.zig"; | ||
const io = @import("std").io; | ||
|
||
pub fn main(args: [][]u8) -> %void { | ||
%%stdout.printf("Hello, world!\n"); | ||
%%io.stdout.printf("Hello, world!\n"); | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,11 +1,6 @@ | ||
#link("c") | ||
export executable "hello"; | ||
|
||
c_import { | ||
@c_include("stdio.h"); | ||
} | ||
const c = @c_import(@c_include("stdio.h")); | ||
|
||
export fn main(argc: c_int, argv: &&u8) -> c_int { | ||
printf(c"Hello, world!\n"); | ||
c.printf(c"Hello, world!\n"); | ||
return 0; | ||
} |
Oops, something went wrong.