mirror of
https://github.com/ton-blockchain/ton
synced 2025-03-09 15:40:10 +00:00
This commit introduces nullable types `T?` that are distinct from non-nullable `T`. Example: `int?` (int or null) and `int` are different now. Previously, `null` could be assigned to any primitive type. Now, it can be assigned only to `T?`. A non-null assertion operator `!` was also introduced, similar to `!` in TypeScript and `!!` in Kotlin. If `int?` still occupies 1 stack slot, `(int,int)?` and other nullable tensors occupy N+1 slots, the last for "null precedence". `v == null` actually compares that slot. Assigning `(int,int)` to `(int,int)?` implicitly creates a null presence slot. Assigning `null` to `(int,int)?` widens this null value to 3 slots. This is called "type transitioning". All stdlib functions prototypes have been updated to reflect whether they return/accept a nullable or a strict value. This commit also contains refactoring from `const FunctionData*` to `FunctionPtr` and similar.
76 lines
2.7 KiB
C++
76 lines
2.7 KiB
C++
/*
|
|
This file is part of TON Blockchain source code.
|
|
|
|
TON Blockchain is free software; you can redistribute it and/or
|
|
modify it under the terms of the GNU General Public License
|
|
as published by the Free Software Foundation; either version 2
|
|
of the License, or (at your option) any later version.
|
|
|
|
TON Blockchain is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
GNU General Public License for more details.
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
along with TON Blockchain. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
In addition, as a special exception, the copyright holders give permission
|
|
to link the code of portions of this program with the OpenSSL library.
|
|
You must obey the GNU General Public License in all respects for all
|
|
of the code used other than OpenSSL. If you modify file(s) with this
|
|
exception, you may extend this exception to your version of the file(s),
|
|
but you are not obligated to do so. If you do not wish to do so, delete this
|
|
exception statement from your version. If you delete this exception statement
|
|
from all source files in the program, then also delete it here.
|
|
*/
|
|
#include "tolk.h"
|
|
#include "compiler-state.h"
|
|
|
|
/*
|
|
* This pipe finds unused symbols (global functions and variables) to strip them off codegen.
|
|
* It happens after converting AST to Op, so it does not traverse AST.
|
|
* In the future, when control flow graph is introduced, this should be done at AST level.
|
|
*/
|
|
|
|
namespace tolk {
|
|
|
|
static void mark_function_used_dfs(const std::unique_ptr<Op>& op);
|
|
|
|
static void mark_function_used(FunctionPtr fun_ref) {
|
|
if (!fun_ref->is_code_function() || fun_ref->is_really_used()) { // already handled
|
|
return;
|
|
}
|
|
|
|
fun_ref->mutate()->assign_is_really_used();
|
|
mark_function_used_dfs(std::get<FunctionBodyCode*>(fun_ref->body)->code->ops);
|
|
}
|
|
|
|
static void mark_global_var_used(GlobalVarPtr glob_ref) {
|
|
glob_ref->mutate()->assign_is_really_used();
|
|
}
|
|
|
|
static void mark_function_used_dfs(const std::unique_ptr<Op>& op) {
|
|
if (!op) {
|
|
return;
|
|
}
|
|
|
|
if (op->f_sym) { // for Op::_Call
|
|
mark_function_used(op->f_sym);
|
|
}
|
|
if (op->g_sym) { // for Op::_GlobVar
|
|
mark_global_var_used(op->g_sym);
|
|
}
|
|
mark_function_used_dfs(op->next);
|
|
mark_function_used_dfs(op->block0);
|
|
mark_function_used_dfs(op->block1);
|
|
}
|
|
|
|
void pipeline_find_unused_symbols() {
|
|
for (FunctionPtr fun_ref : G.all_functions) {
|
|
if (fun_ref->is_method_id_not_empty()) { // get methods, main and other entrypoints, regular functions with @method_id
|
|
mark_function_used(fun_ref);
|
|
}
|
|
}
|
|
}
|
|
|
|
} // namespace tolk
|