Compare commits
	
		
			17 Commits
		
	
	
		
			v1.0.4-rc.
			...
			fd452f6643
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| fd452f6643 | |||
| c437a39e1d | |||
| 8b53b6e4d1 | |||
| ff42dfd3a4 | |||
| d8c5327180 | |||
| 42b26045a6 | |||
| d41f71f17d | |||
| c19c0bc80b | |||
| 86149d246a | |||
| 1e55b11eed | |||
| ce225b2888 | |||
| eb3d345c5a | |||
| 6cf4977a6c | |||
| ae226c9a84 | |||
| bf3b063e96 | |||
| e071877045 | |||
| af90f90156 | 
| @@ -22,10 +22,9 @@ jobs: | |||||||
|       - name: Building/Compiling/Installing Project |       - name: Building/Compiling/Installing Project | ||||||
|         run: | |         run: | | ||||||
|           cd ${{ gitea.workspace }} |           cd ${{ gitea.workspace }} | ||||||
|           vcpkg install |  | ||||||
|           mkdir build |           mkdir build | ||||||
|           cd build |           cd build | ||||||
|           cmake -A x64 -DCMAKE_BUILD_TYPE=Release -DCMAKE_TOOLCHAIN_FILE=$env:VCPKG_ROOT\scripts\buildsystems\vcpkg.cmake .. |           cmake -A x64 -DCMAKE_BUILD_TYPE=Release -DCMAKE_TOOLCHAIN_FILE="$env:VCPKG_ROOT\scripts\buildsystems\vcpkg.cmake" .. | ||||||
|           cmake --build . --config Release |           cmake --build . --config Release | ||||||
|           cmake --install . |           cmake --install . | ||||||
|  |  | ||||||
| @@ -56,6 +55,7 @@ jobs: | |||||||
|           files: |- |           files: |- | ||||||
|             ehs-windows-amd64.zip |             ehs-windows-amd64.zip | ||||||
|           api_key: '${{secrets.RELEASE_TOKEN}}' |           api_key: '${{secrets.RELEASE_TOKEN}}' | ||||||
|  |           pre_release: false | ||||||
|  |  | ||||||
|   Linux-AMD64-Build: |   Linux-AMD64-Build: | ||||||
|     runs-on: linux-x86_64 |     runs-on: linux-x86_64 | ||||||
| @@ -101,6 +101,7 @@ jobs: | |||||||
|           files: |- |           files: |- | ||||||
|             ehs-linux-amd64.zip |             ehs-linux-amd64.zip | ||||||
|           api_key: '${{secrets.RELEASE_TOKEN}}' |           api_key: '${{secrets.RELEASE_TOKEN}}' | ||||||
|  |           pre_release: false | ||||||
|  |  | ||||||
|   Linux-AARCH64-Build: |   Linux-AARCH64-Build: | ||||||
|     runs-on: linux-aarch64 |     runs-on: linux-aarch64 | ||||||
| @@ -145,3 +146,4 @@ jobs: | |||||||
|           files: |- |           files: |- | ||||||
|             ehs-linux-aarch64.zip |             ehs-linux-aarch64.zip | ||||||
|           api_key: '${{secrets.RELEASE_TOKEN}}' |           api_key: '${{secrets.RELEASE_TOKEN}}' | ||||||
|  |           pre_release: false | ||||||
| @@ -194,11 +194,8 @@ set(EHS_SOURCES | |||||||
|     src/io/socket/ehc/NetClientCh.cpp |     src/io/socket/ehc/NetClientCh.cpp | ||||||
|     src/io/socket/ehc/NetServerCh.cpp |     src/io/socket/ehc/NetServerCh.cpp | ||||||
|     src/io/socket/ehc/NetUtils.cpp |     src/io/socket/ehc/NetUtils.cpp | ||||||
|         src/io/socket/BaseICMP.cpp |     include/ehs/io/socket/BaseICMP.h src/io/socket/BaseICMP.cpp | ||||||
|         include/ehs/io/socket/ICMP_LNX.h |  | ||||||
|         src/io/socket/ICMP_LNX.cpp |  | ||||||
|     include/ehs/io/socket/ICMP.h |     include/ehs/io/socket/ICMP.h | ||||||
|         src/io/socket/ICMP_LNX.cpp |  | ||||||
| ) | ) | ||||||
|  |  | ||||||
| if (IS_OS_WINDOWS) | if (IS_OS_WINDOWS) | ||||||
| @@ -217,7 +214,9 @@ if (IS_OS_WINDOWS) | |||||||
|         src/io/COM.cpp include/ehs/io/COM.h |         src/io/COM.cpp include/ehs/io/COM.h | ||||||
|         src/system/CPU_MSVC_AMD64.asm src/HRNG_MSVC.asm src/Math_MSVC_AMD64.asm |         src/system/CPU_MSVC_AMD64.asm src/HRNG_MSVC.asm src/Math_MSVC_AMD64.asm | ||||||
|         src/io/Directory_W32.cpp include/ehs/io/Directory_W32.h |         src/io/Directory_W32.cpp include/ehs/io/Directory_W32.h | ||||||
|  |         include/ehs/io/socket/ICMP_W32.h src/io/socket/ICMP_W32.cpp | ||||||
|     ) |     ) | ||||||
|  |  | ||||||
| elseif (IS_OS_LINUX) | elseif (IS_OS_LINUX) | ||||||
|     list(APPEND EHS_SOURCES |     list(APPEND EHS_SOURCES | ||||||
|         src/io/socket/UDP_BSD.cpp include/ehs/io/socket/UDP_BSD.h |         src/io/socket/UDP_BSD.cpp include/ehs/io/socket/UDP_BSD.h | ||||||
| @@ -234,6 +233,7 @@ elseif (IS_OS_LINUX) | |||||||
|         src/system/User.cpp include/ehs/system/User.h |         src/system/User.cpp include/ehs/system/User.h | ||||||
|         src/io/Directory_LNX.cpp include/ehs/io/Directory_LNX.h |         src/io/Directory_LNX.cpp include/ehs/io/Directory_LNX.h | ||||||
|         src/io/Usb_LNX.cpp include/ehs/io/Usb_LNX.h |         src/io/Usb_LNX.cpp include/ehs/io/Usb_LNX.h | ||||||
|  |         include/ehs/io/socket/ICMP_LNX.h src/io/socket/ICMP_LNX.cpp | ||||||
|     ) |     ) | ||||||
|  |  | ||||||
|     #set(LINUX_WINDOW_SYSTEM "Wayland" CACHE STRING "Linux Window System") |     #set(LINUX_WINDOW_SYSTEM "Wayland" CACHE STRING "Linux Window System") | ||||||
| @@ -262,8 +262,7 @@ endif() | |||||||
| #message("${CMAKE_CXX_FLAGS}") | #message("${CMAKE_CXX_FLAGS}") | ||||||
|  |  | ||||||
| add_library(EHS_Stc STATIC ${EHS_SOURCES}) | add_library(EHS_Stc STATIC ${EHS_SOURCES}) | ||||||
| add_library(EHS_Dyn SHARED ${EHS_SOURCES} | add_library(EHS_Dyn SHARED ${EHS_SOURCES}) | ||||||
|         include/ehs/io/socket/BaseICMP.h) |  | ||||||
| add_executable(StrToHash src/StrToHash.cpp) | add_executable(StrToHash src/StrToHash.cpp) | ||||||
|  |  | ||||||
| target_compile_definitions(EHS_Dyn PRIVATE EHS_LIB_EXPORT) | target_compile_definitions(EHS_Dyn PRIVATE EHS_LIB_EXPORT) | ||||||
| @@ -313,7 +312,7 @@ target_link_libraries(EHS_Dyn OpenSSL::SSL OpenSSL::Crypto ZLIB::ZLIB) | |||||||
| target_link_libraries(StrToHash OpenSSL::SSL OpenSSL::Crypto ZLIB::ZLIB) | target_link_libraries(StrToHash OpenSSL::SSL OpenSSL::Crypto ZLIB::ZLIB) | ||||||
|  |  | ||||||
| if (IS_OS_WINDOWS) | if (IS_OS_WINDOWS) | ||||||
|     target_link_libraries(EHS_Dyn avrt ws2_32) |     target_link_libraries(EHS_Dyn avrt ws2_32 IPHLPAPI) | ||||||
|     target_link_libraries(StrToHash ws2_32 avrt EHS_Stc) |     target_link_libraries(StrToHash ws2_32 avrt EHS_Stc) | ||||||
| elseif (IS_OS_LINUX) | elseif (IS_OS_LINUX) | ||||||
|     if (LINUX_WINDOW_SYSTEM STREQUAL "Wayland") |     if (LINUX_WINDOW_SYSTEM STREQUAL "Wayland") | ||||||
|   | |||||||
| @@ -22,6 +22,7 @@ This project does not fully follow the C++ standard. | |||||||
| - Semaphores | - Semaphores | ||||||
| - CPU information and features at runtime | - CPU information and features at runtime | ||||||
| - HTTP(S) Socket Layer | - HTTP(S) Socket Layer | ||||||
|  | - ICMP & ICMPv6 Socket | ||||||
| - TCP Socket | - TCP Socket | ||||||
| - UDP Socket | - UDP Socket | ||||||
| - COM (Serial) IO | - COM (Serial) IO | ||||||
|   | |||||||
| @@ -25,6 +25,8 @@ namespace ehs | |||||||
| 		IP version; | 		IP version; | ||||||
|  |  | ||||||
| 	public: | 	public: | ||||||
|  | 		virtual ~BaseICMP() = default; | ||||||
|  |  | ||||||
| 		BaseICMP(); | 		BaseICMP(); | ||||||
|  |  | ||||||
| 		BaseICMP(IP version); | 		BaseICMP(IP version); | ||||||
| @@ -37,17 +39,30 @@ namespace ehs | |||||||
|  |  | ||||||
| 		BaseICMP &operator=(const BaseICMP &icmp); | 		BaseICMP &operator=(const BaseICMP &icmp); | ||||||
|  |  | ||||||
| 		virtual UInt_64 Send(const Str_8 &address, ICMP_Header header, const Byte *data, UInt_64 size); | 		virtual void Release(); | ||||||
|  |  | ||||||
| 		virtual UInt_64 Receive(Str_8 &address, ICMP_Header header, Serializer<UInt_64> &data); | 		UInt_64 Send(const Str_8 &address, ICMP_Header header, const Byte *data, UInt_64 size); | ||||||
|  |  | ||||||
|  | 		UInt_64 Receive(Str_8 &address, ICMP_Header &header, Serializer<UInt_64> &data); | ||||||
|  |  | ||||||
| 		void SendEchoRequest(const Str_8 &address, ICMP_EchoRequest er, const Byte *data, UInt_64 size); | 		void SendEchoRequest(const Str_8 &address, ICMP_EchoRequest er, const Byte *data, UInt_64 size); | ||||||
|  |  | ||||||
| 		virtual void SetReceiveTimeout(UInt_64 timeout); | 		virtual void SetReceiveTimeout(UInt_64 timeout); | ||||||
|  |  | ||||||
|  | 		IP GetVersion() const; | ||||||
|  |  | ||||||
| 		virtual bool IsValid() const; | 		virtual bool IsValid() const; | ||||||
|  |  | ||||||
| 	protected: | 	protected: | ||||||
| 		static UInt_16 ComputeChecksum(UInt_16 *buffer, Size length); | 		static UInt_16 ComputeChecksumV4(UInt_16 *buffer, Size length); | ||||||
|  |  | ||||||
|  | 	private: | ||||||
|  | 		virtual UInt_64 SendV6(const Str_8 &address, ICMP_Header header, const Byte *data, UInt_64 size); | ||||||
|  |  | ||||||
|  | 		virtual UInt_64 SendV4(const Str_8 &address, ICMP_Header header, const Byte *data, UInt_64 size); | ||||||
|  |  | ||||||
|  | 		virtual UInt_64 ReceiveV6(Str_8 &address, ICMP_Header &header, Serializer<UInt_64> &data) const; | ||||||
|  |  | ||||||
|  | 		virtual UInt_64 ReceiveV4(Str_8 &address, ICMP_Header &header, Serializer<UInt_64> &data) const; | ||||||
| 	}; | 	}; | ||||||
| } | } | ||||||
|   | |||||||
| @@ -1,7 +1,9 @@ | |||||||
| #pragma once | #pragma once | ||||||
|  |  | ||||||
|  | #include "ehs/system/OS.h" | ||||||
|  |  | ||||||
| #ifdef EHS_OS_WINDOWS | #ifdef EHS_OS_WINDOWS | ||||||
| #include "BaseICMP.h" | 	#include "ICMP_W32.h" | ||||||
| #else | #else | ||||||
| #include "ICMP_LNX.h" | 	#include "ICMP_LNX.h" | ||||||
| #endif | #endif | ||||||
| @@ -2,14 +2,26 @@ | |||||||
|  |  | ||||||
| #include "BaseICMP.h" | #include "BaseICMP.h" | ||||||
|  |  | ||||||
|  | #include <netinet/in.h> | ||||||
|  |  | ||||||
| namespace ehs | namespace ehs | ||||||
| { | { | ||||||
| 	class ICMP : public BaseICMP | 	struct PseudoICMPv6_Header | ||||||
|  | 	{ | ||||||
|  | 		sockaddr_in6 src; | ||||||
|  | 		sockaddr_in6 dst; | ||||||
|  | 		UInt_32 length; | ||||||
|  | 	}; | ||||||
|  |  | ||||||
|  | 	class ICMP final : public BaseICMP | ||||||
| 	{ | 	{ | ||||||
| 	private: | 	private: | ||||||
| 		Int_32 hdl; | 		Int_32 hdl; | ||||||
|  | 		sockaddr_in6 src; | ||||||
|  |  | ||||||
| 	public: | 	public: | ||||||
|  | 		~ICMP() override; | ||||||
|  |  | ||||||
| 		ICMP(); | 		ICMP(); | ||||||
|  |  | ||||||
| 		ICMP(IP version); | 		ICMP(IP version); | ||||||
| @@ -22,12 +34,27 @@ namespace ehs | |||||||
|  |  | ||||||
| 		ICMP &operator=(const ICMP &icmp); | 		ICMP &operator=(const ICMP &icmp); | ||||||
|  |  | ||||||
| 		UInt_64 Send(const Str_8 &address, ICMP_Header header, const Byte *data, UInt_64 size) override; | 		void Release() override; | ||||||
|  |  | ||||||
| 		UInt_64 Receive(Str_8 &address, ICMP_Header header, Serializer<UInt_64> &data) override; |  | ||||||
|  |  | ||||||
| 		void SetReceiveTimeout(UInt_64 timeout) override; | 		void SetReceiveTimeout(UInt_64 timeout) override; | ||||||
|  |  | ||||||
| 		bool IsValid() const override; | 		bool IsValid() const override; | ||||||
|  |  | ||||||
|  | 	private: | ||||||
|  | 		static bool IsLinkLocal(const in6_addr &addr); | ||||||
|  |  | ||||||
|  | 		static sockaddr_in6 RetrieveSrcAddress(); | ||||||
|  |  | ||||||
|  | 		static UInt_32 CalculatePseudoHeaderChecksum(const PseudoICMPv6_Header &header); | ||||||
|  |  | ||||||
|  | 		UInt_16 ComputeChecksumV6(UInt_16* buffer, Size length, const sockaddr_in6& dst); | ||||||
|  |  | ||||||
|  | 		UInt_64 SendV6(const Str_8 &address, ICMP_Header header, const Byte *data, UInt_64 size) override; | ||||||
|  |  | ||||||
|  | 		UInt_64 SendV4(const Str_8 &address, ICMP_Header header, const Byte *data, UInt_64 size) override; | ||||||
|  |  | ||||||
|  | 		UInt_64 ReceiveV6(Str_8 &address, ICMP_Header &header, Serializer<UInt_64> &data) const override; | ||||||
|  |  | ||||||
|  | 		UInt_64 ReceiveV4(Str_8 &address, ICMP_Header &header, Serializer<UInt_64> &data) const override; | ||||||
| 	}; | 	}; | ||||||
| } | } | ||||||
							
								
								
									
										63
									
								
								include/ehs/io/socket/ICMP_W32.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										63
									
								
								include/ehs/io/socket/ICMP_W32.h
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,63 @@ | |||||||
|  | #pragma once | ||||||
|  |  | ||||||
|  | #include "BaseICMP.h" | ||||||
|  | #include "ehs/System/OS.h" | ||||||
|  |  | ||||||
|  | #include <winsock2.h> | ||||||
|  | #include <WS2tcpip.h> | ||||||
|  | #include <iphlpapi.h> | ||||||
|  |  | ||||||
|  | namespace ehs | ||||||
|  | { | ||||||
|  | 	struct PseudoICMPv6_Header | ||||||
|  | 	{ | ||||||
|  | 		sockaddr_in6 src; | ||||||
|  | 		sockaddr_in6 dst; | ||||||
|  | 		UInt_32 length; | ||||||
|  | 	}; | ||||||
|  |  | ||||||
|  | 	class ICMP : public BaseICMP | ||||||
|  | 	{ | ||||||
|  | 	private: | ||||||
|  | 		Socket hdl; | ||||||
|  | 		sockaddr_in6 src; | ||||||
|  |  | ||||||
|  | 	public: | ||||||
|  |         ~ICMP() override; | ||||||
|  |  | ||||||
|  | 		ICMP(); | ||||||
|  |  | ||||||
|  | 		ICMP(IP version); | ||||||
|  |  | ||||||
|  | 		ICMP(ICMP &&icmp) noexcept; | ||||||
|  |  | ||||||
|  | 		ICMP(const ICMP &icmp); | ||||||
|  |  | ||||||
|  | 		ICMP &operator=(ICMP &&icmp) noexcept; | ||||||
|  |  | ||||||
|  | 		ICMP &operator=(const ICMP &icmp); | ||||||
|  |  | ||||||
|  |         void Release() override; | ||||||
|  |  | ||||||
|  | 		void SetReceiveTimeout(UInt_64 timeout) override; | ||||||
|  |  | ||||||
|  | 		bool IsValid() const override; | ||||||
|  |  | ||||||
|  |     private: | ||||||
|  | 		static bool IsLinkLocal(const in6_addr &addr); | ||||||
|  |  | ||||||
|  | 		static sockaddr_in6 RetrieveSrcAddress(); | ||||||
|  |  | ||||||
|  | 		static UInt_32 CalculatePseudoHeaderChecksum(const PseudoICMPv6_Header &header); | ||||||
|  |  | ||||||
|  | 		UInt_16 ComputeChecksumV6(UInt_16* buffer, Size length, const sockaddr_in6& dst); | ||||||
|  |  | ||||||
|  | 		UInt_64 SendV6(const Str_8 &address, ICMP_Header header, const Byte *data, UInt_64 size) override; | ||||||
|  |  | ||||||
|  | 		UInt_64 SendV4(const Str_8 &address, ICMP_Header header, const Byte *data, UInt_64 size) override; | ||||||
|  |  | ||||||
|  | 		UInt_64 ReceiveV6(Str_8 &address, ICMP_Header &header, Serializer<UInt_64> &data) const override; | ||||||
|  |  | ||||||
|  | 		UInt_64 ReceiveV4(Str_8 &address, ICMP_Header &header, Serializer<UInt_64> &data) const override; | ||||||
|  |     }; | ||||||
|  | } | ||||||
| @@ -1,8 +1,6 @@ | |||||||
| #include "ehs/io/socket/BaseICMP.h" | #include "ehs/io/socket/BaseICMP.h" | ||||||
| #include "ehs/Serializer.h" | #include "ehs/Serializer.h" | ||||||
|  |  | ||||||
| #include <netinet/ip.h> |  | ||||||
|  |  | ||||||
| namespace ehs | namespace ehs | ||||||
| { | { | ||||||
|  |  | ||||||
| @@ -47,14 +45,24 @@ namespace ehs | |||||||
| 		return *this; | 		return *this; | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	UInt_64 BaseICMP::Send(const Str_8 &address, ICMP_Header header, const Byte *data, UInt_64 size) | 	void BaseICMP::Release() | ||||||
| 	{ | 	{ | ||||||
| 		return 0; |  | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	UInt_64 BaseICMP::Receive(Str_8 &address, ICMP_Header header, Serializer<UInt_64> &data) | 	UInt_64 BaseICMP::Send(const Str_8 &address, ICMP_Header header, const Byte *data, UInt_64 size) | ||||||
| 	{ | 	{ | ||||||
| 		return 0; | 		if (GetVersion() == IP::V6) | ||||||
|  | 			return SendV6(address, header, data, size); | ||||||
|  |  | ||||||
|  | 		return SendV4(address, header, data, size); | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	UInt_64 BaseICMP::Receive(Str_8 &address, ICMP_Header &header, Serializer<UInt_64> &data) | ||||||
|  | 	{ | ||||||
|  | 		if (GetVersion() == IP::V6) | ||||||
|  | 			return ReceiveV6(address, header, data); | ||||||
|  |  | ||||||
|  | 		return ReceiveV4(address, header, data); | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	void BaseICMP::SendEchoRequest(const Str_8 &address, ICMP_EchoRequest er, const Byte *data, UInt_64 size) | 	void BaseICMP::SendEchoRequest(const Str_8 &address, ICMP_EchoRequest er, const Byte *data, UInt_64 size) | ||||||
| @@ -67,7 +75,7 @@ namespace ehs | |||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		ICMP_Header header = { | 		ICMP_Header header = { | ||||||
| 			8, | 			version == IP::V6 ? (UInt_8)128 : (UInt_8)8, | ||||||
| 			0, | 			0, | ||||||
| 			0 | 			0 | ||||||
| 		}; | 		}; | ||||||
| @@ -87,12 +95,17 @@ namespace ehs | |||||||
| 	{ | 	{ | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
|  | 	IP BaseICMP::GetVersion() const | ||||||
|  | 	{ | ||||||
|  | 		return version; | ||||||
|  | 	} | ||||||
|  |  | ||||||
| 	bool BaseICMP::IsValid() const | 	bool BaseICMP::IsValid() const | ||||||
| 	{ | 	{ | ||||||
| 		return false; | 		return false; | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	UInt_16 BaseICMP::ComputeChecksum(UInt_16 *buffer, Size length) | 	UInt_16 BaseICMP::ComputeChecksumV4(UInt_16 *buffer, Size length) | ||||||
| 	{ | 	{ | ||||||
| 		Size sum = 0; | 		Size sum = 0; | ||||||
| 		while (length > 1) | 		while (length > 1) | ||||||
| @@ -109,4 +122,24 @@ namespace ehs | |||||||
|  |  | ||||||
| 		return (UInt_16)~sum; | 		return (UInt_16)~sum; | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
|  | 	UInt_64 BaseICMP::SendV6(const Str_8 &address, ICMP_Header header, const Byte *data, UInt_64 size) | ||||||
|  | 	{ | ||||||
|  | 		return 0; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	UInt_64 BaseICMP::SendV4(const Str_8 &address, ICMP_Header header, const Byte *data, UInt_64 size) | ||||||
|  | 	{ | ||||||
|  | 		return 0; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	UInt_64 BaseICMP::ReceiveV6(Str_8 &address, ICMP_Header &header, Serializer<UInt_64> &data) const | ||||||
|  | 	{ | ||||||
|  | 		return 0; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	UInt_64 BaseICMP::ReceiveV4(Str_8 &address, ICMP_Header &header, Serializer<UInt_64> &data) const | ||||||
|  | 	{ | ||||||
|  | 		return 0; | ||||||
|  | 	} | ||||||
| } | } | ||||||
|   | |||||||
| @@ -5,18 +5,30 @@ | |||||||
| #include <arpa/inet.h> | #include <arpa/inet.h> | ||||||
| #include <netinet/in.h> | #include <netinet/in.h> | ||||||
| #include <netinet/ip.h> | #include <netinet/ip.h> | ||||||
|  | #include <ifaddrs.h> | ||||||
|  | #include <unistd.h> | ||||||
|  |  | ||||||
| namespace ehs | namespace ehs | ||||||
| { | { | ||||||
|  | 	ICMP::~ICMP() | ||||||
|  | 	{ | ||||||
|  | 		if (close(hdl) == -1) | ||||||
|  | 			EHS_LOG_INT(LogType::ERR, 0, "Failed to close socket."); | ||||||
|  | 	} | ||||||
|  |  | ||||||
| 	ICMP::ICMP() | 	ICMP::ICMP() | ||||||
| 		: hdl(EHS_INVALID_SOCKET) | 		: hdl(EHS_INVALID_SOCKET), src{} | ||||||
| 	{ | 	{ | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	ICMP::ICMP(const IP version) | 	ICMP::ICMP(const IP version) | ||||||
| 		: BaseICMP(version) | 		: BaseICMP(version), src{} | ||||||
| 	{ | 	{ | ||||||
|  | 		if (version == IP::V6) | ||||||
|  | 			hdl = socket(AF_INET6, SOCK_RAW, IPPROTO_ICMPV6); | ||||||
|  | 		else | ||||||
| 			hdl = socket(AF_INET, SOCK_RAW, IPPROTO_ICMP); | 			hdl = socket(AF_INET, SOCK_RAW, IPPROTO_ICMP); | ||||||
|  |  | ||||||
| 		if (hdl < 0) | 		if (hdl < 0) | ||||||
| 		{ | 		{ | ||||||
| 			EHS_LOG_INT(LogType::ERR, 0, "Failed to create ICMP socket with error #" + Str_8::FromNum(errno) + "."); | 			EHS_LOG_INT(LogType::ERR, 0, "Failed to create ICMP socket with error #" + Str_8::FromNum(errno) + "."); | ||||||
| @@ -27,13 +39,14 @@ namespace ehs | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	ICMP::ICMP(ICMP &&icmp) noexcept | 	ICMP::ICMP(ICMP &&icmp) noexcept | ||||||
| 		: BaseICMP((BaseICMP &&)icmp), hdl(icmp.hdl) | 		: BaseICMP((BaseICMP &&)icmp), hdl(icmp.hdl), src(icmp.src) | ||||||
| 	{ | 	{ | ||||||
| 		icmp.hdl = EHS_INVALID_SOCKET; | 		icmp.hdl = EHS_INVALID_SOCKET; | ||||||
|  | 		icmp.src = {}; | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	ICMP::ICMP(const ICMP &icmp) | 	ICMP::ICMP(const ICMP &icmp) | ||||||
| 		: BaseICMP(icmp), hdl(icmp.hdl) | 		: BaseICMP(icmp), hdl(icmp.hdl), src{} | ||||||
| 	{ | 	{ | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| @@ -45,8 +58,10 @@ namespace ehs | |||||||
| 		BaseICMP::operator=((BaseICMP &&)icmp); | 		BaseICMP::operator=((BaseICMP &&)icmp); | ||||||
|  |  | ||||||
| 		hdl = icmp.hdl; | 		hdl = icmp.hdl; | ||||||
|  | 		src = icmp.src; | ||||||
|  |  | ||||||
| 		icmp.hdl = EHS_INVALID_SOCKET; | 		icmp.hdl = EHS_INVALID_SOCKET; | ||||||
|  | 		icmp.src = {}; | ||||||
|  |  | ||||||
| 		return *this; | 		return *this; | ||||||
| 	} | 	} | ||||||
| @@ -59,11 +74,136 @@ namespace ehs | |||||||
| 		BaseICMP::operator=(icmp); | 		BaseICMP::operator=(icmp); | ||||||
|  |  | ||||||
| 		hdl = icmp.hdl; | 		hdl = icmp.hdl; | ||||||
|  | 		src = {}; | ||||||
|  |  | ||||||
| 		return *this; | 		return *this; | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	UInt_64 ICMP::Send(const Str_8 &address, ICMP_Header header, const Byte *data, const UInt_64 size) | 	void ICMP::Release() | ||||||
|  | 	{ | ||||||
|  | 		if (close(hdl) == -1) | ||||||
|  | 		{ | ||||||
|  | 			EHS_LOG_INT(LogType::ERR, 0, "Failed to close socket."); | ||||||
|  |  | ||||||
|  | 			return; | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		hdl = EHS_INVALID_SOCKET; | ||||||
|  |  | ||||||
|  | 		EHS_LOG_SUCCESS(); | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	void ICMP::SetReceiveTimeout(const UInt_64 timeout) | ||||||
|  | 	{ | ||||||
|  | 		timeval result = {}; | ||||||
|  | 		result.tv_sec = (long)timeout; | ||||||
|  | 		result.tv_usec = 0; | ||||||
|  |  | ||||||
|  | 		if (setsockopt(hdl, SOL_SOCKET, SO_RCVTIMEO, &result, sizeof(result)) < 0) | ||||||
|  | 		{ | ||||||
|  | 			EHS_LOG_INT(LogType::WARN, 0, "Failed to set receive timeout with error #" + Str_8::FromNum(errno) + "."); | ||||||
|  |  | ||||||
|  | 			return; | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		EHS_LOG_SUCCESS(); | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	bool ICMP::IsValid() const | ||||||
|  | 	{ | ||||||
|  | 		return hdl != EHS_INVALID_SOCKET; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	bool ICMP::IsLinkLocal(const in6_addr &addr) | ||||||
|  | 	{ | ||||||
|  | 		return addr.s6_addr[0] == 0xfe && (addr.s6_addr[1] & 0xc0) == 0x80; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	sockaddr_in6 ICMP::RetrieveSrcAddress() | ||||||
|  | 	{ | ||||||
|  | 		ifaddrs *ifaddr; | ||||||
|  | 		sockaddr_in6 addr = {}; | ||||||
|  |  | ||||||
|  | 		if (getifaddrs(&ifaddr) == -1) | ||||||
|  | 		{ | ||||||
|  | 			EHS_LOG_INT(LogType::ERR, 0, "Failed to retrieve public address with error #" + Str_8::FromNum(errno) + "."); | ||||||
|  |  | ||||||
|  | 			return addr; | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		for (ifaddrs *ifa = ifaddr; ifa; ifa = ifa->ifa_next) | ||||||
|  | 		{ | ||||||
|  | 			if (ifa->ifa_addr == nullptr || ifa->ifa_addr->sa_family != AF_INET6) | ||||||
|  | 				continue; | ||||||
|  |  | ||||||
|  | 			addr = *(sockaddr_in6 *)ifa->ifa_addr; | ||||||
|  | 			if (!addr.sin6_addr.s6_addr32[0] || IsLinkLocal(addr.sin6_addr)) | ||||||
|  | 				continue; | ||||||
|  |  | ||||||
|  | 			break; | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		freeifaddrs(ifaddr); | ||||||
|  |  | ||||||
|  | 		EHS_LOG_SUCCESS(); | ||||||
|  |  | ||||||
|  | 		return addr; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	UInt_32 ICMP::CalculatePseudoHeaderChecksum(const PseudoICMPv6_Header &header) | ||||||
|  | 	{ | ||||||
|  | 		UInt_32 checksum = 0; | ||||||
|  |  | ||||||
|  | 		for (UInt_8 i = 0; i < 16; ++i) | ||||||
|  | 			checksum += header.src.sin6_addr.s6_addr[i]; | ||||||
|  |  | ||||||
|  | 		for (UInt_8 i = 0; i < 16; ++i) | ||||||
|  | 			checksum += header.dst.sin6_addr.s6_addr[i]; | ||||||
|  |  | ||||||
|  | 		checksum += 58; | ||||||
|  |  | ||||||
|  | 		checksum += htons(header.length); | ||||||
|  |  | ||||||
|  | 		checksum = (checksum >> 16) + (checksum & 0xFFFF); | ||||||
|  | 		checksum += (checksum >> 16); | ||||||
|  |  | ||||||
|  | 		return checksum; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	UInt_16 ICMP::ComputeChecksumV6(UInt_16 *buffer, Size length, const sockaddr_in6 &dst) | ||||||
|  | 	{ | ||||||
|  | 		UInt_32 checksum = 0; | ||||||
|  |  | ||||||
|  | 		if (!src.sin6_addr.s6_addr32[0]) | ||||||
|  | 		{ | ||||||
|  | 			src = RetrieveSrcAddress(); | ||||||
|  | 			if (!src.sin6_addr.s6_addr32[0]) | ||||||
|  | 			{ | ||||||
|  | 				EHS_LOG_INT(LogType::ERR, 0, "Could not retrieve a suitable global address."); | ||||||
|  | 				return checksum; | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		checksum += CalculatePseudoHeaderChecksum({src, dst, (UInt_32)length}); | ||||||
|  |  | ||||||
|  | 		while (length > 1) | ||||||
|  | 		{ | ||||||
|  | 			checksum += *buffer++; | ||||||
|  | 			length -= sizeof(UInt_16); | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		if (length == 1) | ||||||
|  | 			checksum += *(UInt_8 *)buffer; | ||||||
|  |  | ||||||
|  | 		// Carry over any overflow from the 16-bit result | ||||||
|  | 		checksum = (checksum >> 16) + (checksum & 0xFFFF); | ||||||
|  | 		checksum += (checksum >> 16); | ||||||
|  |  | ||||||
|  | 		// Return the 16-bit complement | ||||||
|  | 		return ~checksum; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	UInt_64 ICMP::SendV6(const Str_8 &address, ICMP_Header header, const Byte *data, UInt_64 size) | ||||||
| 	{ | 	{ | ||||||
| 		if (!IsValid()) | 		if (!IsValid()) | ||||||
| 		{ | 		{ | ||||||
| @@ -82,17 +222,17 @@ namespace ehs | |||||||
|  |  | ||||||
| 		payload.SetOffset(payload.GetOffset() + size); | 		payload.SetOffset(payload.GetOffset() + size); | ||||||
|  |  | ||||||
| 		header.checksum = ComputeChecksum((UInt_16 *)&payload[0], payload.Size()); | 		sockaddr_in6 dst = {}; | ||||||
|  | 		dst.sin6_family = AF_INET6; | ||||||
|  | 		inet_pton(AF_INET6, address, &(dst.sin6_addr)); | ||||||
|  |  | ||||||
|  | 		header.checksum = ComputeChecksumV6((UInt_16 *)&payload[0], payload.Size(), dst); | ||||||
|  |  | ||||||
| 		payload.SetOffset(0); | 		payload.SetOffset(0); | ||||||
| 		payload.Write(header); | 		payload.Write(header); | ||||||
| 		payload.SetOffset(payload.Size()); | 		payload.SetOffset(payload.Size()); | ||||||
|  |  | ||||||
| 		sockaddr_in dst_addr = {}; | 		SInt_64 sent = sendto(hdl, payload, payload.Size(), 0, (sockaddr *)&dst, sizeof(sockaddr_in6)); | ||||||
| 		dst_addr.sin_family = AF_INET; |  | ||||||
| 		inet_pton(AF_INET, address, &(dst_addr.sin_addr)); |  | ||||||
|  |  | ||||||
| 		SInt_64 sent = sendto(hdl, payload, payload.Size(), 0, (sockaddr *)&dst_addr, sizeof(dst_addr)); |  | ||||||
| 		if (sent < 0) | 		if (sent < 0) | ||||||
| 		{ | 		{ | ||||||
| 			EHS_LOG_INT(LogType::ERR, 0, "Failed to send packet with error #" + Str_8::FromNum(errno) + "."); | 			EHS_LOG_INT(LogType::ERR, 0, "Failed to send packet with error #" + Str_8::FromNum(errno) + "."); | ||||||
| @@ -105,7 +245,96 @@ namespace ehs | |||||||
| 		return sent; | 		return sent; | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	UInt_64 ICMP::Receive(Str_8 &address, ICMP_Header header, Serializer<UInt_64> &data) | 	UInt_64 ICMP::SendV4(const Str_8 &address, ICMP_Header header, const Byte *data, UInt_64 size) | ||||||
|  | 	{ | ||||||
|  | 		if (!IsValid()) | ||||||
|  | 		{ | ||||||
|  | 			EHS_LOG_INT(LogType::WARN, 0, "Socket is not initialized."); | ||||||
|  |  | ||||||
|  | 			return 0; | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		header.checksum = 0; | ||||||
|  |  | ||||||
|  | 		Serializer<UInt_64> payload(Endianness::LE); | ||||||
|  | 		payload.Write(header); | ||||||
|  | 		payload.Resize(payload.Size() + size); | ||||||
|  |  | ||||||
|  | 		Util::Copy(&payload[payload.GetOffset()], data, size); | ||||||
|  |  | ||||||
|  | 		payload.SetOffset(payload.GetOffset() + size); | ||||||
|  |  | ||||||
|  | 		header.checksum = ComputeChecksumV4((UInt_16 *)&payload[0], payload.Size()); | ||||||
|  |  | ||||||
|  | 		payload.SetOffset(0); | ||||||
|  | 		payload.Write(header); | ||||||
|  | 		payload.SetOffset(payload.Size()); | ||||||
|  |  | ||||||
|  | 		sockaddr_in dst = {}; | ||||||
|  | 		dst.sin_family = AF_INET; | ||||||
|  | 		inet_pton(AF_INET, address, &(dst.sin_addr)); | ||||||
|  |  | ||||||
|  | 		SInt_64 sent = sendto(hdl, payload, payload.Size(), 0, (sockaddr *)&dst, sizeof(sockaddr_in)); | ||||||
|  | 		if (sent < 0) | ||||||
|  | 		{ | ||||||
|  | 			EHS_LOG_INT(LogType::ERR, 0, "Failed to send packet with error #" + Str_8::FromNum(errno) + "."); | ||||||
|  |  | ||||||
|  | 			return 0; | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		EHS_LOG_SUCCESS(); | ||||||
|  |  | ||||||
|  | 		return sent; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	UInt_64 ICMP::ReceiveV6(Str_8 &address, ICMP_Header &header, Serializer<UInt_64> &data) const | ||||||
|  | 	{ | ||||||
|  | 		if (!IsValid()) | ||||||
|  | 		{ | ||||||
|  | 			EHS_LOG_INT(LogType::WARN, 0, "Socket is not initialized."); | ||||||
|  |  | ||||||
|  | 			return 0; | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		Serializer<UInt_64> payload(Endianness::LE); | ||||||
|  | 		payload.Resize(1500); | ||||||
|  |  | ||||||
|  | 		sockaddr_in6 remote = {}; | ||||||
|  | 		socklen_t from_len = sizeof(remote); | ||||||
|  |  | ||||||
|  | 		SInt_64 recv = recvfrom(hdl, payload, 1500, 0, (sockaddr *)&remote, &from_len); | ||||||
|  | 		if (recv < 0) | ||||||
|  | 		{ | ||||||
|  | 			int code = errno; | ||||||
|  | 			if (code == EAGAIN) | ||||||
|  | 				EHS_LOG_SUCCESS(); | ||||||
|  | 			else | ||||||
|  | 				EHS_LOG_INT(LogType::ERR, 0, "Failed to receive packet with error #" + Str_8::FromNum(code) + "."); | ||||||
|  |  | ||||||
|  | 			return 0; | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		payload.Resize(recv); | ||||||
|  |  | ||||||
|  | 		char tmpAddr[INET6_ADDRSTRLEN]; | ||||||
|  |  | ||||||
|  | 		if (!inet_ntop(remote.sin6_family, &remote.sin6_addr, tmpAddr, INET6_ADDRSTRLEN)) | ||||||
|  | 		{ | ||||||
|  | 			EHS_LOG_INT(LogType::ERR, 1, "Failed to convert IPv6 address with error #" + Str_8::FromNum(errno) + "."); | ||||||
|  |  | ||||||
|  | 			return recv; | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		address = tmpAddr; | ||||||
|  | 		header = payload.Read<ICMP_Header>(); | ||||||
|  | 		data = Serializer<UInt_64>(payload.GetEndianness(), &payload[payload.GetOffset()], payload.Size() - payload.GetOffset()); | ||||||
|  |  | ||||||
|  | 		EHS_LOG_SUCCESS(); | ||||||
|  |  | ||||||
|  | 		return recv; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	UInt_64 ICMP::ReceiveV4(Str_8 &address, ICMP_Header &header, Serializer<UInt_64> &data) const | ||||||
| 	{ | 	{ | ||||||
| 		if (!IsValid()) | 		if (!IsValid()) | ||||||
| 		{ | 		{ | ||||||
| @@ -152,25 +381,4 @@ namespace ehs | |||||||
|  |  | ||||||
| 		return recv; | 		return recv; | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	void ICMP::SetReceiveTimeout(UInt_64 timeout) |  | ||||||
| 	{ |  | ||||||
| 		timeval result = {}; |  | ||||||
| 		result.tv_sec = (long)timeout; |  | ||||||
| 		result.tv_usec = 0; |  | ||||||
|  |  | ||||||
| 		if (setsockopt(hdl, SOL_SOCKET, SO_RCVTIMEO, &result, sizeof(result)) < 0) |  | ||||||
| 		{ |  | ||||||
| 			EHS_LOG_INT(LogType::WARN, 0, "Failed to set receive timeout with error #" + Str_8::FromNum(errno) + "."); |  | ||||||
|  |  | ||||||
| 			return; |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		EHS_LOG_SUCCESS(); |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	bool ICMP::IsValid() const |  | ||||||
| 	{ |  | ||||||
| 		return hdl != EHS_INVALID_SOCKET; |  | ||||||
| 	} |  | ||||||
| } | } | ||||||
|   | |||||||
							
								
								
									
										391
									
								
								src/io/socket/ICMP_W32.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										391
									
								
								src/io/socket/ICMP_W32.cpp
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,391 @@ | |||||||
|  | #include "ehs/io/socket/ICMP_W32.h" | ||||||
|  |  | ||||||
|  | struct iphdr | ||||||
|  | { | ||||||
|  | 	u_char  ip_hl:4, ip_v:4; | ||||||
|  | 	u_char  ip_tos; | ||||||
|  | 	u_short ip_len; | ||||||
|  | 	u_short ip_id; | ||||||
|  | 	u_short ip_off; | ||||||
|  | 	u_char  ip_ttl; | ||||||
|  | 	u_char  ip_p; | ||||||
|  | 	u_short ip_sum; | ||||||
|  | 	in_addr ip_src, ip_dst; | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | namespace ehs | ||||||
|  | { | ||||||
|  | 	ICMP::~ICMP() | ||||||
|  |     { | ||||||
|  | 		if (closesocket(hdl) == -1) | ||||||
|  | 			EHS_LOG_INT(LogType::ERR, 0, "Failed to close socket."); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  | 	ICMP::ICMP() | ||||||
|  | 		: hdl(EHS_INVALID_SOCKET) | ||||||
|  | 	{ | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	ICMP::ICMP(const IP version) | ||||||
|  | 		: BaseICMP(version) | ||||||
|  | 	{ | ||||||
|  | 		hdl = socket(AF_INET, SOCK_RAW, IPPROTO_ICMP); | ||||||
|  | 		if (hdl < 0) | ||||||
|  | 		{ | ||||||
|  | 			EHS_LOG_INT(LogType::ERR, 0, "Failed to create ICMP socket with error #" + Str_8::FromNum(errno) + "."); | ||||||
|  | 			return; | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		EHS_LOG_SUCCESS(); | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	ICMP::ICMP(ICMP &&icmp) noexcept | ||||||
|  | 		: BaseICMP((BaseICMP &&)icmp), hdl(icmp.hdl) | ||||||
|  | 	{ | ||||||
|  | 		icmp.hdl = EHS_INVALID_SOCKET; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	ICMP::ICMP(const ICMP &icmp) | ||||||
|  | 		: BaseICMP(icmp), hdl(icmp.hdl) | ||||||
|  | 	{ | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	ICMP & ICMP::operator=(ICMP &&icmp) noexcept | ||||||
|  | 	{ | ||||||
|  | 		if (this == &icmp) | ||||||
|  | 			return *this; | ||||||
|  |  | ||||||
|  | 		BaseICMP::operator=((BaseICMP &&)icmp); | ||||||
|  |  | ||||||
|  | 		hdl = icmp.hdl; | ||||||
|  |  | ||||||
|  | 		icmp.hdl = EHS_INVALID_SOCKET; | ||||||
|  |  | ||||||
|  | 		return *this; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	ICMP & ICMP::operator=(const ICMP &icmp) | ||||||
|  | 	{ | ||||||
|  | 		if (this == &icmp) | ||||||
|  | 			return *this; | ||||||
|  |  | ||||||
|  | 		BaseICMP::operator=(icmp); | ||||||
|  |  | ||||||
|  | 		hdl = icmp.hdl; | ||||||
|  |  | ||||||
|  | 		return *this; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	void ICMP::Release() | ||||||
|  |     { | ||||||
|  | 		if (closesocket(hdl) == -1) | ||||||
|  | 		{ | ||||||
|  | 			EHS_LOG_INT(LogType::ERR, 0, "Failed to close socket."); | ||||||
|  |  | ||||||
|  | 			return; | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  |         hdl = EHS_INVALID_SOCKET; | ||||||
|  |  | ||||||
|  | 		EHS_LOG_SUCCESS(); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  | 	void ICMP::SetReceiveTimeout(UInt_64 timeout) | ||||||
|  | 	{ | ||||||
|  | 		timeval result = {}; | ||||||
|  | 		result.tv_sec = (long)timeout; | ||||||
|  | 		result.tv_usec = 0; | ||||||
|  |  | ||||||
|  | 		if (setsockopt(hdl, SOL_SOCKET, SO_RCVTIMEO, (const char *)&result, sizeof(result)) < 0) | ||||||
|  | 		{ | ||||||
|  | 			EHS_LOG_INT(LogType::WARN, 0, "Failed to set receive timeout with error #" + Str_8::FromNum(errno) + "."); | ||||||
|  |  | ||||||
|  | 			return; | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		EHS_LOG_SUCCESS(); | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	bool ICMP::IsValid() const | ||||||
|  | 	{ | ||||||
|  | 		return hdl != EHS_INVALID_SOCKET; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	bool ICMP::IsLinkLocal(const in6_addr &addr) | ||||||
|  | 	{ | ||||||
|  | 		return addr.s6_addr[0] == 0xfe && (addr.s6_addr[1] & 0xc0) == 0x80; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	sockaddr_in6 ICMP::RetrieveSrcAddress() | ||||||
|  | 	{ | ||||||
|  | 		sockaddr_in6 addr = {}; | ||||||
|  | 		UInt_32 outBufLen = 15000; | ||||||
|  | 		Array<Byte> buffer(outBufLen); | ||||||
|  | 		PIP_ADAPTER_ADDRESSES pAdapterAddresses = (PIP_ADAPTER_ADDRESSES)&buffer[0]; | ||||||
|  |  | ||||||
|  | 		if (GetAdaptersAddresses(AF_INET6, GAA_FLAG_INCLUDE_PREFIX, nullptr, pAdapterAddresses, (PULONG)&outBufLen) == ERROR_BUFFER_OVERFLOW) | ||||||
|  | 		{ | ||||||
|  | 			buffer.Resize(outBufLen); | ||||||
|  | 			pAdapterAddresses = (PIP_ADAPTER_ADDRESSES)&buffer[0]; | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		if (GetAdaptersAddresses(AF_INET6, GAA_FLAG_INCLUDE_PREFIX, nullptr, pAdapterAddresses, (PULONG)&outBufLen) == NO_ERROR) | ||||||
|  | 		{ | ||||||
|  | 			for (PIP_ADAPTER_ADDRESSES adapter = pAdapterAddresses; adapter != nullptr; adapter = adapter->Next) | ||||||
|  | 			{ | ||||||
|  | 				for (PIP_ADAPTER_UNICAST_ADDRESS addrInfo = adapter->FirstUnicastAddress; addrInfo != nullptr; addrInfo = addrInfo->Next) | ||||||
|  | 				{ | ||||||
|  | 					SOCKADDR *sa = addrInfo->Address.lpSockaddr; | ||||||
|  | 					if (sa->sa_family != AF_INET6) | ||||||
|  | 						continue; | ||||||
|  |  | ||||||
|  | 					sockaddr_in6 *ipv6Addr = (sockaddr_in6 *)sa; | ||||||
|  |  | ||||||
|  | 					// Skip link-local addresses | ||||||
|  | 					if (IN6_IS_ADDR_LINKLOCAL(&ipv6Addr->sin6_addr)) | ||||||
|  | 						continue; | ||||||
|  |  | ||||||
|  | 					addr = *ipv6Addr; | ||||||
|  |  | ||||||
|  | 					return addr;  // Return the first suitable address | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		EHS_LOG_SUCCESS(); | ||||||
|  |  | ||||||
|  | 		return addr;  // Return an empty sockaddr_in6 if no valid address was found | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	UInt_32 ICMP::CalculatePseudoHeaderChecksum(const PseudoICMPv6_Header &header) | ||||||
|  | 	{ | ||||||
|  | 		UInt_32 checksum = 0; | ||||||
|  |  | ||||||
|  | 		for (UInt_8 i = 0; i < 16; ++i) | ||||||
|  | 			checksum += header.src.sin6_addr.s6_addr[i]; | ||||||
|  |  | ||||||
|  | 		for (UInt_8 i = 0; i < 16; ++i) | ||||||
|  | 			checksum += header.dst.sin6_addr.s6_addr[i]; | ||||||
|  |  | ||||||
|  | 		checksum += 58; | ||||||
|  |  | ||||||
|  | 		checksum += htons(header.length); | ||||||
|  |  | ||||||
|  | 		checksum = (checksum >> 16) + (checksum & 0xFFFF); | ||||||
|  | 		checksum += (checksum >> 16); | ||||||
|  |  | ||||||
|  | 		return checksum; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	UInt_16 ICMP::ComputeChecksumV6(UInt_16 *buffer, Size length, const sockaddr_in6 &dst) | ||||||
|  | 	{ | ||||||
|  | 		UInt_32 checksum = 0; | ||||||
|  |  | ||||||
|  | 		if (!src.sin6_addr.u.Word[0]) | ||||||
|  | 		{ | ||||||
|  | 			src = RetrieveSrcAddress(); | ||||||
|  | 			if (!src.sin6_addr.u.Word[0]) | ||||||
|  | 			{ | ||||||
|  | 				EHS_LOG_INT(LogType::ERR, 0, "Could not retrieve a suitable global address."); | ||||||
|  | 				return checksum; | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		checksum += CalculatePseudoHeaderChecksum({src, dst, (UInt_32)length}); | ||||||
|  |  | ||||||
|  | 		while (length > 1) | ||||||
|  | 		{ | ||||||
|  | 			checksum += *buffer++; | ||||||
|  | 			length -= sizeof(UInt_16); | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		if (length == 1) | ||||||
|  | 			checksum += *(UInt_8 *)buffer; | ||||||
|  |  | ||||||
|  | 		// Carry over any overflow from the 16-bit result | ||||||
|  | 		checksum = (checksum >> 16) + (checksum & 0xFFFF); | ||||||
|  | 		checksum += (checksum >> 16); | ||||||
|  |  | ||||||
|  | 		// Return the 16-bit complement | ||||||
|  | 		return ~checksum; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	UInt_64 ICMP::SendV6(const Str_8 &address, ICMP_Header header, const Byte *data, UInt_64 size) | ||||||
|  | 	{ | ||||||
|  | 		if (!IsValid()) | ||||||
|  | 		{ | ||||||
|  | 			EHS_LOG_INT(LogType::WARN, 0, "Socket is not initialized."); | ||||||
|  |  | ||||||
|  | 			return 0; | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		header.checksum = 0; | ||||||
|  |  | ||||||
|  | 		Serializer<UInt_64> payload(Endianness::LE); | ||||||
|  | 		payload.Write(header); | ||||||
|  | 		payload.Resize(payload.Size() + size); | ||||||
|  |  | ||||||
|  | 		Util::Copy(&payload[payload.GetOffset()], data, size); | ||||||
|  |  | ||||||
|  | 		payload.SetOffset(payload.GetOffset() + size); | ||||||
|  |  | ||||||
|  | 		sockaddr_in6 dst = {}; | ||||||
|  | 		dst.sin6_family = AF_INET6; | ||||||
|  | 		inet_pton(AF_INET6, address, &(dst.sin6_addr)); | ||||||
|  |  | ||||||
|  | 		header.checksum = ComputeChecksumV6((UInt_16 *)&payload[0], payload.Size(), dst); | ||||||
|  |  | ||||||
|  | 		payload.SetOffset(0); | ||||||
|  | 		payload.Write(header); | ||||||
|  | 		payload.SetOffset(payload.Size()); | ||||||
|  |  | ||||||
|  | 		SInt_64 sent = sendto(hdl, (const char *)&payload[0], payload.Size(), 0, (sockaddr *)&dst, sizeof(sockaddr_in6)); | ||||||
|  | 		if (sent < 0) | ||||||
|  | 		{ | ||||||
|  | 			EHS_LOG_INT(LogType::ERR, 0, "Failed to send packet with error #" + Str_8::FromNum(errno) + "."); | ||||||
|  |  | ||||||
|  | 			return 0; | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		EHS_LOG_SUCCESS(); | ||||||
|  |  | ||||||
|  | 		return sent; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	UInt_64 ICMP::SendV4(const Str_8 &address, ICMP_Header header, const Byte *data, UInt_64 size) | ||||||
|  | 	{ | ||||||
|  | 		if (!IsValid()) | ||||||
|  | 		{ | ||||||
|  | 			EHS_LOG_INT(LogType::WARN, 0, "Socket is not initialized."); | ||||||
|  |  | ||||||
|  | 			return 0; | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		header.checksum = 0; | ||||||
|  |  | ||||||
|  | 		Serializer<UInt_64> payload(Endianness::LE); | ||||||
|  | 		payload.Write(header); | ||||||
|  | 		payload.Resize(payload.Size() + size); | ||||||
|  |  | ||||||
|  | 		Util::Copy(&payload[payload.GetOffset()], data, size); | ||||||
|  |  | ||||||
|  | 		payload.SetOffset(payload.GetOffset() + size); | ||||||
|  |  | ||||||
|  | 		header.checksum = ComputeChecksumV4((UInt_16 *)&payload[0], payload.Size()); | ||||||
|  |  | ||||||
|  | 		payload.SetOffset(0); | ||||||
|  | 		payload.Write(header); | ||||||
|  | 		payload.SetOffset(payload.Size()); | ||||||
|  |  | ||||||
|  | 		sockaddr_in dst_addr = {}; | ||||||
|  | 		dst_addr.sin_family = AF_INET; | ||||||
|  | 		inet_pton(AF_INET, address, &(dst_addr.sin_addr)); | ||||||
|  |  | ||||||
|  | 		SInt_64 sent = sendto(hdl, (const char *)&payload[0], payload.Size(), 0, (sockaddr *)&dst_addr, sizeof(dst_addr)); | ||||||
|  | 		if (sent < 0) | ||||||
|  | 		{ | ||||||
|  | 			EHS_LOG_INT(LogType::ERR, 0, "Failed to send packet with error #" + Str_8::FromNum(errno) + "."); | ||||||
|  |  | ||||||
|  | 			return 0; | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		EHS_LOG_SUCCESS(); | ||||||
|  |  | ||||||
|  | 		return sent; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	UInt_64 ICMP::ReceiveV6(Str_8 &address, ICMP_Header &header, Serializer<UInt_64> &data) const | ||||||
|  | 	{ | ||||||
|  | 		if (!IsValid()) | ||||||
|  | 		{ | ||||||
|  | 			EHS_LOG_INT(LogType::WARN, 0, "Socket is not initialized."); | ||||||
|  |  | ||||||
|  | 			return 0; | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		Serializer<UInt_64> payload(Endianness::LE); | ||||||
|  | 		payload.Resize(1500); | ||||||
|  |  | ||||||
|  | 		sockaddr_in6 remote = {}; | ||||||
|  | 		socklen_t from_len = sizeof(remote); | ||||||
|  |  | ||||||
|  | 		SInt_64 recv = recvfrom(hdl, (char *)&payload[0], 1500, 0, (sockaddr *)&remote, &from_len); | ||||||
|  | 		if (recv < 0) | ||||||
|  | 		{ | ||||||
|  | 			int code = errno; | ||||||
|  | 			if (code == EAGAIN) | ||||||
|  | 				EHS_LOG_SUCCESS(); | ||||||
|  | 			else | ||||||
|  | 				EHS_LOG_INT(LogType::ERR, 0, "Failed to receive packet with error #" + Str_8::FromNum(code) + "."); | ||||||
|  |  | ||||||
|  | 			return 0; | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		payload.Resize(recv); | ||||||
|  |  | ||||||
|  | 		char tmpAddr[INET6_ADDRSTRLEN]; | ||||||
|  |  | ||||||
|  | 		if (!inet_ntop(remote.sin6_family, &remote.sin6_addr, tmpAddr, INET6_ADDRSTRLEN)) | ||||||
|  | 		{ | ||||||
|  | 			EHS_LOG_INT(LogType::ERR, 1, "Failed to convert IPv6 address with error #" + Str_8::FromNum(errno) + "."); | ||||||
|  |  | ||||||
|  | 			return recv; | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		address = tmpAddr; | ||||||
|  | 		header = payload.Read<ICMP_Header>(); | ||||||
|  | 		data = Serializer<UInt_64>(payload.GetEndianness(), &payload[payload.GetOffset()], payload.Size() - payload.GetOffset()); | ||||||
|  |  | ||||||
|  | 		EHS_LOG_SUCCESS(); | ||||||
|  |  | ||||||
|  | 		return recv; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	UInt_64 ICMP::ReceiveV4(Str_8 &address, ICMP_Header &header, Serializer<UInt_64> &data) const | ||||||
|  | 	{ | ||||||
|  | 		if (!IsValid()) | ||||||
|  | 		{ | ||||||
|  | 			EHS_LOG_INT(LogType::WARN, 0, "Socket is not initialized."); | ||||||
|  |  | ||||||
|  | 			return 0; | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		Serializer<UInt_64> payload(Endianness::LE); | ||||||
|  | 		payload.Resize(1500); | ||||||
|  |  | ||||||
|  | 		sockaddr_in remote = {}; | ||||||
|  | 		socklen_t from_len = sizeof(remote); | ||||||
|  |  | ||||||
|  | 		SInt_64 recv = recvfrom(hdl, (char *)&payload[0], 1500, 0, (sockaddr *)&remote, &from_len); | ||||||
|  | 		if (recv < 0) | ||||||
|  | 		{ | ||||||
|  | 			int code = errno; | ||||||
|  | 			if (code == EAGAIN) | ||||||
|  | 				EHS_LOG_SUCCESS(); | ||||||
|  | 			else | ||||||
|  | 				EHS_LOG_INT(LogType::ERR, 0, "Failed to receive packet with error #" + Str_8::FromNum(code) + "."); | ||||||
|  |  | ||||||
|  | 			return 0; | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		payload.Resize(recv); | ||||||
|  |  | ||||||
|  | 		char tmpAddr[INET_ADDRSTRLEN]; | ||||||
|  |  | ||||||
|  | 		if (!inet_ntop(remote.sin_family, &remote.sin_addr, tmpAddr, INET_ADDRSTRLEN)) | ||||||
|  | 		{ | ||||||
|  | 			EHS_LOG_INT(LogType::ERR, 1, "Failed to convert IPv4 address with error #" + Str_8::FromNum(errno) + "."); | ||||||
|  |  | ||||||
|  | 			return recv; | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		address = tmpAddr; | ||||||
|  | 		iphdr ipHeader = payload.Read<iphdr>(); | ||||||
|  | 		header = payload.Read<ICMP_Header>(); | ||||||
|  | 		data = Serializer<UInt_64>(payload.GetEndianness(), &payload[payload.GetOffset()], payload.Size() - payload.GetOffset()); | ||||||
|  |  | ||||||
|  | 		EHS_LOG_SUCCESS(); | ||||||
|  |  | ||||||
|  | 		return recv; | ||||||
|  | 	} | ||||||
|  | } | ||||||
| @@ -43,4 +43,9 @@ namespace ehs | |||||||
| 	void CPU::GetBrand(Char_8* input) | 	void CPU::GetBrand(Char_8* input) | ||||||
| 	{ | 	{ | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
|  |     UInt_8 CPU::GetCacheLineSize() | ||||||
|  |     { | ||||||
|  |       return 0; | ||||||
|  |     } | ||||||
| } | } | ||||||
		Reference in New Issue
	
	Block a user