/* * Copyright (c) 2012-2017 The ANTLR Project. All rights reserved. * Use of this file is governed by the BSD 3-clause license that * can be found in the LICENSE.txt file in the project root. */ package org.antlr.v4.codegen.target; import org.antlr.v4.codegen.CodeGenerator; import org.antlr.v4.codegen.Target; import org.antlr.v4.codegen.UnicodeEscapes; import org.antlr.v4.tool.ErrorType; import org.antlr.v4.tool.ast.GrammarAST; import org.stringtemplate.v4.NumberRenderer; import org.stringtemplate.v4.ST; import org.stringtemplate.v4.STErrorListener; import org.stringtemplate.v4.STGroup; import org.stringtemplate.v4.StringRenderer; import org.stringtemplate.v4.misc.STMessage; import java.util.Arrays; import java.util.HashSet; import java.util.Set; public class CppTarget extends Target { protected static final String[] cppKeywords = { "alignas", "alignof", "and", "and_eq", "asm", "auto", "bitand", "bitor", "bool", "break", "case", "catch", "char", "char16_t", "char32_t", "class", "compl", "concept", "const", "constexpr", "const_cast", "continue", "decltype", "default", "delete", "do", "double", "dynamic_cast", "else", "enum", "explicit", "export", "extern", "false", "float", "for", "friend", "goto", "if", "inline", "int", "long", "mutable", "namespace", "new", "noexcept", "not", "not_eq", "nullptr", "operator", "or", "or_eq", "private", "protected", "public", "register", "reinterpret_cast", "requires", "return", "short", "signed", "sizeof", "static", "static_assert", "static_cast", "struct", "switch", "template", "this", "thread_local", "throw", "true", "try", "typedef", "typeid", "typename", "union", "unsigned", "using", "virtual", "void", "volatile", "wchar_t", "while", "xor", "xor_eq" }; /** Avoid grammar symbols in this set to prevent conflicts in gen'd code. */ protected final Set<String> badWords = new HashSet<String>(); public CppTarget(CodeGenerator gen) { super(gen, "Cpp"); } public String getVersion() { return "4.7"; } public boolean needsHeader() { return true; } public Set<String> getBadWords() { if (badWords.isEmpty()) { addBadWords(); } return badWords; } protected void addBadWords() { badWords.addAll(Arrays.asList(cppKeywords)); badWords.add("rule"); badWords.add("parserRule"); } @Override public String encodeIntAsCharEscape(int v) { return "0x" + Integer.toHexString(v) + ", "; } @Override public int getSerializedATNSegmentLimit() { // 65535 is the class file format byte limit for a UTF-8 encoded string literal // 3 is the maximum number of bytes it takes to encode a value in the range 0-0xFFFF return 65535 / 3; } @Override public String getRecognizerFileName(boolean header) { ST extST = getTemplates().getInstanceOf(header ? "headerFileExtension" : "codeFileExtension"); String recognizerName = gen.g.getRecognizerName(); return recognizerName+extST.render(); } @Override public String getListenerFileName(boolean header) { assert gen.g.name != null; ST extST = getTemplates().getInstanceOf(header ? "headerFileExtension" : "codeFileExtension"); String listenerName = gen.g.name + "Listener"; return listenerName+extST.render(); } @Override public String getVisitorFileName(boolean header) { assert gen.g.name != null; ST extST = getTemplates().getInstanceOf(header ? "headerFileExtension" : "codeFileExtension"); String listenerName = gen.g.name + "Visitor"; return listenerName+extST.render(); } @Override public String getBaseListenerFileName(boolean header) { assert gen.g.name != null; ST extST = getTemplates().getInstanceOf(header ? "headerFileExtension" : "codeFileExtension"); String listenerName = gen.g.name + "BaseListener"; return listenerName+extST.render(); } @Override public String getBaseVisitorFileName(boolean header) { assert gen.g.name != null; ST extST = getTemplates().getInstanceOf(header ? "headerFileExtension" : "codeFileExtension"); String listenerName = gen.g.name + "BaseVisitor"; return listenerName+extST.render(); } @Override protected boolean visibleGrammarSymbolCausesIssueInGeneratedCode(GrammarAST idNode) { return getBadWords().contains(idNode.getText()); } @Override protected STGroup loadTemplates() { STGroup result = super.loadTemplates(); result.registerRenderer(Integer.class, new NumberRenderer()); result.registerRenderer(String.class, new StringRenderer()); result.setListener(new STErrorListener() { @Override public void compileTimeError(STMessage msg) { reportError(msg); } @Override public void runTimeError(STMessage msg) { reportError(msg); } @Override public void IOError(STMessage msg) { reportError(msg); } @Override public void internalError(STMessage msg) { reportError(msg); } private void reportError(STMessage msg) { getCodeGenerator().tool.errMgr.toolError(ErrorType.STRING_TEMPLATE_WARNING, msg.cause, msg.toString()); } }); return result; } @Override protected void appendUnicodeEscapedCodePoint(int codePoint, StringBuilder sb) { // C99 and Python share the same escaping style. UnicodeEscapes.appendPythonStyleEscapedCodePoint(codePoint, sb); } }