statement.cpp 5.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189
  1. // Part of the Carbon Language project, under the Apache License v2.0 with LLVM
  2. // Exceptions. See /LICENSE for license information.
  3. // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
  4. #include "executable_semantics/ast/statement.h"
  5. #include <iostream>
  6. namespace Carbon {
  7. auto MakeExpStmt(int line_num, Expression* exp) -> Statement* {
  8. auto* s = new Statement();
  9. s->line_num = line_num;
  10. s->tag = StatementKind::ExpressionStatement;
  11. s->u.exp = exp;
  12. return s;
  13. }
  14. auto MakeAssign(int line_num, Expression* lhs, Expression* rhs) -> Statement* {
  15. auto* s = new Statement();
  16. s->line_num = line_num;
  17. s->tag = StatementKind::Assign;
  18. s->u.assign.lhs = lhs;
  19. s->u.assign.rhs = rhs;
  20. return s;
  21. }
  22. auto MakeVarDef(int line_num, Expression* pat, Expression* init) -> Statement* {
  23. auto* s = new Statement();
  24. s->line_num = line_num;
  25. s->tag = StatementKind::VariableDefinition;
  26. s->u.variable_definition.pat = pat;
  27. s->u.variable_definition.init = init;
  28. return s;
  29. }
  30. auto MakeIf(int line_num, Expression* cond, Statement* then_stmt,
  31. Statement* else_stmt) -> Statement* {
  32. auto* s = new Statement();
  33. s->line_num = line_num;
  34. s->tag = StatementKind::If;
  35. s->u.if_stmt.cond = cond;
  36. s->u.if_stmt.then_stmt = then_stmt;
  37. s->u.if_stmt.else_stmt = else_stmt;
  38. return s;
  39. }
  40. auto MakeWhile(int line_num, Expression* cond, Statement* body) -> Statement* {
  41. auto* s = new Statement();
  42. s->line_num = line_num;
  43. s->tag = StatementKind::While;
  44. s->u.while_stmt.cond = cond;
  45. s->u.while_stmt.body = body;
  46. return s;
  47. }
  48. auto MakeBreak(int line_num) -> Statement* {
  49. auto* s = new Statement();
  50. s->line_num = line_num;
  51. s->tag = StatementKind::Break;
  52. return s;
  53. }
  54. auto MakeContinue(int line_num) -> Statement* {
  55. auto* s = new Statement();
  56. s->line_num = line_num;
  57. s->tag = StatementKind::Continue;
  58. return s;
  59. }
  60. auto MakeReturn(int line_num, Expression* e) -> Statement* {
  61. auto* s = new Statement();
  62. s->line_num = line_num;
  63. s->tag = StatementKind::Return;
  64. s->u.return_stmt = e;
  65. return s;
  66. }
  67. auto MakeSeq(int line_num, Statement* s1, Statement* s2) -> Statement* {
  68. auto* s = new Statement();
  69. s->line_num = line_num;
  70. s->tag = StatementKind::Sequence;
  71. s->u.sequence.stmt = s1;
  72. s->u.sequence.next = s2;
  73. return s;
  74. }
  75. auto MakeBlock(int line_num, Statement* stmt) -> Statement* {
  76. auto* s = new Statement();
  77. s->line_num = line_num;
  78. s->tag = StatementKind::Block;
  79. s->u.block.stmt = stmt;
  80. return s;
  81. }
  82. auto MakeMatch(int line_num, Expression* exp,
  83. std::list<std::pair<Expression*, Statement*>>* clauses)
  84. -> Statement* {
  85. auto* s = new Statement();
  86. s->line_num = line_num;
  87. s->tag = StatementKind::Match;
  88. s->u.match_stmt.exp = exp;
  89. s->u.match_stmt.clauses = clauses;
  90. return s;
  91. }
  92. void PrintStatement(Statement* s, int depth) {
  93. if (!s) {
  94. return;
  95. }
  96. if (depth == 0) {
  97. std::cout << " ... ";
  98. return;
  99. }
  100. switch (s->tag) {
  101. case StatementKind::Match:
  102. std::cout << "match (";
  103. PrintExp(s->u.match_stmt.exp);
  104. std::cout << ") {";
  105. if (depth < 0 || depth > 1) {
  106. std::cout << std::endl;
  107. for (auto& clause : *s->u.match_stmt.clauses) {
  108. std::cout << "case ";
  109. PrintExp(clause.first);
  110. std::cout << " =>" << std::endl;
  111. PrintStatement(clause.second, depth - 1);
  112. std::cout << std::endl;
  113. }
  114. } else {
  115. std::cout << "...";
  116. }
  117. std::cout << "}";
  118. break;
  119. case StatementKind::While:
  120. std::cout << "while (";
  121. PrintExp(s->u.while_stmt.cond);
  122. std::cout << ")" << std::endl;
  123. PrintStatement(s->u.while_stmt.body, depth - 1);
  124. break;
  125. case StatementKind::Break:
  126. std::cout << "break;";
  127. break;
  128. case StatementKind::Continue:
  129. std::cout << "continue;";
  130. break;
  131. case StatementKind::VariableDefinition:
  132. std::cout << "var ";
  133. PrintExp(s->u.variable_definition.pat);
  134. std::cout << " = ";
  135. PrintExp(s->u.variable_definition.init);
  136. std::cout << ";";
  137. break;
  138. case StatementKind::ExpressionStatement:
  139. PrintExp(s->u.exp);
  140. std::cout << ";";
  141. break;
  142. case StatementKind::Assign:
  143. PrintExp(s->u.assign.lhs);
  144. std::cout << " = ";
  145. PrintExp(s->u.assign.rhs);
  146. std::cout << ";";
  147. break;
  148. case StatementKind::If:
  149. std::cout << "if (";
  150. PrintExp(s->u.if_stmt.cond);
  151. std::cout << ")" << std::endl;
  152. PrintStatement(s->u.if_stmt.then_stmt, depth - 1);
  153. std::cout << std::endl << "else" << std::endl;
  154. PrintStatement(s->u.if_stmt.else_stmt, depth - 1);
  155. break;
  156. case StatementKind::Return:
  157. std::cout << "return ";
  158. PrintExp(s->u.return_stmt);
  159. std::cout << ";";
  160. break;
  161. case StatementKind::Sequence:
  162. PrintStatement(s->u.sequence.stmt, depth);
  163. if (depth < 0 || depth > 1) {
  164. std::cout << std::endl;
  165. }
  166. PrintStatement(s->u.sequence.next, depth - 1);
  167. break;
  168. case StatementKind::Block:
  169. std::cout << "{" << std::endl;
  170. PrintStatement(s->u.block.stmt, depth - 1);
  171. std::cout << std::endl << "}" << std::endl;
  172. }
  173. }
  174. } // namespace Carbon