mirror of
				https://github.com/ton-blockchain/ton
				synced 2025-03-09 15:40:10 +00:00 
			
		
		
		
	This is a very big change. If FunC has `.methods()` and `~methods()`, Tolk has only dot, one and only way to call a `.method()`. A method may mutate an object, or may not. It's a behavioral and semantic difference from FunC. - `cs.loadInt(32)` modifies a slice and returns an integer - `b.storeInt(x, 32)` modifies a builder - `b = b.storeInt()` also works, since it not only modifies, but returns - chained methods also work, they return `self` - everything works exactly as expected, similar to JS - no runtime overhead, exactly same Fift instructions - custom methods are created with ease - tilda `~` does not exist in Tolk at all
		
			
				
	
	
		
			155 lines
		
	
	
	
		
			7.8 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			155 lines
		
	
	
	
		
			7.8 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /*
 | |
|     This file is part of TON Blockchain Library.
 | |
| 
 | |
|     TON Blockchain Library is free software: you can redistribute it and/or modify
 | |
|     it under the terms of the GNU Lesser 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 Library 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 Lesser General Public License for more details.
 | |
| 
 | |
|     You should have received a copy of the GNU Lesser General Public License
 | |
|     along with TON Blockchain Library.  If not, see <http://www.gnu.org/licenses/>.
 | |
| */
 | |
| #pragma once
 | |
| 
 | |
| #include "ast.h"
 | |
| #include "platform-utils.h"
 | |
| 
 | |
| /*
 | |
|  *   A module implementing base functionality of read-only traversing a vertex tree.
 | |
|  *   Since a vertex in general doesn't store a vector of children, iterating is possible only for concrete node_type.
 | |
|  * E.g., for ast_if_statement, visit nodes cond, if-body and else-body. For ast_string_const, nothing. And so on.
 | |
|  *   Visitors below are helpers to inherit from and handle specific vertex types.
 | |
|  *
 | |
|  *   Note, that absence of "children" in ASTNodeBase is not a drawback. Instead, it encourages you to think
 | |
|  * about types and match the type system.
 | |
|  *
 | |
|  *   The visitor is read-only, it does not modify visited nodes (except if you purposely call mutating methods).
 | |
|  *   For example, if you want to replace "beginCell()" call with "begin_cell", a visitor isn't enough for you.
 | |
|  *   To replace vertices, consider another API: ast-replacer.h.
 | |
|  */
 | |
| 
 | |
| namespace tolk {
 | |
| 
 | |
| class ASTVisitor {
 | |
| protected:
 | |
|   GNU_ATTRIBUTE_ALWAYS_INLINE static void visit_children(const ASTNodeLeaf* v) {
 | |
|     static_cast<void>(v);
 | |
|   }
 | |
| 
 | |
|   GNU_ATTRIBUTE_ALWAYS_INLINE void visit_children(const ASTNodeUnary* v) {
 | |
|     visit(v->child);
 | |
|   }
 | |
| 
 | |
|   GNU_ATTRIBUTE_ALWAYS_INLINE void visit_children(const ASTNodeBinary* v) {
 | |
|     visit(v->lhs);
 | |
|     visit(v->rhs);
 | |
|   }
 | |
| 
 | |
|   GNU_ATTRIBUTE_ALWAYS_INLINE void visit_children(const ASTNodeVararg* v) {
 | |
|     for (AnyV child : v->children) {
 | |
|       visit(child);
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   virtual void visit(AnyV v) = 0;
 | |
| 
 | |
| public:
 | |
|   virtual ~ASTVisitor() = default;
 | |
| };
 | |
| 
 | |
| class ASTVisitorFunctionBody : public ASTVisitor {
 | |
| protected:
 | |
|   using parent = ASTVisitorFunctionBody;
 | |
| 
 | |
|   virtual void visit(V<ast_empty> v)                     { return visit_children(v); }
 | |
|   virtual void visit(V<ast_parenthesized_expr> v)        { return visit_children(v); }
 | |
|   virtual void visit(V<ast_tensor> v)                    { return visit_children(v); }
 | |
|   virtual void visit(V<ast_tensor_square> v)             { return visit_children(v); }
 | |
|   virtual void visit(V<ast_identifier> v)                { return visit_children(v); }
 | |
|   virtual void visit(V<ast_int_const> v)                 { return visit_children(v); }
 | |
|   virtual void visit(V<ast_string_const> v)              { return visit_children(v); }
 | |
|   virtual void visit(V<ast_bool_const> v)                { return visit_children(v); }
 | |
|   virtual void visit(V<ast_null_keyword> v)              { return visit_children(v); }
 | |
|   virtual void visit(V<ast_self_keyword> v)              { return visit_children(v); }
 | |
|   virtual void visit(V<ast_function_call> v)             { return visit_children(v); }
 | |
|   virtual void visit(V<ast_dot_method_call> v)            { return visit_children(v); }
 | |
|   virtual void visit(V<ast_underscore> v)                { return visit_children(v); }
 | |
|   virtual void visit(V<ast_unary_operator> v)            { return visit_children(v); }
 | |
|   virtual void visit(V<ast_binary_operator> v)           { return visit_children(v); }
 | |
|   virtual void visit(V<ast_ternary_operator> v)          { return visit_children(v); }
 | |
|   virtual void visit(V<ast_return_statement> v)          { return visit_children(v); }
 | |
|   virtual void visit(V<ast_sequence> v)                  { return visit_children(v); }
 | |
|   virtual void visit(V<ast_repeat_statement> v)          { return visit_children(v); }
 | |
|   virtual void visit(V<ast_while_statement> v)           { return visit_children(v); }
 | |
|   virtual void visit(V<ast_do_while_statement> v)        { return visit_children(v); }
 | |
|   virtual void visit(V<ast_try_catch_statement> v)       { return visit_children(v); }
 | |
|   virtual void visit(V<ast_if_statement> v)              { return visit_children(v); }
 | |
|   virtual void visit(V<ast_local_var> v)                 { return visit_children(v); }
 | |
|   virtual void visit(V<ast_local_vars_declaration> v)    { return visit_children(v); }
 | |
|   virtual void visit(V<ast_asm_body> v)                  { return visit_children(v); }
 | |
| 
 | |
|   void visit(AnyV v) final {
 | |
|     switch (v->type) {
 | |
|       case ast_empty:                           return visit(v->as<ast_empty>());
 | |
|       case ast_parenthesized_expr:              return visit(v->as<ast_parenthesized_expr>());
 | |
|       case ast_tensor:                          return visit(v->as<ast_tensor>());
 | |
|       case ast_tensor_square:                   return visit(v->as<ast_tensor_square>());
 | |
|       case ast_identifier:                      return visit(v->as<ast_identifier>());
 | |
|       case ast_int_const:                       return visit(v->as<ast_int_const>());
 | |
|       case ast_string_const:                    return visit(v->as<ast_string_const>());
 | |
|       case ast_bool_const:                      return visit(v->as<ast_bool_const>());
 | |
|       case ast_null_keyword:                    return visit(v->as<ast_null_keyword>());
 | |
|       case ast_self_keyword:                    return visit(v->as<ast_self_keyword>());
 | |
|       case ast_function_call:                   return visit(v->as<ast_function_call>());
 | |
|       case ast_dot_method_call:                 return visit(v->as<ast_dot_method_call>());
 | |
|       case ast_underscore:                      return visit(v->as<ast_underscore>());
 | |
|       case ast_unary_operator:                  return visit(v->as<ast_unary_operator>());
 | |
|       case ast_binary_operator:                 return visit(v->as<ast_binary_operator>());
 | |
|       case ast_ternary_operator:                return visit(v->as<ast_ternary_operator>());
 | |
|       case ast_return_statement:                return visit(v->as<ast_return_statement>());
 | |
|       case ast_sequence:                        return visit(v->as<ast_sequence>());
 | |
|       case ast_repeat_statement:                return visit(v->as<ast_repeat_statement>());
 | |
|       case ast_while_statement:                 return visit(v->as<ast_while_statement>());
 | |
|       case ast_do_while_statement:              return visit(v->as<ast_do_while_statement>());
 | |
|       case ast_throw_statement:                 return visit(v->as<ast_throw_statement>());
 | |
|       case ast_assert_statement:                return visit(v->as<ast_assert_statement>());
 | |
|       case ast_try_catch_statement:             return visit(v->as<ast_try_catch_statement>());
 | |
|       case ast_if_statement:                    return visit(v->as<ast_if_statement>());
 | |
|       case ast_local_var:                       return visit(v->as<ast_local_var>());
 | |
|       case ast_local_vars_declaration:          return visit(v->as<ast_local_vars_declaration>());
 | |
|       case ast_asm_body:                        return visit(v->as<ast_asm_body>());
 | |
|       default:
 | |
|         throw UnexpectedASTNodeType(v, "ASTVisitorFunctionBody::visit");
 | |
|     }
 | |
|   }
 | |
| 
 | |
| public:
 | |
|   void start_visiting_function(V<ast_function_declaration> v_function) {
 | |
|     visit(v_function->get_body());
 | |
|   }
 | |
| };
 | |
| 
 | |
| class ASTVisitorAllFunctionsInFile : public ASTVisitorFunctionBody {
 | |
| protected:
 | |
|   using parent = ASTVisitorAllFunctionsInFile;
 | |
| 
 | |
|   virtual bool should_enter_function(V<ast_function_declaration> v) = 0;
 | |
| 
 | |
| public:
 | |
|   void start_visiting_file(V<ast_tolk_file> v_file) {
 | |
|     for (AnyV v : v_file->get_toplevel_declarations()) {
 | |
|       if (auto v_func = v->try_as<ast_function_declaration>()) {
 | |
|         if (should_enter_function(v_func)) {
 | |
|           visit(v_func->get_body());
 | |
|         }
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| };
 | |
| 
 | |
| } // namespace tolk
 |