Added more unit tests
This commit is contained in:
		
							parent
							
								
									53b626fd2f
								
							
						
					
					
						commit
						e93945f63a
					
				
							
								
								
									
										24
									
								
								.vscode/launch.json
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										24
									
								
								.vscode/launch.json
									
									
									
									
										vendored
									
									
								
							| @ -75,6 +75,30 @@ | |||||||
|                     "ignoreFailures": true |                     "ignoreFailures": true | ||||||
|                 } |                 } | ||||||
|             ] |             ] | ||||||
|  |         }, | ||||||
|  |         { | ||||||
|  |             "name": "Char Test", | ||||||
|  |             "type": "cppdbg", | ||||||
|  |             "request": "launch", | ||||||
|  |             "program": "${workspaceFolder}/build/tests/char_tests", | ||||||
|  |             "args": [], | ||||||
|  |             "stopAtEntry": false, | ||||||
|  |             "cwd": "${workspaceFolder}", | ||||||
|  |             "environment": [], | ||||||
|  |             "externalConsole": false, | ||||||
|  |             "MIMode": "gdb", | ||||||
|  |             "setupCommands": [ | ||||||
|  |                 { | ||||||
|  |                     "description": "Enable pretty-printing for gdb", | ||||||
|  |                     "text": "-enable-pretty-printing", | ||||||
|  |                     "ignoreFailures": true | ||||||
|  |                 }, | ||||||
|  |                 { | ||||||
|  |                     "description": "Set Disassembly Flavor to Intel", | ||||||
|  |                     "text": "-gdb-set disassembly-flavor intel", | ||||||
|  |                     "ignoreFailures": true | ||||||
|  |                 } | ||||||
|  |             ] | ||||||
|         } |         } | ||||||
|     ] |     ] | ||||||
| } | } | ||||||
							
								
								
									
										3
									
								
								.vscode/settings.json
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										3
									
								
								.vscode/settings.json
									
									
									
									
										vendored
									
									
								
							| @ -67,6 +67,7 @@ | |||||||
|         "thread": "cpp", |         "thread": "cpp", | ||||||
|         "cinttypes": "cpp", |         "cinttypes": "cpp", | ||||||
|         "typeinfo": "cpp", |         "typeinfo": "cpp", | ||||||
|         "variant": "cpp" |         "variant": "cpp", | ||||||
|  |         "list": "cpp" | ||||||
|     } |     } | ||||||
| } | } | ||||||
| @ -28,4 +28,5 @@ ADD_SUBDIRECTORY(tests) | |||||||
| ENABLE_TESTING() | ENABLE_TESTING() | ||||||
| ADD_TEST(NAME integer_tests COMMAND integer_tests) | ADD_TEST(NAME integer_tests COMMAND integer_tests) | ||||||
| ADD_TEST(NAME double_tests COMMAND double_tests) | ADD_TEST(NAME double_tests COMMAND double_tests) | ||||||
| ADD_TEST(NAME bool_tests COMMAND bool_tests) | ADD_TEST(NAME bool_tests COMMAND bool_tests) | ||||||
|  | ADD_TEST(NAME char_tests COMMAND char_tests) | ||||||
|  | |||||||
							
								
								
									
										20
									
								
								Hoo.g4
									
									
									
									
									
								
							
							
						
						
									
										20
									
								
								Hoo.g4
									
									
									
									
									
								
							| @ -2,9 +2,25 @@ grammar Hoo; | |||||||
| 
 | 
 | ||||||
| import hoolexer; | import hoolexer; | ||||||
| 
 | 
 | ||||||
| statement: literalStatement; | statement: expressionStatement; | ||||||
| 
 | 
 | ||||||
| literalStatement: literal ';'; | expressionStatement: expression ';'; | ||||||
|  | 
 | ||||||
|  | expression: | ||||||
|  | 	primary							# PrimaryExpression | ||||||
|  | 	| expression '*' expression		# MultiplicationExpression | ||||||
|  | 	| expression '/' expression		# DivisionExpression | ||||||
|  | 	| expression '%' expression		# ReminderExpression | ||||||
|  | 	| expression '+' expression		# AdditiveExpression | ||||||
|  | 	| expression '-' expression		# SubtractExpression | ||||||
|  | 	| expression '<' '<' expression	# LeftShiftExpression | ||||||
|  | 	| expression '>' '>' expression	# RightShiftExpression | ||||||
|  | 	| expression '&' '&' expression	# LogicalAndExpression | ||||||
|  | 	| expression '|' '|' expression	# LogicalOrExpression; | ||||||
|  | 
 | ||||||
|  | primary: | ||||||
|  | 	literal					# PrimaryLiteral | ||||||
|  | 	| '(' expression ')'	# NestedExpression; | ||||||
| 
 | 
 | ||||||
| literal: | literal: | ||||||
| 	INTEGER_LITERAL | 	INTEGER_LITERAL | ||||||
|  | |||||||
| @ -11,12 +11,25 @@ ADD_LIBRARY(hoocore STATIC | |||||||
|     ParseError.hpp |     ParseError.hpp | ||||||
|     ParseErrorHandler.hpp |     ParseErrorHandler.hpp | ||||||
|     ParseErrorHandler.cpp |     ParseErrorHandler.cpp | ||||||
|     ParseErrorException.hpp) |     ParseErrorException.hpp | ||||||
|  |     Utility.hpp | ||||||
|  |     UTF8Char.h) | ||||||
|  | 
 | ||||||
|  | ADD_LIBRARY(hooruntime STATIC | ||||||
|  |         UTF8Char.h | ||||||
|  |         UTF8Char.cpp | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
| ADD_EXECUTABLE(hoo | ADD_EXECUTABLE(hoo | ||||||
|     Hoo.cpp Visitor.cpp |     Hoo.cpp Visitor.cpp | ||||||
|     ${ANTLR_GENERATED_DIR}/HooBaseVisitor.cpp |     ${ANTLR_GENERATED_DIR}/HooBaseVisitor.cpp | ||||||
|     ${ANTLR_GENERATED_DIR}/HooVisitor.cpp |     ${ANTLR_GENERATED_DIR}/HooVisitor.cpp | ||||||
|     ${ANTLR_GENERATED_DIR}/HooLexer.cpp |     ${ANTLR_GENERATED_DIR}/HooLexer.cpp | ||||||
|     ${ANTLR_GENERATED_DIR}/HooParser.cpp) |     ${ANTLR_GENERATED_DIR}/HooParser.cpp) | ||||||
| ADD_DEPENDENCIES(hoo HooBaseVisitor) | 
 | ||||||
| TARGET_LINK_LIBRARIES(hoo antlr4-runtime LLVMCore LLVMSupport) | ADD_DEPENDENCIES(hoo HooBaseVisitor hoocore hooruntime) | ||||||
|  | TARGET_LINK_LIBRARIES(hoo hoocore | ||||||
|  |     hooruntime | ||||||
|  |     antlr4-runtime | ||||||
|  |     LLVMCore | ||||||
|  |     LLVMSupport) | ||||||
|  | |||||||
							
								
								
									
										5
									
								
								src/UTF8Char.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										5
									
								
								src/UTF8Char.cpp
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,5 @@ | |||||||
|  | #include "UTF8Char.h" | ||||||
|  | 
 | ||||||
|  | #include <string> | ||||||
|  | 
 | ||||||
|  | const std::string CHAR_TYPE_NAME = "hoo.UTF8Char"; | ||||||
							
								
								
									
										25
									
								
								src/UTF8Char.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										25
									
								
								src/UTF8Char.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,25 @@ | |||||||
|  | #ifndef UTF8_CHAR_H | ||||||
|  | #define UTF8_CHAR_H | ||||||
|  | 
 | ||||||
|  | #include <stdint.h> | ||||||
|  | #include <string> | ||||||
|  | 
 | ||||||
|  | extern "C" const std::string CHAR_TYPE_NAME; | ||||||
|  | const uint64_t UTF8_CHAR_BYTES = 2; | ||||||
|  | 
 | ||||||
|  | #ifdef __cplusplus | ||||||
|  | extern "C" | ||||||
|  | { | ||||||
|  | #endif | ||||||
|  |     typedef struct | ||||||
|  |     { | ||||||
|  | 
 | ||||||
|  |         uint8_t length; | ||||||
|  |         uint16_t bytes[UTF8_CHAR_BYTES]; | ||||||
|  | 
 | ||||||
|  |     } UTF8Char; | ||||||
|  | 
 | ||||||
|  | #ifdef __cplusplus | ||||||
|  | } | ||||||
|  | #endif | ||||||
|  | #endif // UTF8Char
 | ||||||
							
								
								
									
										198
									
								
								src/Utility.hpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										198
									
								
								src/Utility.hpp
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,198 @@ | |||||||
|  | #pragma once | ||||||
|  | 
 | ||||||
|  | #include "UTF8Char.h" | ||||||
|  | 
 | ||||||
|  | #include <string> | ||||||
|  | #include <vector> | ||||||
|  | #include <cstdint> | ||||||
|  | #include <stdexcept> | ||||||
|  | #include <map> | ||||||
|  | 
 | ||||||
|  | static const std::map<char, unsigned char> HEX_DIGITS = { | ||||||
|  |     {'0', 0}, | ||||||
|  |     {'1', 1}, | ||||||
|  |     {'2', 2}, | ||||||
|  |     {'3', 3}, | ||||||
|  |     {'4', 4}, | ||||||
|  |     {'5', 5}, | ||||||
|  |     {'6', 6}, | ||||||
|  |     {'7', 7}, | ||||||
|  |     {'8', 8}, | ||||||
|  |     {'9', 9}, | ||||||
|  |     {'a', 10}, | ||||||
|  |     {'b', 11}, | ||||||
|  |     {'c', 12}, | ||||||
|  |     {'d', 13}, | ||||||
|  |     {'e', 14}, | ||||||
|  |     {'f', 15}, | ||||||
|  |     {'A', 10}, | ||||||
|  |     {'B', 11}, | ||||||
|  |     {'C', 12}, | ||||||
|  |     {'D', 13}, | ||||||
|  |     {'E', 14}, | ||||||
|  |     {'F', 15}, | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | class Utility | ||||||
|  | { | ||||||
|  | public: | ||||||
|  |     static const std::uint8_t CHAR_NEWLINE = 10; | ||||||
|  |     static const std::uint8_t CHAR_CARRIAGE_RETURN = 13; | ||||||
|  |     static const std::uint8_t CHAR_TAB = 9; | ||||||
|  |     static const std::uint8_t CHAR_BACKSPACE = 8; | ||||||
|  |     static const std::uint8_t CHAR_FORMFEED = 12; | ||||||
|  |     static const std::uint8_t CHAR_BACK_SLASH = 92; | ||||||
|  |     static const std::uint8_t CHAR_SINGLE_QUOTE = 39; | ||||||
|  | 
 | ||||||
|  |     static const int CHAR_A_UPPER = 65; | ||||||
|  |     static const int CHAR_F_UPPER = 70; | ||||||
|  |     static const int CHAR_0_DIGIT = 48; | ||||||
|  |     static const int CHAR_9_DIGIT = 57; | ||||||
|  |     static const int CHAR_A_LOWER = 97; | ||||||
|  |     static const int CHAR_F_LOWER = 102; | ||||||
|  | 
 | ||||||
|  |     static UTF8Char getCharType(std::uint8_t ansiChar) | ||||||
|  |     { | ||||||
|  |         auto charType = UTF8Char(); | ||||||
|  |         charType.length = 1; | ||||||
|  |         charType.bytes[0] = ansiChar; | ||||||
|  |         return charType; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     static const int UTF8_HEXDIGIT_COUNT = 4; | ||||||
|  |     static const int UTF8_BYTE_COUNT = 2; | ||||||
|  | 
 | ||||||
|  |     static std::array<std::string, 2> splitUnicodeString(const std::string &input) | ||||||
|  |     { | ||||||
|  |         if (input.length() != UTF8_HEXDIGIT_COUNT) | ||||||
|  |         { | ||||||
|  |             throw std::invalid_argument("Invalid hex string."); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         std::array<std::string, 2> result; | ||||||
|  |         result[0] = input.substr(0, UTF8_BYTE_COUNT); | ||||||
|  |         result[1] = input.substr(2, UTF8_BYTE_COUNT); | ||||||
|  | 
 | ||||||
|  |         return result; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     static uint8_t | ||||||
|  |     hexStringToByte(const std::string &hexByteString) | ||||||
|  |     { | ||||||
|  |         if (hexByteString.length() != UTF8_BYTE_COUNT) | ||||||
|  |         { | ||||||
|  |             throw std::invalid_argument("Hex string must be exactly 2 characters long."); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         char c0 = hexByteString[0]; | ||||||
|  |         char c1 = hexByteString[1]; | ||||||
|  |         auto o0 = HEX_DIGITS.at(c0); | ||||||
|  |         auto o1 = HEX_DIGITS.at(c1); | ||||||
|  |         uint8_t result = (o0 * 16) + o1; | ||||||
|  |         return result; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     static uint16_t combineUint8s(uint8_t byte1, uint8_t byte2) | ||||||
|  |     { | ||||||
|  |         uint16_t result = (static_cast<uint16_t>(byte1) << 8) | byte2; | ||||||
|  |         return result; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     static std::vector<uint8_t> utf16CodePointToUTF8(uint16_t codePoint) | ||||||
|  |     { | ||||||
|  |         std::vector<uint8_t> utf8Bytes; | ||||||
|  | 
 | ||||||
|  |         if (codePoint <= 0x7F) | ||||||
|  |         { | ||||||
|  |             utf8Bytes.push_back(static_cast<uint8_t>(codePoint)); | ||||||
|  |         } | ||||||
|  |         else if (codePoint <= 0x7FF) | ||||||
|  |         { | ||||||
|  |             utf8Bytes.push_back(static_cast<uint8_t>(0xC0 | (codePoint >> 6))); | ||||||
|  |             utf8Bytes.push_back(static_cast<uint8_t>(0x80 | (codePoint & 0x3F))); | ||||||
|  |         } | ||||||
|  |         else if (codePoint <= 0xFFFF) | ||||||
|  |         { | ||||||
|  |             utf8Bytes.push_back(static_cast<uint8_t>(0xE0 | (codePoint >> 12))); | ||||||
|  |             utf8Bytes.push_back(static_cast<uint8_t>(0x80 | ((codePoint >> 6) & 0x3F))); | ||||||
|  |             utf8Bytes.push_back(static_cast<uint8_t>(0x80 | (codePoint & 0x3F))); | ||||||
|  |         } | ||||||
|  |         else | ||||||
|  |         { | ||||||
|  |             throw std::invalid_argument("Invalid Unicode code point (outside BMP)."); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         return utf8Bytes; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     static UTF8Char hexStringToUTF8(const std::string &hexString) | ||||||
|  |     { | ||||||
|  |         auto byteStrings = splitUnicodeString(hexString); | ||||||
|  |         auto byte1 = hexStringToByte(byteStrings[0]); | ||||||
|  |         auto byte2 = hexStringToByte(byteStrings[1]); | ||||||
|  |         auto codePoint = combineUint8s(byte1, byte2); | ||||||
|  |         auto utf8Bytes = utf16CodePointToUTF8(codePoint); | ||||||
|  |         auto utf8Char = UTF8Char{static_cast<uint8_t>(utf8Bytes.size()), {0, 0}}; | ||||||
|  |         for (auto index = 0; index < utf8Bytes.size(); ++index) | ||||||
|  |         { | ||||||
|  |             utf8Char.bytes[index] = utf8Bytes[index]; | ||||||
|  |         } | ||||||
|  |         return utf8Char; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     static UTF8Char | ||||||
|  |     getChar(std::string charText) | ||||||
|  |     { | ||||||
|  |         charText = charText.substr(1, charText.size() - 2); | ||||||
|  |         bool escape = false; | ||||||
|  |         for (auto index = 0; index < charText.size(); ++index) | ||||||
|  |         { | ||||||
|  |             auto c = charText[index]; | ||||||
|  |             if (escape) | ||||||
|  |             { | ||||||
|  |                 switch (c) | ||||||
|  |                 { | ||||||
|  |                 case 'n': | ||||||
|  |                     return getCharType(CHAR_NEWLINE); | ||||||
|  |                 case 'r': | ||||||
|  |                     return getCharType(CHAR_CARRIAGE_RETURN); | ||||||
|  |                 case 't': | ||||||
|  |                     return getCharType(CHAR_TAB); | ||||||
|  |                 case 'b': | ||||||
|  |                     return getCharType(CHAR_BACKSPACE); | ||||||
|  |                 case 'f': | ||||||
|  |                     return getCharType(CHAR_FORMFEED); | ||||||
|  |                 case '\\': | ||||||
|  |                     return getCharType(CHAR_BACK_SLASH); | ||||||
|  |                 case '\'': | ||||||
|  |                     return getCharType(CHAR_SINGLE_QUOTE); | ||||||
|  |                 case 'u': | ||||||
|  |                 { | ||||||
|  |                     std::string unicodeHexDigits = charText.substr(index + 1, 4); | ||||||
|  |                     auto utf8Char = hexStringToUTF8(unicodeHexDigits); | ||||||
|  |                     return utf8Char; | ||||||
|  |                 } | ||||||
|  |                 default: | ||||||
|  |                     throw std::runtime_error("Invalid char escape sequence"); | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |             else | ||||||
|  |             { | ||||||
|  |                 if (c == '\\') | ||||||
|  |                 { | ||||||
|  |                     escape = true; | ||||||
|  |                     continue; | ||||||
|  |                 } | ||||||
|  |                 else if (c <= 0x7F) | ||||||
|  |                 { | ||||||
|  |                     return getCharType((std::uint8_t)c); | ||||||
|  |                 } | ||||||
|  |                 else | ||||||
|  |                 { | ||||||
|  |                     throw std::runtime_error("Invalid ansi character"); | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         return UTF8Char({0, {0, 0}}); | ||||||
|  |     } | ||||||
|  | }; | ||||||
| @ -1,6 +1,8 @@ | |||||||
| #include "Visitor.hpp" | #include "Visitor.hpp" | ||||||
| #include "Node.hpp" | #include "Node.hpp" | ||||||
| #include "ParseErrorException.hpp" | #include "ParseErrorException.hpp" | ||||||
|  | #include "Utility.hpp" | ||||||
|  | #include "UTF8Char.h" | ||||||
| 
 | 
 | ||||||
| #include <llvm/IR/Constants.h> | #include <llvm/IR/Constants.h> | ||||||
| #include <llvm/IR/Type.h> | #include <llvm/IR/Type.h> | ||||||
| @ -15,12 +17,11 @@ Visitor::Visitor(const std::string &moduleName) : _moduleName(moduleName), | |||||||
| 
 | 
 | ||||||
| std::any Visitor::visitLiteral(HooParser::LiteralContext *ctx) | std::any Visitor::visitLiteral(HooParser::LiteralContext *ctx) | ||||||
| { | { | ||||||
|  |     auto value = ctx->INTEGER_LITERAL(); | ||||||
| #ifndef NDEBUG | #ifndef NDEBUG | ||||||
|     auto text = ctx->getText(); |     auto text = ctx->getText(); | ||||||
|     std::cout << "Literal: " << text << std::endl; |     std::cout << "Literal: " << text << std::endl; | ||||||
| #endif | #endif | ||||||
| 
 |  | ||||||
|     auto value = ctx->INTEGER_LITERAL(); |  | ||||||
|     if (value) |     if (value) | ||||||
|     { |     { | ||||||
|         auto decimalText = value->getText(); |         auto decimalText = value->getText(); | ||||||
| @ -59,9 +60,19 @@ std::any Visitor::visitLiteral(HooParser::LiteralContext *ctx) | |||||||
|     if (value) |     if (value) | ||||||
|     { |     { | ||||||
|         auto charText = value->getText(); |         auto charText = value->getText(); | ||||||
|         char charValue = charText[1]; |         auto charValue = Utility::getChar(charText); | ||||||
|         llvm::Type *charType = llvm::Type::getInt8Ty(*_context); | 
 | ||||||
|         llvm::Constant *charConstant = llvm::ConstantInt::get(charType, charValue, true); |         auto byteType = llvm::Type::getInt8Ty(*_context); | ||||||
|  |         auto byteArrayType = llvm::ArrayType::get(byteType, UTF8_CHAR_BYTES); | ||||||
|  |         llvm::StructType *charType = llvm::StructType::create(*_context, CHAR_TYPE_NAME); | ||||||
|  |         charType->setBody({byteType, byteArrayType}); | ||||||
|  |         llvm::Constant *charConstant = llvm::ConstantStruct::get( | ||||||
|  |             charType, | ||||||
|  |             {llvm::ConstantInt::get(byteType, charValue.length), | ||||||
|  |              llvm::ConstantArray::get(byteArrayType, { | ||||||
|  |                                                          llvm::ConstantInt::get(byteType, charValue.bytes[0]), | ||||||
|  |                                                          llvm::ConstantInt::get(byteType, charValue.bytes[1]), | ||||||
|  |                                                      })}); | ||||||
|         return std::any{Node(NODE_LITERAL, DATATYPE_CHAR, charConstant)}; |         return std::any{Node(NODE_LITERAL, DATATYPE_CHAR, charConstant)}; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
| @ -82,23 +93,44 @@ std::any Visitor::visitLiteral(HooParser::LiteralContext *ctx) | |||||||
|     throw ParseErrorException(_moduleName, line_no, char_pos, message); |     throw ParseErrorException(_moduleName, line_no, char_pos, message); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| std::any Visitor::visitLiteralStatement(HooParser::LiteralStatementContext *ctx) | std::any Visitor::visitPrimaryLiteral(HooParser::PrimaryLiteralContext *ctx) | ||||||
| { | { | ||||||
|     auto literal_context = ctx->literal(); |     auto literal = ctx->literal(); | ||||||
|     auto result = visitLiteral(literal_context); |     if (literal != nullptr) | ||||||
|     return result; |     { | ||||||
|  |         auto node = visitLiteral(literal); | ||||||
|  |         return node; | ||||||
|  |     } | ||||||
|  |     return std::any(); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | std::any Visitor::visitNestedExpression(HooParser::NestedExpressionContext *ctx) | ||||||
|  | { | ||||||
|  |     auto expr_ctx = ctx->expression(); | ||||||
|  |     auto node = visit(expr_ctx); | ||||||
|  |     return node; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | std::any Visitor::visitExpressionStatement(HooParser::ExpressionStatementContext *ctx) | ||||||
|  | { | ||||||
|  |     auto expressionCtx = ctx->expression(); | ||||||
|  |     if (expressionCtx != nullptr) | ||||||
|  |     { | ||||||
|  |         auto node = visit(expressionCtx); | ||||||
|  |         return node; | ||||||
|  |     } | ||||||
|  |     return std::any(); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| std::any Visitor::visitStatement(HooParser::StatementContext *ctx) | std::any Visitor::visitStatement(HooParser::StatementContext *ctx) | ||||||
| { | { | ||||||
|     auto listeral_stmt_ctx = ctx->literalStatement(); |     auto expr_stmt_ctx = ctx->expressionStatement(); | ||||||
|     std::cout << "Statement: " << ctx->getText() << std::endl; |     if (expr_stmt_ctx != nullptr) | ||||||
|     if (listeral_stmt_ctx != nullptr) |  | ||||||
|     { |     { | ||||||
|         auto result = visitLiteralStatement(listeral_stmt_ctx); |         auto node = visitExpressionStatement(expr_stmt_ctx); | ||||||
|         return result; |         return node; | ||||||
|     } |     } | ||||||
|     return nullptr; |     return std::any(); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| std::any Visitor::visitUnit(HooParser::UnitContext *ctx) | std::any Visitor::visitUnit(HooParser::UnitContext *ctx) | ||||||
|  | |||||||
| @ -21,7 +21,9 @@ public: | |||||||
| 
 | 
 | ||||||
| public: | public: | ||||||
|     std::any visitLiteral(HooParser::LiteralContext *ctx) override; |     std::any visitLiteral(HooParser::LiteralContext *ctx) override; | ||||||
|     std::any visitLiteralStatement(HooParser::LiteralStatementContext *ctx) override; |     std::any visitPrimaryLiteral(HooParser::PrimaryLiteralContext *ctx) override; | ||||||
|  |     std::any visitNestedExpression(HooParser::NestedExpressionContext *ctx) override; | ||||||
|  |     std::any visitExpressionStatement(HooParser::ExpressionStatementContext *ctx) override; | ||||||
|     std::any visitStatement(HooParser::StatementContext *ctx) override; |     std::any visitStatement(HooParser::StatementContext *ctx) override; | ||||||
|     std::any visitUnit(HooParser::UnitContext *ctx) override; |     std::any visitUnit(HooParser::UnitContext *ctx) override; | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -1,13 +1,17 @@ | |||||||
| FIND_PACKAGE (GTest REQUIRED) | FIND_PACKAGE (GTest REQUIRED) | ||||||
| 
 | 
 | ||||||
| ADD_EXECUTABLE(integer_tests integer_tests.cpp) | ADD_EXECUTABLE(integer_tests integer_tests.cpp) | ||||||
| TARGET_LINK_LIBRARIES(integer_tests GTest::GTest GTest::Main hoocore antlr4-runtime LLVMCore LLVMSupport) | TARGET_LINK_LIBRARIES(integer_tests GTest::GTest GTest::Main hoocore hooruntime antlr4-runtime LLVMCore LLVMSupport) | ||||||
| TARGET_INCLUDE_DIRECTORIES(integer_tests PRIVATE ${CMAKE_CURRENT_SOURCE_DIR}/../src) | TARGET_INCLUDE_DIRECTORIES(integer_tests PRIVATE ${CMAKE_CURRENT_SOURCE_DIR}/../src) | ||||||
| 
 | 
 | ||||||
| ADD_EXECUTABLE(double_tests double_tests.cpp) | ADD_EXECUTABLE(double_tests double_tests.cpp) | ||||||
| TARGET_LINK_LIBRARIES(double_tests GTest::GTest GTest::Main hoocore antlr4-runtime LLVMCore LLVMSupport) | TARGET_LINK_LIBRARIES(double_tests GTest::GTest GTest::Main hoocore hooruntime antlr4-runtime LLVMCore LLVMSupport) | ||||||
| TARGET_INCLUDE_DIRECTORIES(double_tests PRIVATE ${CMAKE_CURRENT_SOURCE_DIR}/../src) | TARGET_INCLUDE_DIRECTORIES(double_tests PRIVATE ${CMAKE_CURRENT_SOURCE_DIR}/../src) | ||||||
| 
 | 
 | ||||||
| ADD_EXECUTABLE(bool_tests bool_tests.cpp) | ADD_EXECUTABLE(bool_tests bool_tests.cpp) | ||||||
| TARGET_LINK_LIBRARIES(bool_tests GTest::GTest GTest::Main hoocore antlr4-runtime LLVMCore LLVMSupport) | TARGET_LINK_LIBRARIES(bool_tests GTest::GTest GTest::Main hoocore hooruntime antlr4-runtime LLVMCore LLVMSupport) | ||||||
| TARGET_INCLUDE_DIRECTORIES(bool_tests PRIVATE ${CMAKE_CURRENT_SOURCE_DIR}/../src) | TARGET_INCLUDE_DIRECTORIES(bool_tests PRIVATE ${CMAKE_CURRENT_SOURCE_DIR}/../src) | ||||||
|  | 
 | ||||||
|  | ADD_EXECUTABLE(char_tests char_tests.cpp) | ||||||
|  | TARGET_LINK_LIBRARIES(char_tests GTest::GTest GTest::Main hoocore hooruntime antlr4-runtime LLVMCore LLVMSupport) | ||||||
|  | TARGET_INCLUDE_DIRECTORIES(char_tests PRIVATE ${CMAKE_CURRENT_SOURCE_DIR}/../src) | ||||||
| @ -55,7 +55,7 @@ TEST_F(BoolTest, LiteralFalse) | |||||||
| TEST_F(BoolTest, InvalidLiteral) | TEST_F(BoolTest, InvalidLiteral) | ||||||
| { | { | ||||||
|     auto compiler = std::make_unique<Compiler>("notbool;", "main"); |     auto compiler = std::make_unique<Compiler>("notbool;", "main"); | ||||||
|     ASSERT_THROW(compiler->compile(), ParseErrorException); |     ASSERT_THROW(compiler->compile(), ParseCollectiveErrorException); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| TEST_F(BoolTest, MissingSemicolon) | TEST_F(BoolTest, MissingSemicolon) | ||||||
| @ -67,5 +67,5 @@ TEST_F(BoolTest, MissingSemicolon) | |||||||
| TEST_F(BoolTest, MixedCaseLiteral) | TEST_F(BoolTest, MixedCaseLiteral) | ||||||
| { | { | ||||||
|     auto compiler = std::make_unique<Compiler>("True;", "main"); |     auto compiler = std::make_unique<Compiler>("True;", "main"); | ||||||
|     ASSERT_THROW(compiler->compile(), ParseErrorException); |     ASSERT_THROW(compiler->compile(), ParseCollectiveErrorException); | ||||||
| } | } | ||||||
|  | |||||||
							
								
								
									
										67
									
								
								tests/char_tests.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										67
									
								
								tests/char_tests.cpp
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,67 @@ | |||||||
|  | #include "Compiler.hpp" | ||||||
|  | #include "Node.hpp" | ||||||
|  | #include "UTF8Char.h" | ||||||
|  | 
 | ||||||
|  | #include "llvm/IR/Constants.h" | ||||||
|  | #include <gtest/gtest.h> | ||||||
|  | 
 | ||||||
|  | // TEST(CharTest, SingleChar)
 | ||||||
|  | // {
 | ||||||
|  | //     auto compiler = std::make_unique<Compiler>("'a';", "main");
 | ||||||
|  | //     auto result = compiler->compile();
 | ||||||
|  | //     auto charNode = std::any_cast<Node>(result);
 | ||||||
|  | 
 | ||||||
|  | //     ASSERT_EQ(charNode.getNodeType(), NODE_LITERAL);
 | ||||||
|  | //     ASSERT_EQ(charNode.getDataType(), DATATYPE_CHAR);
 | ||||||
|  | 
 | ||||||
|  | //     auto value = charNode.getValue();
 | ||||||
|  | //     auto expected_value = llvm::ConstantInt::get(llvm::Type::getInt8Ty(*compiler->getContext()), 97);
 | ||||||
|  | 
 | ||||||
|  | //     ASSERT_EQ(value, expected_value);
 | ||||||
|  | // }
 | ||||||
|  | 
 | ||||||
|  | // TEST(CharTest, SpecialChar)
 | ||||||
|  | // {
 | ||||||
|  | //     auto compiler = std::make_unique<Compiler>("'\\n';", "main");
 | ||||||
|  | //     auto result = compiler->compile();
 | ||||||
|  | //     auto charNode = std::any_cast<Node>(result);
 | ||||||
|  | 
 | ||||||
|  | //     ASSERT_EQ(charNode.getNodeType(), NODE_LITERAL);
 | ||||||
|  | //     ASSERT_EQ(charNode.getDataType(), DATATYPE_CHAR);
 | ||||||
|  | 
 | ||||||
|  | //     auto value = charNode.getValue();
 | ||||||
|  | //     auto expected_value = llvm::ConstantInt::get(llvm::Type::getInt8Ty(*compiler->getContext()), 10);
 | ||||||
|  | 
 | ||||||
|  | //     ASSERT_EQ(value, expected_value);
 | ||||||
|  | // }
 | ||||||
|  | 
 | ||||||
|  | TEST(CharTest, UnicodeChar) | ||||||
|  | { | ||||||
|  |     auto compiler = std::make_unique<Compiler>("'\\u00E9';", "main"); | ||||||
|  |     auto result = compiler->compile(); | ||||||
|  |     auto charNode = std::any_cast<Node>(result); | ||||||
|  | 
 | ||||||
|  |     ASSERT_EQ(charNode.getNodeType(), NODE_LITERAL); | ||||||
|  |     ASSERT_EQ(charNode.getDataType(), DATATYPE_CHAR); | ||||||
|  | 
 | ||||||
|  |     auto value = charNode.getValue(); | ||||||
|  |     auto value_name = value->getType()->getStructName().str(); | ||||||
|  |     ASSERT_EQ(value_name, CHAR_TYPE_NAME); | ||||||
|  |     auto structValue = llvm::dyn_cast<llvm::ConstantStruct>(value); | ||||||
|  |     ASSERT_NE(structValue, nullptr); | ||||||
|  |     llvm::Value *length = llvm::dyn_cast<llvm::ConstantInt>(structValue->getAggregateElement((unsigned int)0)); | ||||||
|  |     ASSERT_NE(length, nullptr); | ||||||
|  |     auto length_value = llvm::dyn_cast<llvm::ConstantInt>(length)->getZExtValue(); | ||||||
|  |     ASSERT_EQ(length_value, 2); | ||||||
|  | 
 | ||||||
|  |     llvm::ConstantDataArray *array = llvm::dyn_cast<llvm::ConstantDataArray>(structValue->getAggregateElement((unsigned int)1)); | ||||||
|  |     ASSERT_NE(array, nullptr); | ||||||
|  |     auto byte1 = array->getElementAsConstant(0); | ||||||
|  |     auto byte2 = array->getElementAsConstant(1); | ||||||
|  | 
 | ||||||
|  |     auto byte1_value = llvm::dyn_cast<llvm::ConstantInt>(byte1)->getZExtValue(); | ||||||
|  |     ASSERT_EQ(byte1_value, 195); | ||||||
|  | 
 | ||||||
|  |     auto byte2_value = llvm::dyn_cast<llvm::ConstantInt>(byte2)->getZExtValue(); | ||||||
|  |     ASSERT_EQ(byte2_value, 169); | ||||||
|  | } | ||||||
| @ -1,9 +1,3 @@ | |||||||
| #include "Compiler.hpp" |  | ||||||
| #include "Node.hpp" |  | ||||||
| #include "llvm/IR/Constants.h" |  | ||||||
| 
 |  | ||||||
| #include <gtest/gtest.h> |  | ||||||
| 
 |  | ||||||
| #include "Compiler.hpp" | #include "Compiler.hpp" | ||||||
| #include "Node.hpp" | #include "Node.hpp" | ||||||
| #include "llvm/IR/Constants.h" | #include "llvm/IR/Constants.h" | ||||||
| @ -38,6 +32,11 @@ TEST(IntegerTest, LiteralOne) | |||||||
|     testIntegerLiteral("1;", 1); |     testIntegerLiteral("1;", 1); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | TEST(NestedIntegerTest, LiteralOne) | ||||||
|  | { | ||||||
|  |     testIntegerLiteral("(1);", 1); | ||||||
|  | } | ||||||
|  | 
 | ||||||
| TEST(IntegerTest, LiteralNumberPositive) | TEST(IntegerTest, LiteralNumberPositive) | ||||||
| { | { | ||||||
|     testIntegerLiteral("67890;", 67890); |     testIntegerLiteral("67890;", 67890); | ||||||
|  | |||||||
		Loading…
	
		Reference in New Issue
	
	Block a user