/** * BSD-style license; for more info see http://pmd.sourceforge.net/license.html */ package net.sourceforge.pmd.cpd; import static org.junit.Assert.assertEquals; import static org.junit.Assert.assertTrue; import java.util.Properties; import org.apache.commons.io.IOUtils; import org.junit.Test; import net.sourceforge.pmd.PMD; public class CPPTokenizerTest { @Test public void testUTFwithBOM() { Tokens tokens = parse("\ufeffint start()\n{ int ret = 1;\nreturn ret;\n}\n"); assertTrue(TokenEntry.getEOF() != tokens.getTokens().get(0)); assertEquals(15, tokens.size()); } @Test public void testUnicodeSupport() { String code = "\ufeff" + "#include <iostream>\n" + "#include <string>\n" + "\n" + "// example\n" + "\n" + "int main()\n" + "{\n" + " std::string text(\"ąęćśźńó\");\n" + " std::cout << text;\n" + " return 0;\n" + "}\n"; Tokens tokens = parse(code); assertTrue(TokenEntry.getEOF() != tokens.getTokens().get(0)); assertEquals(24, tokens.size()); } @Test public void testMultiLineMacros() { Tokens tokens = parse(TEST1); assertEquals(7, tokens.size()); } @Test public void testDollarSignInIdentifier() { parse(TEST2); } @Test public void testDollarSignStartingIdentifier() { parse(TEST3); } @Test public void testWideCharacters() { parse(TEST4); } @Test public void testContinuationIntraToken() { Tokens tokens = parse(TEST5); assertEquals(7, tokens.size()); } @Test public void testContinuationInterToken() { Tokens tokens = parse(TEST6); assertEquals(17, tokens.size()); } @Test public void testTokenizerWithSkipBlocks() throws Exception { String test = IOUtils.toString(CPPTokenizerTest.class.getResourceAsStream("cpp/cpp_with_asm.cpp")); Tokens tokens = parse(test, true); assertEquals(19, tokens.size()); } @Test public void testTokenizerWithSkipBlocksPattern() throws Exception { String test = IOUtils.toString(CPPTokenizerTest.class.getResourceAsStream("cpp/cpp_with_asm.cpp")); Tokens tokens = parse(test, true, "#if debug|#endif"); assertEquals(31, tokens.size()); } @Test public void testTokenizerWithoutSkipBlocks() throws Exception { String test = IOUtils.toString(CPPTokenizerTest.class.getResourceAsStream("cpp/cpp_with_asm.cpp")); Tokens tokens = parse(test, false); assertEquals(37, tokens.size()); } @Test // ASM code containing the '@' character public void testAsmWithAtSign() { Tokens tokens = parse(TEST7); assertEquals(22, tokens.size()); } @Test public void testEOLCommentInPreprocessingDirective() { parse("#define LSTFVLES_CPP //*" + PMD.EOL); } @Test public void testEmptyCharacter() { Tokens tokens = parse("std::wstring wsMessage( sMessage.length(), L'');" + PMD.EOL); assertEquals(15, tokens.size()); } @Test public void testHexCharacter() { Tokens tokens = parse("if (*pbuf == '\\0x05')" + PMD.EOL); assertEquals(8, tokens.size()); } @Test public void testWhiteSpaceEscape() { Tokens tokens = parse("szPath = m_sdcacheDir + _T(\"\\ oMedia\");" + PMD.EOL); assertEquals(10, tokens.size()); } @Test public void testRawStringLiteral() { String code = "const char* const KDefaultConfig = R\"(\n" + " [Sinks.1]\n" + " Destination=Console\n" + " AutoFlush=true\n" + " Format=\"[%TimeStamp%] %ThreadId% %QueryIdHigh% %QueryIdLow% %LoggerFile%:%Line% (%Severity%) - %Message%\"\n" + " Filter=\"%Severity% >= WRN\"\n" + ")\";\n"; Tokens tokens = parse(code); assertTrue(TokenEntry.getEOF() != tokens.getTokens().get(0)); assertEquals(9, tokens.size()); } private Tokens parse(String snippet) { return parse(snippet, false); } private Tokens parse(String snippet, boolean skipBlocks) { return parse(snippet, skipBlocks, null); } private Tokens parse(String snippet, boolean skipBlocks, String skipPattern) { Properties properties = new Properties(); properties.setProperty(Tokenizer.OPTION_SKIP_BLOCKS, Boolean.toString(skipBlocks)); if (skipPattern != null) { properties.setProperty(Tokenizer.OPTION_SKIP_BLOCKS_PATTERN, skipPattern); } CPPTokenizer tokenizer = new CPPTokenizer(); tokenizer.setProperties(properties); SourceCode code = new SourceCode(new SourceCode.StringCodeLoader(snippet)); Tokens tokens = new Tokens(); tokenizer.tokenize(code, tokens); return tokens; } private static final String TEST1 = "#define FOO a +\\" + PMD.EOL + " b +\\" + PMD.EOL + " c +\\" + PMD.EOL + " d +\\" + PMD.EOL + " e +\\" + PMD.EOL + " f +\\" + PMD.EOL + " g" + PMD.EOL + " void main() {}"; private static final String TEST2 = " void main() { int x$y = 42; }"; private static final String TEST3 = " void main() { int $x = 42; }"; private static final String TEST4 = " void main() { char x = L'a'; }"; private static final String TEST5 = "v\\" + PMD.EOL + "o\\" + PMD.EOL + "i\\" + PMD.EOL + "d\\" + PMD.EOL + " \\" + PMD.EOL + "m\\" + PMD.EOL + "a\\" + PMD.EOL + "i\\" + PMD.EOL + "n\\" + PMD.EOL + "(\\" + PMD.EOL + ")\\" + PMD.EOL + " \\" + PMD.EOL + "{\\" + PMD.EOL + " \\" + PMD.EOL + "}\\" + PMD.EOL; private static final String TEST6 = "#include <iostream>" + PMD.EOL + PMD.EOL + "int main()" + PMD.EOL + "{" + PMD.EOL + " std::cout << \"Hello, \" \\" + PMD.EOL + " \"world!\\n\";" + PMD.EOL + " return 0;" + PMD.EOL + "}"; private static final String TEST7 = "asm void eSPI_boot()" + PMD.EOL + "{" + PMD.EOL + " // setup stack pointer" + PMD.EOL + " lis r1, _stack_addr@h" + PMD.EOL + " ori r1, r1, _stack_addr@l" + PMD.EOL + "}"; }