Init commit with a ring buffer
This commit is contained in:
		
						commit
						0afef34a63
					
				
							
								
								
									
										7
									
								
								.gitignore
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										7
									
								
								.gitignore
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @ -0,0 +1,7 @@ | ||||
| # binaries | ||||
| test/bin | ||||
| 
 | ||||
| #eclipse based files | ||||
| .cproject | ||||
| .project | ||||
| .settings/ | ||||
							
								
								
									
										48
									
								
								include/core/core.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										48
									
								
								include/core/core.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,48 @@ | ||||
| /*!
 | ||||
|  * \file core/core.h | ||||
|  * | ||||
|  * \copyright Copyright (C) 2020-2021 Christos Choutouridis <christos@choutouridis.net> | ||||
|  * | ||||
|  * <dl class=\"section copyright\"><dt>License</dt><dd> | ||||
|  * The MIT License (MIT) | ||||
|  * | ||||
|  * Permission is hereby granted, free of charge, to any person obtaining a copy | ||||
|  * of this software and associated documentation files (the "Software"), to deal | ||||
|  * in the Software without restriction, including without limitation the rights | ||||
|  * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell | ||||
|  * copies of the Software, and to permit persons to whom the Software is | ||||
|  * furnished to do so, subject to the following conditions: | ||||
|  * | ||||
|  * The above copyright notice and this permission notice shall be included in all | ||||
|  * copies or substantial portions of the Software. | ||||
|  * | ||||
|  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR | ||||
|  * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, | ||||
|  * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE | ||||
|  * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER | ||||
|  * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, | ||||
|  * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE | ||||
|  * SOFTWARE. | ||||
|  * </dd></dl> | ||||
|  */ | ||||
| 
 | ||||
| #ifndef TBX_CORE_CORE_H_ | ||||
| #define TBX_CORE_CORE_H_ | ||||
| 
 | ||||
| //#define __cplusplus 201703L // for developing puproses only
 | ||||
| 
 | ||||
| #include <cstddef> | ||||
| #include <cstdint> | ||||
| 
 | ||||
| namespace tbx { | ||||
| 
 | ||||
|     //! \name size and index
 | ||||
|     //! @{
 | ||||
|     using size_t   = std::size_t; | ||||
|     using index_t  = size_t;      //!< index_t and size_t mend to be interchangeable
 | ||||
| 
 | ||||
|     using ptrdiff_t= std::ptrdiff_t; | ||||
|     //! @}
 | ||||
| } | ||||
| 
 | ||||
| #endif /* TBX_CORE_CORE_H_ */ | ||||
							
								
								
									
										55
									
								
								include/core/crtp.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										55
									
								
								include/core/crtp.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,55 @@ | ||||
| /*!
 | ||||
|  * \file core/crtp.h | ||||
|  * \brief | ||||
|  *      A CRTP idiom helper header | ||||
|  * | ||||
|  * \copyright Copyright (C) 2020 Christos Choutouridis <christos@choutouridis.net> | ||||
|  * | ||||
|  * <dl class=\"section copyright\"><dt>License</dt><dd> | ||||
|  * The MIT License (MIT) | ||||
|  * | ||||
|  * Permission is hereby granted, free of charge, to any person obtaining a copy | ||||
|  * of this software and associated documentation files (the "Software"), to deal | ||||
|  * in the Software without restriction, including without limitation the rights | ||||
|  * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell | ||||
|  * copies of the Software, and to permit persons to whom the Software is | ||||
|  * furnished to do so, subject to the following conditions: | ||||
|  * | ||||
|  * The above copyright notice and this permission notice shall be included in all | ||||
|  * copies or substantial portions of the Software. | ||||
|  * | ||||
|  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR | ||||
|  * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, | ||||
|  * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE | ||||
|  * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER | ||||
|  * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, | ||||
|  * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE | ||||
|  * SOFTWARE. | ||||
|  * </dd></dl> | ||||
|  */ | ||||
| 
 | ||||
| #ifndef TBX_CORE_CRTP_H_ | ||||
| #define TBX_CORE_CRTP_H_ | ||||
| 
 | ||||
| /*!
 | ||||
|  * \defgroup crtp   CRTP idiom support header | ||||
|  * | ||||
|  */ | ||||
| //!@{
 | ||||
| namespace tbx { | ||||
|    //! CRTP support tag type
 | ||||
|    struct crtp_tag { }; | ||||
|    //! virtual support tag type
 | ||||
|    struct virtual_tag { }; | ||||
| 
 | ||||
|    //! \def CRTP boilerplate lines
 | ||||
|    #define _CRTP_IMPL(T) \ | ||||
|    constexpr T& impl() { return *static_cast<T*>(this); } \ | ||||
|    constexpr const T& impl() const { return *static_cast<const T*>(this); } | ||||
| 
 | ||||
| } | ||||
| //!@}
 | ||||
| 
 | ||||
| 
 | ||||
| 
 | ||||
| #endif /* TBX_CORE_CRTP_H_ */ | ||||
							
								
								
									
										227
									
								
								include/core/ring_iterator.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										227
									
								
								include/core/ring_iterator.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,227 @@ | ||||
| /*!
 | ||||
|  * \file cont/ring_iterator.h | ||||
|  * \brief | ||||
|  *      A ring/circular iterator. | ||||
|  * | ||||
|  * \copyright Copyright (C) 2021 Christos Choutouridis <christos@choutouridis.net> | ||||
|  * | ||||
|  * <dl class=\"section copyright\"><dt>License</dt><dd> | ||||
|  * The MIT License (MIT) | ||||
|  * | ||||
|  * Permission is hereby granted, free of charge, to any person obtaining a copy | ||||
|  * of this software and associated documentation files (the "Software"), to deal | ||||
|  * in the Software without restriction, including without limitation the rights | ||||
|  * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell | ||||
|  * copies of the Software, and to permit persons to whom the Software is | ||||
|  * furnished to do so, subject to the following conditions: | ||||
|  * | ||||
|  * The above copyright notice and this permission notice shall be included in all | ||||
|  * copies or substantial portions of the Software. | ||||
|  * | ||||
|  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR | ||||
|  * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, | ||||
|  * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE | ||||
|  * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER | ||||
|  * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, | ||||
|  * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE | ||||
|  * SOFTWARE. | ||||
|  * </dd></dl> | ||||
|  */ | ||||
| 
 | ||||
| #ifndef TBX_CORE_RING_ITERATOR_H_ | ||||
| #define TBX_CORE_RING_ITERATOR_H_ | ||||
| 
 | ||||
| #include <core/core.h> | ||||
| 
 | ||||
| #include <iterator> | ||||
| #include <type_traits> | ||||
| 
 | ||||
| namespace tbx { | ||||
| 
 | ||||
| template<typename Iter_t, size_t N> | ||||
| class ring_iterator { | ||||
|     //! \name STL iterator traits "forwarding"
 | ||||
|     //! @{
 | ||||
|     protected: | ||||
|         using traits_type = std::iterator_traits<Iter_t>; | ||||
| 
 | ||||
|     public: | ||||
|         using iterator_type     = Iter_t; | ||||
|         using iterator_category = typename traits_type::iterator_category; | ||||
|         using value_type        = typename traits_type::value_type; | ||||
|         using difference_type   = typename traits_type::difference_type; | ||||
|         using reference         = typename traits_type::reference; | ||||
|         using pointer           = typename traits_type::pointer; | ||||
|     //! @}
 | ||||
| 
 | ||||
| 
 | ||||
|     //! \name Constructor / Destructor
 | ||||
|     //! @{
 | ||||
|     public: | ||||
|         ring_iterator(const Iter_t base =nullptr) noexcept : | ||||
|             base_(base), iter_(base) { } | ||||
| 
 | ||||
|         ring_iterator(const Iter_t base, size_t elem) noexcept : | ||||
|             base_(base), iter_(base + elem) { } | ||||
| 
 | ||||
|         ring_iterator(const ring_iterator& it) noexcept : | ||||
|             base_(it.base_), iter_(it.iter_) { } | ||||
| 
 | ||||
|         ring_iterator& operator= (const ring_iterator& it) noexcept { | ||||
|             base_ = it.base_; | ||||
|             iter_ = it.iter_; | ||||
|             return *this; | ||||
|         } | ||||
|     //! @}
 | ||||
| 
 | ||||
|     //! \name Forward iterator requirements
 | ||||
|     //! @{
 | ||||
|     public: | ||||
|         reference operator*() const noexcept { | ||||
|             return *iter_; | ||||
|         } | ||||
|         pointer operator->() const noexcept { | ||||
|             return iter_; | ||||
|         } | ||||
|         ring_iterator& operator++() noexcept { | ||||
|             if (static_cast<size_t>(++iter_ - base_) >= N) | ||||
|                 iter_ = base_; | ||||
|             return *this; | ||||
|         } | ||||
|         ring_iterator operator++(int) noexcept { | ||||
|             ring_iterator it = *this; | ||||
|             if (static_cast<size_t>(++iter_ - base_) >= N) | ||||
|                 iter_ = base_; | ||||
|             return it; | ||||
|         } | ||||
|     //! @}
 | ||||
| 
 | ||||
|     //! \name Bidirectional iterator requirements
 | ||||
|     //! @{
 | ||||
|     public: | ||||
|         ring_iterator& operator--() noexcept { | ||||
|             if (--iter_ < base_) | ||||
|                 iter_ = base_ + N -1; | ||||
|             return *this; | ||||
|         } | ||||
|         ring_iterator operator--(int) noexcept { | ||||
|             ring_iterator it = *this; | ||||
|             if (--iter_ < base_) | ||||
|                 iter_ = base_ + N -1; | ||||
|             return it; | ||||
|         } | ||||
|     //! @}
 | ||||
| 
 | ||||
|     //! \name Random access iterator requirements
 | ||||
|     //! @{
 | ||||
|         reference operator[](difference_type n) const noexcept { | ||||
|             difference_type k = iter_ - base_;          // ptrdiff from base_
 | ||||
|             return (static_cast<size_t>(k + n) < N) ? | ||||
|                 base_[k + n] :      // on range
 | ||||
|                 base_[k + n - N];   // out of range, loop
 | ||||
|         } | ||||
|         ring_iterator& operator+=(difference_type n) noexcept { | ||||
|             difference_type k = iter_ - base_;          // ptrdiff from base_
 | ||||
|             iter_ += (static_cast<size_t>(k + n) < N) ? | ||||
|                 n :                 // on range
 | ||||
|                 n - N;              // out of range, loop
 | ||||
|             return *this; | ||||
|         } | ||||
|         ring_iterator operator+(difference_type n) const noexcept { | ||||
|             difference_type k = iter_ - base_;          // ptrdiff from base_
 | ||||
|             return (static_cast<size_t>(k + n) < N) ? | ||||
|                 ring_iterator(base_, k + n) :           // on range
 | ||||
|                 ring_iterator(base_, k + n - N);        // out of range, loop
 | ||||
|         } | ||||
|         ring_iterator& operator-=(difference_type n) noexcept { | ||||
|             difference_type k = iter_ - base_;          // ptrdiff from base_
 | ||||
|             iter_ -= ((k - n) < 0)? | ||||
|                 n - N:                                  // out of range, loop
 | ||||
|                 n;                                      // on range
 | ||||
|             return *this; | ||||
|         } | ||||
|         ring_iterator operator-(difference_type n) const noexcept { | ||||
|             difference_type k = iter_ - base_;          // ptrdiff from base_
 | ||||
|             return ((k - n) < 0) ? | ||||
|                 ring_iterator(base_, k - n + N) :       // out of range, loop
 | ||||
|                 ring_iterator(base_, k - n);            // on range
 | ||||
|         } | ||||
|     //! @}
 | ||||
| 
 | ||||
|     //! \name Data members and access
 | ||||
|     //! @{
 | ||||
|         const Iter_t& base() const noexcept { | ||||
|             return base_; | ||||
|         } | ||||
|         const Iter_t& iter() const noexcept { | ||||
|             return iter_; | ||||
|         } | ||||
|         size_t size() noexcept { | ||||
|             return N; | ||||
|         } | ||||
| 
 | ||||
|     protected: | ||||
|         Iter_t base_; | ||||
|         Iter_t iter_; | ||||
|     //! @}
 | ||||
| }; | ||||
| 
 | ||||
| // Forward iterator requirements
 | ||||
| template<typename Iter_L, typename Iter_R, size_t N> | ||||
| inline bool operator==(const ring_iterator<Iter_L, N>& lhs, const ring_iterator<Iter_R, N>& rhs) | ||||
| noexcept { | ||||
|     return lhs.iter() == rhs.iter(); | ||||
| } | ||||
| 
 | ||||
| template<typename Iter_L, typename Iter_R, size_t N> | ||||
| inline bool operator!=(const ring_iterator<Iter_L, N>& lhs, const ring_iterator<Iter_R, N>& rhs) | ||||
| noexcept { | ||||
|     return lhs.iter() != rhs.iter(); | ||||
| } | ||||
| 
 | ||||
| // Random access iterator requirements
 | ||||
| template<typename Iter_L, typename Iter_R, size_t N> | ||||
| inline bool operator<(const ring_iterator<Iter_L, N>& lhs, const ring_iterator<Iter_R, N>& rhs) | ||||
| noexcept { | ||||
|     return lhs.iter() < rhs.iter(); | ||||
| } | ||||
| 
 | ||||
| template<typename Iter_L, typename Iter_R, size_t N> | ||||
| inline bool operator<=(const ring_iterator<Iter_L, N>& lhs, const ring_iterator<Iter_R, N>& rhs) | ||||
| noexcept { | ||||
|     return lhs.iter() <= rhs.iter(); | ||||
| } | ||||
| 
 | ||||
| template<typename Iter_L, typename Iter_R, size_t N> | ||||
| inline bool operator>(const ring_iterator<Iter_L, N>& lhs, const ring_iterator<Iter_R, N>& rhs) | ||||
| noexcept { | ||||
|     return lhs.iter() > rhs.iter(); | ||||
| } | ||||
| 
 | ||||
| template<typename Iter_L, typename Iter_R, size_t N> | ||||
| inline bool operator>=(const ring_iterator<Iter_L, N>& lhs, const ring_iterator<Iter_R, N>& rhs) | ||||
| noexcept { | ||||
|     return lhs.iter() >= rhs.iter(); | ||||
| } | ||||
| 
 | ||||
| template<typename Iter_L, typename Iter_R, size_t N> | ||||
| inline auto operator-(const ring_iterator<Iter_L, N>& lhs, const ring_iterator<Iter_R, N>& rhs) | ||||
| noexcept | ||||
| -> decltype(lhs.iter() - rhs.iter()) { | ||||
|     auto diff = lhs.iter() - rhs.iter(); | ||||
|     return diff < 0 ? | ||||
|             diff + N :  // loop
 | ||||
|             diff;       // no loop
 | ||||
| } | ||||
| 
 | ||||
| template<typename Iter, size_t N> | ||||
| inline ring_iterator<Iter, N> operator+(std::ptrdiff_t lhs, const ring_iterator<Iter, N>& rhs) | ||||
| noexcept { | ||||
|     ring_iterator<Iter, N> it(rhs.iter()); | ||||
|     return it += lhs; | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| } //namespace tbx;
 | ||||
| 
 | ||||
| #endif /* TBX_CORE_RING_ITERATOR_H_ */ | ||||
							
								
								
									
										243
									
								
								test/Makefile
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										243
									
								
								test/Makefile
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,243 @@ | ||||
| #
 | ||||
| # Makefile for tbx unit testing
 | ||||
| #
 | ||||
| # Copyright (C) 2021 Christos Choutouridis <christos@choutouridis.net>
 | ||||
| #
 | ||||
| # The MIT License (MIT)
 | ||||
| #
 | ||||
| # Permission is hereby granted, free of charge, to any person obtaining a copy
 | ||||
| # of this software and associated documentation files (the "Software"), to deal
 | ||||
| # in the Software without restriction, including without limitation the rights
 | ||||
| # to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
 | ||||
| # copies of the Software, and to permit persons to whom the Software is
 | ||||
| # furnished to do so, subject to the following conditions:
 | ||||
| #
 | ||||
| # The above copyright notice and this permission notice shall be included in all
 | ||||
| # copies or substantial portions of the Software.
 | ||||
| #
 | ||||
| # THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
 | ||||
| # IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
 | ||||
| # FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
 | ||||
| # AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
 | ||||
| # LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
 | ||||
| # OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
 | ||||
| # SOFTWARE.
 | ||||
| #
 | ||||
| 
 | ||||
| #  notes:
 | ||||
| # ==============
 | ||||
| # This makefile provides support for unix-like local builds and
 | ||||
| # docker based builds for gcc and clang.
 | ||||
| # ** msvc is not currently supported.
 | ||||
| #
 | ||||
| # Use cases:
 | ||||
| # 1) build localy using **local rules** in terminal.
 | ||||
| #    example: 
 | ||||
| #       make -j4 MK_ARG="-std=c++14 -fconcepts" build-gcc
 | ||||
| #
 | ||||
| # 2) build localy using **docker based rules** via terminal or script
 | ||||
| #    in order to compile for all compilers/versions/dialects.
 | ||||
| #    example in bash:
 | ||||
| #       #!/bin/bash
 | ||||
| #       for im in gcc:8 gcc:9 gcc:10; do
 | ||||
| #          for dial in -std=c++14 -std=c++17; do
 | ||||
| #             make IMAGE="$im" MK_ARG="$dial" dock-gcc
 | ||||
| #          done
 | ||||
| #       done
 | ||||
| #
 | ||||
| # 3) build inside a docker instance using **local rules** in order to
 | ||||
| #    build and test using a CD/CI system.
 | ||||
| #    example in a yml file:
 | ||||
| #       image: gcc:8
 | ||||
| #       build:
 | ||||
| #          stage: build
 | ||||
| #          - make MK_ARG="-std=c++14" build-gcc
 | ||||
| #          - make MK_ARG="-std=c++14 -fconcepts" build-gcc
 | ||||
| #          - make MK_ARG="-std=c++17" build-gcc
 | ||||
| #          - make MK_ARG="-std=c++17 -fconcepts" build-gcc
 | ||||
| #          - make MK_ARG="-std=c++2a" build-gcc
 | ||||
| #          ... etc
 | ||||
| #       test:
 | ||||
| #          stage: test
 | ||||
| #          - bin/utlTest
 | ||||
| #          ... etc
 | ||||
| 
 | ||||
| #
 | ||||
| 
 | ||||
| #
 | ||||
| # ============== Project settings ==============
 | ||||
| #
 | ||||
| # note: STM32f101RC/D/E device configuration on SMT32CubeF1 library requires:
 | ||||
| #   1) STM2F101xE pre-define
 | ||||
| #   2) startup_stm32f101xe.s startup file
 | ||||
| #
 | ||||
| 
 | ||||
| # Project's name
 | ||||
| PROJECT         := tbx | ||||
| # Excecutable's name
 | ||||
| TARGET          := tbxTest | ||||
| 
 | ||||
| # Source directories list(space seperated). Relative path, under current directory only
 | ||||
| SRC_DIR_LIST    := tests gtest | ||||
| 
 | ||||
| # Source files list(space seperated).
 | ||||
| SRC_FILES_LIST  :=  | ||||
| 
 | ||||
| # Include directories list(space seperated). Relative path
 | ||||
| INC_DIR_LIST    := ../include gtest | ||||
|                     | ||||
| # Exclude files list(space seperated). Filenames only.
 | ||||
| # EXC_FILE_LIST := bad.cpp old.cpp
 | ||||
| 
 | ||||
| # Build directories
 | ||||
| BUILD_DIR       := bin | ||||
| OBJ_DIR         := $(BUILD_DIR)/obj | ||||
| DEP_DIR         := $(BUILD_DIR)/.dep | ||||
| 
 | ||||
| 
 | ||||
| # ============== Compiler settings ==============
 | ||||
| CLANGXX         := clang++ | ||||
| GCCXX           := g++ | ||||
| CSIZE           := size | ||||
| 
 | ||||
| ODUMP           := objdump | ||||
| OCOPY           := objcopy | ||||
| 
 | ||||
| # Compiler flags for debug and release
 | ||||
| DEB_CFLAGS      := -std=c++17 -DDEBUG -g3 -Wall -Wextra | ||||
| REL_CFLAGS      := -std=c++17 -Wall -Wextra -O2 | ||||
| # Pre-defines
 | ||||
| # PRE_DEFS := MYCAB=1729 SUPER_MODE
 | ||||
| 
 | ||||
| # ============== Linker settings ==============
 | ||||
| # Linker flags
 | ||||
| LDFLAGS         := -pthread | ||||
| # Map output file
 | ||||
| MAP_FILE        := output.map | ||||
| MAP_FLAG        := -Xlinker -Map=$(BUILD_DIR)/$(MAP_FILE) | ||||
| 
 | ||||
| # ============== Docker settings ==============
 | ||||
| # We need:
 | ||||
| #  1) Bind the entire project directory(the dir that icludes all the code) as volume.
 | ||||
| #  2) In docker instance change to working directory(where the makefile is). 
 | ||||
| # For utl we use the directories `${PWD%/*}` and `test`.
 | ||||
| # We double-$$ for double evaluating.
 | ||||
| DOCKER_VOL_DIR  := "$${PWD%/*}" | ||||
| DOCKER_WRK_DIR  := test | ||||
| DOCKER_RUN      := docker run --rm -v $(DOCKER_VOL_DIR):/usr/src/$(PROJECT) -w /usr/src/$(PROJECT)/$(DOCKER_WRK_DIR) | ||||
| 
 | ||||
| # ============== Default settings ==============
 | ||||
| # compiler and compiler flags. By default docker is not used.
 | ||||
| CFLAGS          := $(DEB_CFLAGS) | ||||
| CXX             := $(GCCXX) | ||||
| DOCKER          :=  | ||||
| 
 | ||||
| #
 | ||||
| # =========== Main body and Patterns ===========
 | ||||
| #
 | ||||
| ifeq ($(OS), Windows_NT) | ||||
| 	TARGET := $(TARGET).exe | ||||
| endif | ||||
| INC     := $(foreach dir,$(INC_DIR_LIST),-I$(dir)) | ||||
| DEF     := $(foreach def,$(PRE_DEFS),-D$(def)) | ||||
| EXC     := $(foreach fil,$(EXC_FILE_LIST),                              \
 | ||||
|                $(foreach dir,$(SRC_DIR_LIST),$(wildcard $(dir)/$(fil))) \
 | ||||
|            ) | ||||
| # source files, object and dependencies list
 | ||||
| # recursive search into current and source directories
 | ||||
| SRC     := $(wildcard *.cpp) | ||||
| SRC     += $(foreach dir,$(SRC_DIR_LIST),$(wildcard $(dir)/*.cpp)) | ||||
| SRC     += $(foreach dir,$(SRC_DIR_LIST),$(wildcard $(dir)/**/*.cpp)) | ||||
| #SRC     := $(abspath $(filter-out $(EXC),$(SRC)))
 | ||||
| SRC     := $(filter-out $(EXC),$(SRC)) | ||||
| 
 | ||||
| OBJ     := $(foreach file,$(SRC:%.cpp=%.o),$(OBJ_DIR)/$(file)) | ||||
| DEP     := $(foreach file,$(SRC:%.cpp=%.d),$(DEP_DIR)/$(file)) | ||||
| 
 | ||||
| 
 | ||||
| # Make Dependencies pattern.
 | ||||
| # This little trick enables recompilation only when dependencies change
 | ||||
| # and it does so for changes both in source AND header files ;)
 | ||||
| # 
 | ||||
| # It is based on Tom Tromey's method.
 | ||||
| # 
 | ||||
| # Invoke cpp to create makefile rules with dependencies for each source file
 | ||||
| $(DEP_DIR)/%.d: %.cpp | ||||
| 	@mkdir -p $(@D) | ||||
| 	@$(DOCKER) $(CXX) -E $(CFLAGS) $(INC) $(DEF) -MM -MT $(OBJ_DIR)/$(<:.cpp=.o) -MF $@ $< | ||||
| 
 | ||||
| # objects depent on .cpp AND dependency files, which have an empty recipe 
 | ||||
| $(OBJ_DIR)/%.o: %.cpp $(DEP_DIR)/%.d | ||||
| 	@mkdir -p $(@D) | ||||
| 	$(DOCKER) $(CXX) -c $(CFLAGS) $(INC) $(DEF) -o $@ $< | ||||
| 
 | ||||
| # empty recipe for dependency files. This prevents make errors
 | ||||
| $(DEP): | ||||
| 
 | ||||
| # now include all dependencies
 | ||||
| # After all they are makefile dependency rules ;)
 | ||||
| include $(wildcard $(DEP)) | ||||
| 
 | ||||
| # main target rule
 | ||||
| $(BUILD_DIR)/$(TARGET): $(OBJ) | ||||
| 	@mkdir -p $(@D) | ||||
| 	@echo Linking to target: $(TARGET) | ||||
| 	$(DOCKER) $(CXX) $(LDFLAGS) $(MAP_FLAG) -o $(@D)/$(TARGET) $(OBJ) | ||||
| 	$(DOCKER) $(ODUMP) -h -S $(BUILD_DIR)/$(TARGET) > $(BUILD_DIR)/$(basename $(TARGET)).list  | ||||
| 	$(DOCKER) $(OCOPY) -O ihex $(BUILD_DIR)/$(TARGET) $(BUILD_DIR)/$(basename $(TARGET)).hex | ||||
| 	@echo | ||||
| 	@echo Print size information | ||||
| 	@$(CSIZE) $(@D)/$(TARGET) | ||||
| 	@echo Done | ||||
| 
 | ||||
| .PHONY: clean | ||||
| clean: | ||||
| 	@echo Cleaning build directories | ||||
| 	@rm -rf $(OBJ_DIR) | ||||
| 	@rm -rf $(DEP_DIR) | ||||
| 	@rm -rf $(BUILD_DIR) | ||||
| 
 | ||||
| 
 | ||||
| #
 | ||||
| # ================ Local build rules =================
 | ||||
| # examples:
 | ||||
| # make MK_ARG="-std=c++14 -fconcepts" build-gcc
 | ||||
| # make MK_ARG="-std=c++17" build-clang
 | ||||
| #
 | ||||
| .PHONY: build-gcc | ||||
| build-gcc: CFLAGS += $(MK_ARG) | ||||
| build-gcc: $(BUILD_DIR)/$(TARGET) | ||||
| 
 | ||||
| .PHONY: build-clang | ||||
| build-clang: CXX := $(CLANGXX) | ||||
| build-clang: CFLAGS += $(MK_ARG) | ||||
| build-clang: $(BUILD_DIR)/$(TARGET) | ||||
| 
 | ||||
| .PHONY: debug | ||||
| debug: $(BUILD_DIR)/$(TARGET) | ||||
| 
 | ||||
| .PHONY: release | ||||
| release: CFLAGS := $(REL_FLAGS) | ||||
| release: clean $(BUILD_DIR)/$(TARGET) | ||||
| 
 | ||||
| .PHONY: all | ||||
| all: clean release | ||||
| 
 | ||||
| #
 | ||||
| # ================ Docker based rules ================ 
 | ||||
| # examples:
 | ||||
| # make IMAGE="gcc:8.3" MK_ARG="-std=c++14 -fconcepts" dock-gcc
 | ||||
| # make IMAGE="a-clang-image" MK_ARG="-std=c++17" dock-clang
 | ||||
| #
 | ||||
| .PHONY: dock-gcc | ||||
| dock-gcc: DOCKER := $(DOCKER_RUN) $(IMAGE) | ||||
| dock-gcc: CFLAGS += $(MK_ARG) | ||||
| dock-gcc: $(BUILD_DIR)/$(TARGET) | ||||
| 
 | ||||
| .PHONY: dock-clang | ||||
| dock-clang: CXX := $(CLANGXX) | ||||
| dock-clang: DOCKER := $(DOCKER_RUN) $(IMAGE) | ||||
| dock-clang: CFLAGS += $(MK_ARG) | ||||
| dock-clang: $(BUILD_DIR)/$(TARGET) | ||||
| 
 | ||||
| 
 | ||||
							
								
								
									
										11673
									
								
								test/gtest/gtest/gtest-all.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										11673
									
								
								test/gtest/gtest/gtest-all.cpp
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										17103
									
								
								test/gtest/gtest/gtest.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										17103
									
								
								test/gtest/gtest/gtest.h
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										34
									
								
								test/main.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										34
									
								
								test/main.cpp
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,34 @@ | ||||
| /*!
 | ||||
|  * \file main.cpp | ||||
|  * \brief   Test project main file | ||||
|  * | ||||
|  * \copyright Copyright (C) 2020-2021 Christos Choutouridis <christos@choutouridis.net> | ||||
|  * | ||||
|  * <dl class=\"section copyright\"><dt>License</dt><dd> | ||||
|  * Permission is hereby granted, free of charge, to any person obtaining a copy | ||||
|  * of this software and associated documentation files (the "Software"), to deal | ||||
|  * in the Software without restriction, including without limitation the rights | ||||
|  * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell | ||||
|  * copies of the Software, and to permit persons to whom the Software is | ||||
|  * furnished to do so, subject to the following conditions: | ||||
|  * | ||||
|  * The above copyright notice and this permission notice shall be included in all | ||||
|  * copies or substantial portions of the Software. | ||||
|  * | ||||
|  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR | ||||
|  * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, | ||||
|  * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE | ||||
|  * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER | ||||
|  * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, | ||||
|  * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE | ||||
|  * SOFTWARE. | ||||
|  * </dd></dl> | ||||
|  * | ||||
|  */ | ||||
| #include <gtest/gtest.h> | ||||
| 
 | ||||
| GTEST_API_ int main(int argc, char **argv) { | ||||
|    testing::InitGoogleTest(&argc, argv); | ||||
|    return RUN_ALL_TESTS(); | ||||
| } | ||||
| 
 | ||||
							
								
								
									
										231
									
								
								test/tests/ring_iterator.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										231
									
								
								test/tests/ring_iterator.cpp
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,231 @@ | ||||
| /*!
 | ||||
|  * \file ring_iterator.cpp | ||||
|  * \brief | ||||
|  *      Unit tests for ring_iterator | ||||
|  * | ||||
|  * \copyright Copyright (C) 2020 Christos Choutouridis <christos@choutouridis.net> | ||||
|  * | ||||
|  * <dl class=\"section copyright\"><dt>License</dt><dd> | ||||
|  * The MIT License (MIT) | ||||
|  * | ||||
|  * Permission is hereby granted, free of charge, to any person obtaining a copy | ||||
|  * of this software and associated documentation files (the "Software"), to deal | ||||
|  * in the Software without restriction, including without limitation the rights | ||||
|  * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell | ||||
|  * copies of the Software, and to permit persons to whom the Software is | ||||
|  * furnished to do so, subject to the following conditions: | ||||
|  * | ||||
|  * The above copyright notice and this permission notice shall be included in all | ||||
|  * copies or substantial portions of the Software. | ||||
|  * | ||||
|  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR | ||||
|  * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, | ||||
|  * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE | ||||
|  * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER | ||||
|  * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, | ||||
|  * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE | ||||
|  * SOFTWARE. | ||||
|  * </dd></dl> | ||||
|  * | ||||
|  */ | ||||
| #include <core/ring_iterator.h> | ||||
| #include <gtest/gtest.h> | ||||
| 
 | ||||
| #include <array> | ||||
| #include <type_traits> | ||||
| 
 | ||||
| namespace Tring_iterator { | ||||
|     using namespace tbx; | ||||
| 
 | ||||
|     // Test construction
 | ||||
|     TEST(Tring_iterator, construct) { | ||||
|         int A[10]; | ||||
| 
 | ||||
|         //default constructor
 | ||||
|         ring_iterator<int*, 10> i1; | ||||
|         EXPECT_EQ(nullptr, i1.base()); | ||||
|         EXPECT_EQ(nullptr, i1.iter()); | ||||
|         EXPECT_EQ(10UL, i1.size()); | ||||
| 
 | ||||
|         // implementation specific (you can remove it freely)
 | ||||
|         EXPECT_EQ(2*sizeof(int*), sizeof(i1)); | ||||
| 
 | ||||
|         // basic
 | ||||
|         ring_iterator<int*, 10> i2(A); | ||||
|         EXPECT_EQ(A, i2.base()); | ||||
|         EXPECT_EQ(A, i2.iter()); | ||||
|         EXPECT_EQ(10UL, i2.size()); | ||||
| 
 | ||||
|         // basic from assignment
 | ||||
|         ring_iterator<int*, 10> i3 = A; | ||||
|         EXPECT_EQ(A, i3.base()); | ||||
|         EXPECT_EQ(A, i3.iter()); | ||||
|         EXPECT_EQ(10UL, i3.size()); | ||||
| 
 | ||||
|         // basic with offset
 | ||||
|         ring_iterator<int*, 10> i4(A, 5); | ||||
|         EXPECT_EQ(A, i4.base()); | ||||
|         EXPECT_EQ(&A[5], i4.iter()); | ||||
|         EXPECT_EQ(10UL, i4.size()); | ||||
| 
 | ||||
|         // copy (Legacy iterator)
 | ||||
|         auto i5 = i2; | ||||
|         EXPECT_EQ(A, i5.base()); | ||||
|         EXPECT_EQ(A, i5.iter()); | ||||
|         EXPECT_EQ(10UL, i5.size()); | ||||
| 
 | ||||
|         // arbitrary type
 | ||||
|         struct TT { int a,b,c; }; | ||||
|         std::array<TT, 10> t; | ||||
|         ring_iterator<TT*, 10> it(t.data(), 2); | ||||
|         EXPECT_EQ(t.begin(), it.base()); | ||||
|         EXPECT_EQ(&t[2], it.iter()); | ||||
|         EXPECT_EQ(10UL, it.size()); | ||||
|     } | ||||
| 
 | ||||
|     // Legacy iterator
 | ||||
|     TEST(Tring_iterator, LegacyIterator) { | ||||
| 
 | ||||
|         EXPECT_EQ(true, (std::is_same<int, typename ring_iterator<int*, 10>::value_type>::value)); | ||||
|         EXPECT_EQ(true, (std::is_same<std::ptrdiff_t, typename ring_iterator<int*, 10>::difference_type>::value)); | ||||
|         EXPECT_EQ(true, (std::is_same<int&, typename ring_iterator<int*, 10>::reference>::value)); | ||||
|         EXPECT_EQ(true, (std::is_same<int*, typename ring_iterator<int*, 10>::pointer>::value)); | ||||
|         EXPECT_EQ(true, (std::is_same<std::random_access_iterator_tag, typename ring_iterator<int*, 10>::iterator_category>::value)); | ||||
| 
 | ||||
|         int A[10] {0, 1, 2, 3, 4, 5, 6, 7, 8 , 9}; | ||||
|         ring_iterator<int*, 10> i1(A); | ||||
| 
 | ||||
|         // copy constructible/assignable
 | ||||
|         auto i2 = i1; | ||||
|         EXPECT_EQ(A, i2.base()); | ||||
|         EXPECT_EQ(A, i2.iter()); | ||||
|         EXPECT_EQ(10UL, i2.size()); | ||||
| 
 | ||||
|         // dereferenceable - incrementable
 | ||||
|         ring_iterator<int*, 10> i3(A); | ||||
|         EXPECT_EQ(true, (std::is_reference<decltype(*i3)>::value)); | ||||
|         EXPECT_EQ(true, (std::is_same<ring_iterator<int*, 10>&, decltype(++i3)>::value)); | ||||
|         EXPECT_EQ(true, (std::is_reference<decltype((*i3++))>::value)); | ||||
| 
 | ||||
|         // more practical
 | ||||
|         ring_iterator<int*, 10> i4(A); | ||||
|         ring_iterator<int*, 10> i5(A, 9); | ||||
|         EXPECT_EQ(A[0], *i4); | ||||
|         EXPECT_EQ(&A[1], (++i4).iter()); | ||||
|         // check loop
 | ||||
|         EXPECT_EQ(A[9], *i5); | ||||
|         EXPECT_EQ(&A[0], (++i5).iter()); | ||||
|     } | ||||
| 
 | ||||
|     // Legacy input iterator
 | ||||
|     TEST(Tring_iterator, LegacyInputIterator) { | ||||
|         int A[10] {0, 1, 2, 3, 4, 5, 6, 7, 8 , 9}; | ||||
|         ring_iterator<int*, 10> i1(A), i2(A), i3(A, 1); | ||||
| 
 | ||||
|         struct T { int m; }; | ||||
|         T B[5] { {0}, {1}, {2}, {3}, {4}}; | ||||
|         ring_iterator<T*, 5> it(B); | ||||
| 
 | ||||
|         EXPECT_EQ (true, (std::is_same<bool, decltype(i1 == i2)>::value)); | ||||
|         EXPECT_EQ (true, (std::is_same<bool, decltype(i1 != i2)>::value)); | ||||
|         EXPECT_EQ (true, (std::is_same<int&, decltype(*i1)>::value)); | ||||
|         EXPECT_EQ (true, (std::is_same<int, decltype(it->m)>::value)); | ||||
|         EXPECT_EQ (true, (std::is_same<ring_iterator<int*, 10>&, decltype(++i1)>::value)); | ||||
|         EXPECT_EQ (true, (std::is_same<int&, decltype(*i1++)>::value)); | ||||
| 
 | ||||
|         // more practical
 | ||||
|         EXPECT_EQ (true, i1 == i2); | ||||
|         EXPECT_EQ (true, i1 != i3); | ||||
|         EXPECT_EQ (0, *i1); | ||||
|         EXPECT_EQ (0, it->m); | ||||
|         EXPECT_EQ (true, (++i1 == i3)); | ||||
|         EXPECT_EQ (1, *i1++); | ||||
|         EXPECT_EQ (2, *i1); | ||||
|     } | ||||
| 
 | ||||
|     // Legacy input iterator
 | ||||
|     TEST(Tring_iterator, LegacyOutputIterator) { | ||||
|         int A[10] {0, 1, 2, 3, 4, 5, 6, 7, 8 , 9}; | ||||
|         ring_iterator<int*, 10> it(A); | ||||
| 
 | ||||
|         EXPECT_EQ (true, (std::is_assignable<decltype(*it), int>::value)); | ||||
|         EXPECT_EQ (true, (std::is_assignable<decltype(*it++), int>::value)); | ||||
| 
 | ||||
|         // more practical
 | ||||
|         *it = 42; | ||||
|         EXPECT_EQ (42, A[0]); | ||||
|         *it++ = 7; | ||||
|         EXPECT_EQ (7, A[0]); | ||||
|         EXPECT_EQ (&A[1], it.iter()); | ||||
|     } | ||||
| 
 | ||||
|     // Legacy forward iterator
 | ||||
|     TEST(Tring_iterator, LegacyForwardIterator) | ||||
|      { | ||||
|         int A[10] {0, 1, 2, 3, 4, 5, 6, 7, 8 , 9}; | ||||
|         ring_iterator<int*, 10> it(A); | ||||
| 
 | ||||
|         EXPECT_EQ (0, *it++); | ||||
|         EXPECT_EQ (1, *it); | ||||
|     } | ||||
| 
 | ||||
|     // Legacy bidirectional iterator
 | ||||
|     TEST(Tring_iterator, LegacyBidirectionalIterator) { | ||||
|         int A[10] {0, 1, 2, 3, 4, 5, 6, 7, 8 , 9}; | ||||
|         ring_iterator<int*, 10> it(A); | ||||
| 
 | ||||
|         EXPECT_EQ (true, (std::is_same<ring_iterator<int*, 10>&, decltype(--it)>::value)); | ||||
|         EXPECT_EQ (true, (std::is_same<ring_iterator<int*, 10>, decltype(it--)>::value)); | ||||
|         EXPECT_EQ (true, (std::is_same<int&, decltype(*it--)>::value)); | ||||
| 
 | ||||
|         // more practical
 | ||||
|         ring_iterator<int*, 10> i1(A), i2(A, 9); | ||||
|         EXPECT_EQ (9, *i2--);   // check loop also
 | ||||
|         EXPECT_EQ (8, *i2); | ||||
|         EXPECT_EQ (0, *i1--);   // check loop also
 | ||||
|         EXPECT_EQ (9, *i1); | ||||
|     } | ||||
| 
 | ||||
|     // Legacy random access iterator
 | ||||
|     TEST(Tring_iterator, LegacyRandomAccessIterator) { | ||||
|         int A[10] {0, 1, 2, 3, 4, 5, 6, 7, 8 , 9}; | ||||
|         ring_iterator<int*, 10> it1(A), it2(A, 7); | ||||
| 
 | ||||
|         EXPECT_EQ (true, (std::is_same<ring_iterator<int*, 10>&, decltype(it1 += 7)>::value)); | ||||
|         EXPECT_EQ (true, (std::is_same<ring_iterator<int*, 10>, decltype(it1 + 7)>::value)); | ||||
|         EXPECT_EQ (true, (std::is_same<ring_iterator<int*, 10>, decltype(7 + it1)>::value)); | ||||
| 
 | ||||
|         EXPECT_EQ (true, (std::is_same<ring_iterator<int*, 10>&, decltype(it1 -= 7)>::value)); | ||||
|         EXPECT_EQ (true, (std::is_same<ring_iterator<int*, 10>, decltype(it1 - 7)>::value)); | ||||
| 
 | ||||
|         EXPECT_EQ (true, (std::is_same<std::ptrdiff_t, decltype(it1 - it2)>::value)); | ||||
|         EXPECT_EQ (true, (std::is_same<int&, decltype(it1[7])>::value)); | ||||
| 
 | ||||
|         EXPECT_EQ (true, (std::is_same<bool, decltype(it1 < it2)>::value)); | ||||
|         EXPECT_EQ (true, (std::is_same<bool, decltype(it1 > it2)>::value)); | ||||
|         EXPECT_EQ (true, (std::is_same<bool, decltype(it1 <= it2)>::value)); | ||||
|         EXPECT_EQ (true, (std::is_same<bool, decltype(it1 >= it2)>::value)); | ||||
| 
 | ||||
|         // more practical
 | ||||
|         ring_iterator<int*, 10> i1(A), i2(A); | ||||
|         i1 += 7; | ||||
|         EXPECT_EQ (7, *i1); | ||||
|         i1 -= 7; | ||||
|         EXPECT_EQ (0, *i1); | ||||
|         i1 += 11; | ||||
|         EXPECT_EQ (1, *i1); | ||||
|         i1 -= 2; | ||||
|         EXPECT_EQ (9, *i1); | ||||
| 
 | ||||
|         EXPECT_EQ (7, *(i2+7)); | ||||
|         EXPECT_EQ (7, *(7+i2)); | ||||
|         EXPECT_EQ (1, *(i2+11)); | ||||
|         EXPECT_EQ (1, *(11+i2)); | ||||
|         EXPECT_EQ (7, *(i1-2)); | ||||
|         EXPECT_EQ (8, *(i2-2)); | ||||
| 
 | ||||
|         EXPECT_EQ (9, (i1 - i2)); | ||||
|         EXPECT_EQ (1, (i2 - i1));   // loop
 | ||||
| 
 | ||||
|     } | ||||
| } | ||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user