001/* 002 * Copyright (c) 2012-2021 Institut National des Sciences Appliquées de Lyon (INSA Lyon) and others 003 * 004 * This program and the accompanying materials are made available under the 005 * terms of the Eclipse Public License 2.0 which is available at 006 * http://www.eclipse.org/legal/epl-2.0. 007 * 008 * SPDX-License-Identifier: EPL-2.0 009 */ 010 011package gololang.ir; 012 013import java.util.List; 014import java.util.LinkedList; 015 016/** 017 * Represents a conditional branching. 018 * 019 * <p>Typically an {@code if else} node. 020 */ 021public final class ConditionalBranching extends GoloStatement<ConditionalBranching> { 022 023 private ExpressionStatement<?> condition; 024 private Block trueBlock; 025 private ConditionalBranching elseConditionalBranching; 026 private Block falseBlock; 027 028 private ConditionalBranching() { 029 super(); 030 } 031 032 /** 033 * Creates an empty conditional branch. 034 */ 035 public static ConditionalBranching branch() { 036 return new ConditionalBranching(); 037 } 038 039 /** 040 * Full branch construction in one call. 041 * 042 * <p>Less readable than the fluent API, but useful when doing meta-generation 043 * 044 * @param condition the test condition 045 * @param trueBlock the block to execute when the condition is true 046 * @param falseBlock the block to execute when the condition is false 047 * @param elseBranch a nested conditional branch 048 */ 049 public static ConditionalBranching create(Object condition, 050 Object trueBlock, 051 Object falseBlock, 052 Object elseBranch) { 053 return new ConditionalBranching().condition(condition) 054 .whenTrue(trueBlock) 055 .whenFalse(falseBlock) 056 .elseBranch(elseBranch); 057 } 058 059 protected ConditionalBranching self() { return this; } 060 061 /** 062 * Defines the condition of the branching. 063 * 064 * <p>This is a builder method. 065 * 066 * @param cond the expression defining the condition. If {@code null}, the condition is set to {@code false}. 067 * @return the branching. 068 */ 069 public ConditionalBranching condition(Object cond) { 070 if (cond == null) { 071 this.condition = ConstantStatement.of(false); 072 } else { 073 this.condition = ExpressionStatement.of(cond); 074 } 075 makeParentOf(this.condition); 076 return this; 077 } 078 079 /** 080 * Defines the block executed when the condition evaluates to {@code true}. 081 * 082 * <p>This is a builder method. 083 */ 084 public ConditionalBranching whenTrue(Object block) { 085 this.trueBlock = makeParentOf(Block.of(block)); 086 return this; 087 } 088 089 /** 090 * Defines the block executed when the condition evaluates to {@code false}. 091 * 092 * <p>This is a builder method. 093 * 094 * @see #otherwise(Object) 095 */ 096 public ConditionalBranching whenFalse(Object block) { 097 if (block == null) { 098 this.falseBlock = null; 099 } else { 100 this.falseBlock = makeParentOf(Block.of(block)); 101 } 102 return this; 103 } 104 105 /** 106 * Defines a nested conditional branch. 107 * 108 * <p>This is a builder method. 109 * 110 * <p>This represents an {@code else if} branch. 111 * 112 * @see #otherwise(Object) 113 */ 114 public ConditionalBranching elseBranch(Object elseBranch) { 115 this.elseConditionalBranching = makeParentOf((ConditionalBranching) elseBranch); 116 return this; 117 } 118 119 /** 120 * Defines a block to execute when {@code false} or a nested branch according to the given element. 121 * 122 * <p>This is a builder method; it's the preferred way to define an alternative to the true block. 123 * 124 * @param alternative a {@link Block} to execute when {@code false} or a {@code ConditionalBranching} defining a 125 * nested branch. 126 */ 127 public ConditionalBranching otherwise(Object alternative) { 128 if (alternative instanceof ConditionalBranching) { 129 return elseBranch(alternative); 130 } 131 return whenFalse(alternative); 132 } 133 134 public ExpressionStatement<?> getCondition() { 135 return condition; 136 } 137 138 public Block getTrueBlock() { 139 return trueBlock; 140 } 141 142 public Block getFalseBlock() { 143 return falseBlock; 144 } 145 146 public boolean hasFalseBlock() { 147 return falseBlock != null; 148 } 149 150 public ConditionalBranching getElseConditionalBranching() { 151 return elseConditionalBranching; 152 } 153 154 public boolean hasElseConditionalBranching() { 155 return elseConditionalBranching != null; 156 } 157 158 public boolean returnsFromBothBranches() { 159 if (hasFalseBlock()) { 160 return trueBlock.hasReturn() && falseBlock.hasReturn(); 161 } else if (hasElseConditionalBranching()) { 162 return trueBlock.hasReturn() && elseConditionalBranching.returnsFromBothBranches(); 163 } else { 164 return false; 165 } 166 } 167 168 @Override 169 public String toString() { 170 return String.format("if %s %s%s", condition, trueBlock, 171 hasFalseBlock() ? " else " + falseBlock.toString() 172 : hasElseConditionalBranching() ? " else " + elseConditionalBranching.toString() 173 : ""); 174 } 175 176 /** 177 * {@inheritDoc} 178 */ 179 @Override 180 public void accept(GoloIrVisitor visitor) { 181 visitor.visitConditionalBranching(this); 182 } 183 184 /** 185 * {@inheritDoc} 186 */ 187 @Override 188 public List<GoloElement<?>> children() { 189 LinkedList<GoloElement<?>> children = new LinkedList<>(); 190 children.add(condition); 191 children.add(trueBlock); 192 if (falseBlock != null) { 193 children.add(falseBlock); 194 } 195 if (elseConditionalBranching != null) { 196 children.add(elseConditionalBranching); 197 } 198 return children; 199 } 200 201 /** 202 * {@inheritDoc} 203 */ 204 @Override 205 protected void replaceElement(GoloElement<?> original, GoloElement<?> newElement) { 206 if (condition == original) { 207 condition(newElement); 208 } else if (trueBlock == original) { 209 whenTrue(newElement); 210 } else if (elseConditionalBranching == original && newElement instanceof ConditionalBranching) { 211 elseBranch(newElement); 212 } else if (elseConditionalBranching == original) { 213 whenFalse(newElement instanceof Noop ? null : newElement); 214 elseBranch(null); 215 } else if (falseBlock == original && newElement instanceof ConditionalBranching) { 216 elseBranch(newElement); 217 whenFalse(null); 218 } else if (falseBlock == original) { 219 whenFalse(newElement instanceof Noop ? null : newElement); 220 } else { 221 throw cantReplace(original, newElement); 222 } 223 } 224 225}