Program Listing for File CdrSizeCalculator.hpp

Return to documentation for file (include/fastcdr/CdrSizeCalculator.hpp)

// Copyright 2023 Proyectos y Sistemas de Mantenimiento SL (eProsima).
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
//     http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.

#ifndef _FASTCDR_CDRSIZECALCULATOR_HPP_
#define _FASTCDR_CDRSIZECALCULATOR_HPP_

#include <array>
#include <bitset>
#include <cstdint>
#include <limits>
#include <map>
#include <vector>

#include "fastcdr_dll.h"

#include "CdrEncoding.hpp"
#include "cdr/fixed_size_string.hpp"
#include "detail/container_recursive_inspector.hpp"
#include "exceptions/BadParamException.h"
#include "xcdr/external.hpp"
#include "xcdr/MemberId.hpp"
#include "xcdr/optional.hpp"

namespace eprosima {
namespace fastcdr {

class CdrSizeCalculator;

template<class _T>
extern size_t calculate_serialized_size(
        CdrSizeCalculator&,
        const _T&,
        size_t&);

class CdrSizeCalculator
{
public:

    Cdr_DllAPI CdrSizeCalculator(
            CdrVersion cdr_version);

    Cdr_DllAPI CdrSizeCalculator(
            CdrVersion cdr_version,
            EncodingAlgorithmFlag encoding);

    Cdr_DllAPI CdrVersion get_cdr_version() const;

    Cdr_DllAPI EncodingAlgorithmFlag get_encoding() const;

    template<class _T, typename std::enable_if<!std::is_enum<_T>::value>::type* = nullptr, typename = void>
    size_t calculate_serialized_size(
            const _T& data,
            size_t& current_alignment)
    {
        return eprosima::fastcdr::calculate_serialized_size(*this, data, current_alignment);
    }

    template<class _T,
            typename std::enable_if<std::is_enum<_T>::value>::type* = nullptr,
            typename std::enable_if<std::is_same<typename std::underlying_type<_T>::type,
            int32_t>::value>::type* = nullptr>
    size_t calculate_serialized_size(
            const _T& data,
            size_t& current_alignment)
    {
        return calculate_serialized_size(static_cast<int32_t>(data), current_alignment);
    }

    template<class _T,
            typename std::enable_if<std::is_enum<_T>::value>::type* = nullptr,
            typename std::enable_if<std::is_same<typename std::underlying_type<_T>::type,
            uint32_t>::value>::type* = nullptr>
    size_t calculate_serialized_size(
            const _T& data,
            size_t& current_alignment)
    {
        return calculate_serialized_size(static_cast<uint32_t>(data), current_alignment);
    }

    template<class _T,
            typename std::enable_if<std::is_enum<_T>::value>::type* = nullptr,
            typename std::enable_if<std::is_same<typename std::underlying_type<_T>::type,
            int16_t>::value>::type* = nullptr>
    size_t calculate_serialized_size(
            const _T& data,
            size_t& current_alignment)
    {
        return calculate_serialized_size(static_cast<int16_t>(data), current_alignment);
    }

    template<class _T,
            typename std::enable_if<std::is_enum<_T>::value>::type* = nullptr,
            typename std::enable_if<std::is_same<typename std::underlying_type<_T>::type,
            uint16_t>::value>::type* = nullptr>
    size_t calculate_serialized_size(
            const _T& data,
            size_t& current_alignment)
    {
        return calculate_serialized_size(static_cast<uint16_t>(data), current_alignment);
    }

    template<class _T,
            typename std::enable_if<std::is_enum<_T>::value>::type* = nullptr,
            typename std::enable_if<std::is_same<typename std::underlying_type<_T>::type,
            int8_t>::value>::type* = nullptr>
    size_t calculate_serialized_size(
            const _T& data,
            size_t& current_alignment)
    {
        return calculate_serialized_size(static_cast<int8_t>(data), current_alignment);
    }

    template<class _T,
            typename std::enable_if<std::is_enum<_T>::value>::type* = nullptr,
            typename std::enable_if<std::is_same<typename std::underlying_type<_T>::type,
            uint8_t>::value>::type* = nullptr>
    size_t calculate_serialized_size(
            const _T& data,
            size_t& current_alignment)
    {
        return calculate_serialized_size(static_cast<uint8_t>(data), current_alignment);
    }

    TEMPLATE_SPEC
    size_t calculate_serialized_size(
            const int8_t& data,
            size_t& current_alignment)
    {
        static_cast<void>(data);
        ++current_alignment;
        return 1;
    }

    TEMPLATE_SPEC
    size_t calculate_serialized_size(
            const uint8_t& data,
            size_t& current_alignment)
    {
        static_cast<void>(data);
        ++current_alignment;
        return 1;
    }

    TEMPLATE_SPEC
    size_t calculate_serialized_size(
            const char& data,
            size_t& current_alignment)
    {
        static_cast<void>(data);
        ++current_alignment;
        return 1;
    }

    TEMPLATE_SPEC
    size_t calculate_serialized_size(
            const bool& data,
            size_t& current_alignment)
    {
        static_cast<void>(data);
        ++current_alignment;
        return 1;
    }

    TEMPLATE_SPEC
    size_t calculate_serialized_size(
            const wchar_t& data,
            size_t& current_alignment)
    {
        static_cast<void>(data);
        size_t calculated_size {2 + alignment(current_alignment, 2)};
        current_alignment += calculated_size;
        return calculated_size;
    }

    TEMPLATE_SPEC
    size_t calculate_serialized_size(
            const int16_t& data,
            size_t& current_alignment)
    {
        static_cast<void>(data);
        size_t calculated_size {2 + alignment(current_alignment, 2)};
        current_alignment += calculated_size;
        return calculated_size;
    }

    TEMPLATE_SPEC
    size_t calculate_serialized_size(
            const uint16_t& data,
            size_t& current_alignment)
    {
        static_cast<void>(data);
        size_t calculated_size {2 + alignment(current_alignment, 2)};
        current_alignment += calculated_size;
        return calculated_size;
    }

    TEMPLATE_SPEC
    size_t calculate_serialized_size(
            const int32_t& data,
            size_t& current_alignment)
    {
        static_cast<void>(data);
        size_t calculated_size {4 + alignment(current_alignment, 4)};
        current_alignment += calculated_size;
        return calculated_size;
    }

    TEMPLATE_SPEC
    size_t calculate_serialized_size(
            const uint32_t& data,
            size_t& current_alignment)
    {
        static_cast<void>(data);
        size_t calculated_size {4 + alignment(current_alignment, 4)};
        current_alignment += calculated_size;
        return calculated_size;
    }

    TEMPLATE_SPEC
    size_t calculate_serialized_size(
            const int64_t& data,
            size_t& current_alignment)
    {
        static_cast<void>(data);
        size_t calculated_size {8 + alignment(current_alignment, align64_)};
        current_alignment += calculated_size;
        return calculated_size;
    }

    TEMPLATE_SPEC
    size_t calculate_serialized_size(
            const uint64_t& data,
            size_t& current_alignment)
    {
        static_cast<void>(data);
        size_t calculated_size {8 + alignment(current_alignment, align64_)};
        current_alignment += calculated_size;
        return calculated_size;
    }

    TEMPLATE_SPEC
    size_t calculate_serialized_size(
            const float& data,
            size_t& current_alignment)
    {
        static_cast<void>(data);
        size_t calculated_size {4 + alignment(current_alignment, 4)};
        current_alignment += calculated_size;
        return calculated_size;
    }

    TEMPLATE_SPEC
    size_t calculate_serialized_size(
            const double& data,
            size_t& current_alignment)
    {
        static_cast<void>(data);
        size_t calculated_size {8 + alignment(current_alignment, align64_)};
        current_alignment += calculated_size;
        return calculated_size;
    }

    TEMPLATE_SPEC
    size_t calculate_serialized_size(
            const long double& data,
            size_t& current_alignment)
    {
        static_cast<void>(data);
        size_t calculated_size {16 + alignment(current_alignment, align64_)};
        current_alignment += calculated_size;
        return calculated_size;
    }

    TEMPLATE_SPEC
    size_t calculate_serialized_size(
            const std::string& data,
            size_t& current_alignment)
    {
        size_t calculated_size {4 + alignment(current_alignment, 4) + data.size() + 1};
        current_alignment += calculated_size;
        serialized_member_size_ = SERIALIZED_MEMBER_SIZE;

        return calculated_size;
    }

    TEMPLATE_SPEC
    size_t calculate_serialized_size(
            const std::wstring& data,
            size_t& current_alignment)
    {
        size_t calculated_size {4 + alignment(current_alignment, 4) + data.size() * 2};
        current_alignment += calculated_size;

        return calculated_size;
    }

    template <size_t MAX_CHARS>
    size_t calculate_serialized_size(
            const fixed_string<MAX_CHARS>& data,
            size_t& current_alignment)
    {
        size_t calculated_size {4 + alignment(current_alignment, 4) + data.size() + 1};
        current_alignment += calculated_size;
        serialized_member_size_ = SERIALIZED_MEMBER_SIZE;

        return calculated_size;
    }

    template<class _T, typename std::enable_if<!std::is_enum<_T>::value &&
            !std::is_arithmetic<_T>::value>::type* = nullptr>
    size_t calculate_serialized_size(
            const std::vector<_T>& data,
            size_t& current_alignment)
    {
        size_t initial_alignment {current_alignment};

        if (CdrVersion::XCDRv2 == cdr_version_)
        {
            // DHEADER
            current_alignment += 4 + alignment(current_alignment, 4);
        }

        current_alignment += 4 + alignment(current_alignment, 4);

        size_t calculated_size {current_alignment - initial_alignment};
        calculated_size += calculate_array_serialized_size(data.data(), data.size(), current_alignment);

        if (CdrVersion::XCDRv2 == cdr_version_)
        {
            // Inform DHEADER can be joined with NEXTINT
            serialized_member_size_ = SERIALIZED_MEMBER_SIZE;
        }

        return calculated_size;
    }

    template<class _T, typename std::enable_if<std::is_enum<_T>::value ||
            std::is_arithmetic<_T>::value>::type* = nullptr>
    size_t calculate_serialized_size(
            const std::vector<_T>& data,
            size_t& current_alignment)
    {
        size_t initial_alignment {current_alignment};

        current_alignment += 4 + alignment(current_alignment, 4);

        size_t calculated_size {current_alignment - initial_alignment};
        calculated_size += calculate_array_serialized_size(data.data(), data.size(), current_alignment);

        if (CdrVersion::XCDRv2 == cdr_version_)
        {
            serialized_member_size_ = get_serialized_member_size<_T>();
        }

        return calculated_size;
    }

    TEMPLATE_SPEC
    size_t calculate_serialized_size(
            const std::vector<bool>& data,
            size_t& current_alignment)
    {
        size_t calculated_size {data.size() + 4 + alignment(current_alignment, 4)};
        current_alignment += calculated_size;

        return calculated_size;
    }

    template<class _T, size_t _Size>
    size_t calculate_serialized_size(
            const std::array<_T, _Size>& data,
            size_t& current_alignment)
    {
        size_t initial_alignment {current_alignment};

        if (CdrVersion::XCDRv2 == cdr_version_ &&
                !is_multi_array_primitive(&data))
        {
            // DHEADER
            current_alignment += 4 + alignment(current_alignment, 4);
        }

        size_t calculated_size {current_alignment - initial_alignment};
        calculated_size += calculate_array_serialized_size(data.data(), data.size(), current_alignment);

        if (CdrVersion::XCDRv2 == cdr_version_ &&
                !is_multi_array_primitive(&data))
        {
            // Inform DHEADER can be joined with NEXTINT
            serialized_member_size_ = SERIALIZED_MEMBER_SIZE;
        }

        return calculated_size;
    }

    template<class _K, class _V, typename std::enable_if<!std::is_enum<_V>::value &&
            !std::is_arithmetic<_V>::value>::type* = nullptr>
    size_t calculate_serialized_size(
            const std::map<_K, _V>& data,
            size_t& current_alignment)
    {
        size_t initial_alignment {current_alignment};

        if (CdrVersion::XCDRv2 == cdr_version_)
        {
            // DHEADER
            current_alignment += 4 + alignment(current_alignment, 4);
        }

        current_alignment += 4 + alignment(current_alignment, 4);

        size_t calculated_size {current_alignment - initial_alignment};
        for (auto it = data.begin(); it != data.end(); ++it)
        {
            calculated_size += calculate_serialized_size(it->first, current_alignment);
            calculated_size += calculate_serialized_size(it->second, current_alignment);
        }

        if (CdrVersion::XCDRv2 == cdr_version_)
        {
            // Inform DHEADER can be joined with NEXTINT
            serialized_member_size_ = SERIALIZED_MEMBER_SIZE;
        }

        return calculated_size;
    }

    template<class _K, class _V, typename std::enable_if<std::is_enum<_V>::value ||
            std::is_arithmetic<_V>::value>::type* = nullptr>
    size_t calculate_serialized_size(
            const std::map<_K, _V>& data,
            size_t& current_alignment)
    {
        size_t initial_alignment {current_alignment};

        current_alignment += 4 + alignment(current_alignment, 4);

        size_t calculated_size {current_alignment - initial_alignment};
        for (auto it = data.begin(); it != data.end(); ++it)
        {
            calculated_size += calculate_serialized_size(it->first, current_alignment);
            calculated_size += calculate_serialized_size(it->second, current_alignment);
        }

        return calculated_size;
    }

    template<size_t N, typename std::enable_if < (N < 9) > ::type* = nullptr>
    size_t calculate_serialized_size(
            const std::bitset<N>& data,
            size_t& current_alignment)
    {
        static_cast<void>(data);
        ++current_alignment;
        return 1;
    }

    template<size_t N, typename std::enable_if < (8 < N && N < 17) > ::type* = nullptr>
    size_t calculate_serialized_size(
            const std::bitset<N>& data,
            size_t& current_alignment)
    {
        static_cast<void>(data);
        size_t calculated_size {2 + alignment(current_alignment, 2)};
        current_alignment += calculated_size;
        return calculated_size;
    }

    template<size_t N, typename std::enable_if < (16 < N && N < 33) > ::type* = nullptr>
    size_t calculate_serialized_size(
            const std::bitset<N>& data,
            size_t& current_alignment)
    {
        static_cast<void>(data);
        size_t calculated_size {4 + alignment(current_alignment, 4)};
        current_alignment += calculated_size;
        return calculated_size;
    }

    template<size_t N, typename std::enable_if < (32 < N && N < 65) > ::type* = nullptr>
    size_t calculate_serialized_size(
            const std::bitset<N>& data,
            size_t& current_alignment)
    {
        static_cast<void>(data);
        size_t calculated_size {8 + alignment(current_alignment, align64_)};
        current_alignment += calculated_size;
        return calculated_size;
    }

    template<class _T>
    size_t calculate_serialized_size(
            const optional<_T>& data,
            size_t& current_alignment)
    {
        size_t initial_alignment = current_alignment;

        if (CdrVersion::XCDRv2 == cdr_version_ &&
                EncodingAlgorithmFlag::PL_CDR2 != current_encoding_)
        {
            // Take into account the boolean is_present;
            ++current_alignment;
        }

        size_t calculated_size {current_alignment - initial_alignment};

        if (data.has_value())
        {
            calculated_size += calculate_serialized_size(data.value(), current_alignment);
        }

        return calculated_size;
    }

    template<class _T>
    size_t calculate_serialized_size(
            const external<_T>& data,
            size_t& current_alignment)
    {
        if (!data)
        {
            throw exception::BadParamException("External member is null");
        }

        return calculate_serialized_size(*data, current_alignment);
    }

    template<class _T>
    size_t calculate_array_serialized_size(
            const _T* data,
            size_t num_elements,
            size_t& current_alignment)
    {
        size_t calculated_size {0};

        for (size_t count = 0; count < num_elements; ++count)
        {
            calculated_size += calculate_serialized_size(data[count], current_alignment);
        }

        return calculated_size;
    }

    TEMPLATE_SPEC
    size_t calculate_array_serialized_size(
            const int8_t* data,
            size_t num_elements,
            size_t& current_alignment)
    {
        static_cast<void>(data);
        current_alignment += num_elements;
        return num_elements;
    }

    TEMPLATE_SPEC
    size_t calculate_array_serialized_size(
            const uint8_t* data,
            size_t num_elements,
            size_t& current_alignment)
    {
        static_cast<void>(data);
        current_alignment += num_elements;
        return num_elements;
    }

    TEMPLATE_SPEC
    size_t calculate_array_serialized_size(
            const char* data,
            size_t num_elements,
            size_t& current_alignment)
    {
        static_cast<void>(data);
        current_alignment += num_elements;
        return num_elements;
    }

    TEMPLATE_SPEC
    size_t calculate_array_serialized_size(
            const wchar_t* data,
            size_t num_elements,
            size_t& current_alignment)
    {
        static_cast<void>(data);
        size_t calculated_size {num_elements* 2 + alignment(current_alignment, 2)};
        current_alignment += calculated_size;
        return calculated_size;
    }

    TEMPLATE_SPEC
    size_t calculate_array_serialized_size(
            const int16_t* data,
            size_t num_elements,
            size_t& current_alignment)
    {
        static_cast<void>(data);
        size_t calculated_size {num_elements* 2 + alignment(current_alignment, 2)};
        current_alignment += calculated_size;
        return calculated_size;
    }

    TEMPLATE_SPEC
    size_t calculate_array_serialized_size(
            const uint16_t* data,
            size_t num_elements,
            size_t& current_alignment)
    {
        static_cast<void>(data);
        size_t calculated_size {num_elements* 2 + alignment(current_alignment, 2)};
        current_alignment += calculated_size;
        return calculated_size;
    }

    TEMPLATE_SPEC
    size_t calculate_array_serialized_size(
            const int32_t* data,
            size_t num_elements,
            size_t& current_alignment)
    {
        static_cast<void>(data);
        size_t calculated_size {num_elements* 4 + alignment(current_alignment, 4)};
        current_alignment += calculated_size;
        return calculated_size;
    }

    TEMPLATE_SPEC
    size_t calculate_array_serialized_size(
            const uint32_t* data,
            size_t num_elements,
            size_t& current_alignment)
    {
        static_cast<void>(data);
        size_t calculated_size {num_elements* 4 + alignment(current_alignment, 4)};
        current_alignment += calculated_size;
        return calculated_size;
    }

    TEMPLATE_SPEC
    size_t calculate_array_serialized_size(
            const int64_t* data,
            size_t num_elements,
            size_t& current_alignment)
    {
        static_cast<void>(data);
        size_t calculated_size {num_elements* 8 + alignment(current_alignment, align64_)};
        current_alignment += calculated_size;
        return calculated_size;
    }

    TEMPLATE_SPEC
    size_t calculate_array_serialized_size(
            const uint64_t* data,
            size_t num_elements,
            size_t& current_alignment)
    {
        static_cast<void>(data);
        size_t calculated_size {num_elements* 8 + alignment(current_alignment, align64_)};
        current_alignment += calculated_size;
        return calculated_size;
    }

    TEMPLATE_SPEC
    size_t calculate_array_serialized_size(
            const float* data,
            size_t num_elements,
            size_t& current_alignment)
    {
        static_cast<void>(data);
        size_t calculated_size {num_elements* 4 + alignment(current_alignment, 4)};
        current_alignment += calculated_size;
        return calculated_size;
    }

    TEMPLATE_SPEC
    size_t calculate_array_serialized_size(
            const double* data,
            size_t num_elements,
            size_t& current_alignment)
    {
        static_cast<void>(data);
        size_t calculated_size {num_elements* 8 + alignment(current_alignment, align64_)};
        current_alignment += calculated_size;
        return calculated_size;
    }

    TEMPLATE_SPEC
    size_t calculate_array_serialized_size(
            const long double* data,
            size_t num_elements,
            size_t& current_alignment)
    {
        static_cast<void>(data);
        size_t calculated_size {num_elements* 16 + alignment(current_alignment, align64_)};
        current_alignment += calculated_size;
        return calculated_size;
    }

    template<class _T, size_t _N>
    size_t calculate_array_serialized_size(
            const std::array<_T, _N>* data,
            size_t num_elements,
            size_t& current_alignment)
    {
        return calculate_array_serialized_size(data->data(), num_elements * data->size(), current_alignment);
    }

    template<class _T, typename std::enable_if<std::is_enum<_T>::value ||
            std::is_arithmetic<_T>::value>::type* = nullptr>
    size_t calculate_array_serialized_size(
            const std::vector<_T>& data,
            size_t& current_alignment)
    {
        return calculate_array_serialized_size(data.data(), data.size(), current_alignment);
    }

    template<class _T, typename std::enable_if<!std::is_enum<_T>::value &&
            !std::is_arithmetic<_T>::value>::type* = nullptr>
    size_t calculate_array_serialized_size(
            const std::vector<_T>& data,
            size_t& current_alignment)
    {
        size_t initial_alignment {current_alignment};

        if (CdrVersion::XCDRv2 == cdr_version_)
        {
            // DHEADER
            current_alignment += 4 + alignment(current_alignment, 4);
        }

        size_t calculated_size {current_alignment - initial_alignment};
        calculated_size += calculate_array_serialized_size(data.data(), data.size(), current_alignment);

        if (CdrVersion::XCDRv2 == cdr_version_)
        {
            // Inform DHEADER can be joined with NEXTINT
            serialized_member_size_ = SERIALIZED_MEMBER_SIZE;
        }

        return calculated_size;
    }

    TEMPLATE_SPEC
    size_t calculate_array_serialized_size(
            const std::vector<bool>& data,
            size_t& current_alignment)
    {
        current_alignment += data.size();
        return data.size();
    }

    template<class _T>
    size_t calculate_member_serialized_size(
            const MemberId& id,
            const _T& data,
            size_t& current_alignment)
    {
        size_t initial_alignment {current_alignment};

        if (EncodingAlgorithmFlag::PL_CDR == current_encoding_ ||
                EncodingAlgorithmFlag::PL_CDR2 == current_encoding_)
        {
            // Align to 4 for the XCDR header before calculating the data serialized size.
            current_alignment += alignment(current_alignment, 4);
        }

        size_t prev_size {current_alignment - initial_alignment};
        size_t extra_size {0};

        if (EncodingAlgorithmFlag::PL_CDR == current_encoding_)
        {
            current_alignment = 0;
        }

        size_t calculated_size {calculate_serialized_size(data, current_alignment)};

        if (CdrVersion::XCDRv2 == cdr_version_ && EncodingAlgorithmFlag::PL_CDR2 == current_encoding_ &&
                0 < calculated_size)
        {

            if (8 < calculated_size ||
                    (1 != calculated_size && 2 != calculated_size && 4 != calculated_size &&
                    8 != calculated_size))
            {
                extra_size = 8; // Long EMHEADER.
                if (NO_SERIALIZED_MEMBER_SIZE != serialized_member_size_)
                {
                    calculated_size -= 4; // Join NEXTINT and DHEADER.
                }
            }
            else
            {
                extra_size = 4; // EMHEADER;
            }
        }
        else if (CdrVersion::XCDRv1 == cdr_version_ && EncodingAlgorithmFlag::PL_CDR == current_encoding_ &&
                0 < calculated_size)
        {
            extra_size = 4; // ShortMemberHeader

            if (0x3F00 < id.id || calculated_size > std::numeric_limits<uint16_t>::max())
            {
                extra_size += 8; // LongMemberHeader
            }

        }

        calculated_size += prev_size + extra_size;
        if (EncodingAlgorithmFlag::PL_CDR != current_encoding_)
        {
            current_alignment += extra_size;
        }

        serialized_member_size_ = NO_SERIALIZED_MEMBER_SIZE;

        return calculated_size;
    }

    template<class _T>
    size_t calculate_member_serialized_size(
            const MemberId& id,
            const optional<_T>& data,
            size_t& current_alignment)
    {
        size_t initial_alignment = current_alignment;

        if (CdrVersion::XCDRv2 != cdr_version_ ||
                EncodingAlgorithmFlag::PL_CDR2 == current_encoding_)
        {
            if (data.has_value() || EncodingAlgorithmFlag::PLAIN_CDR == current_encoding_)
            {
                // Align to 4 for the XCDR header before calculating the data serialized size.
                current_alignment += alignment(current_alignment, 4);
            }
        }

        size_t prev_size = {current_alignment - initial_alignment};
        size_t extra_size {0};

        if (CdrVersion::XCDRv1 == cdr_version_ &&
                (data.has_value() || EncodingAlgorithmFlag::PLAIN_CDR == current_encoding_))
        {
            current_alignment = 0;
        }

        size_t calculated_size {calculate_serialized_size(data, current_alignment)};

        if (CdrVersion::XCDRv2 == cdr_version_ && EncodingAlgorithmFlag::PL_CDR2 == current_encoding_ &&
                0 < calculated_size)
        {
            if (8 < calculated_size)
            {
                extra_size = 8; // Long EMHEADER.
                if (NO_SERIALIZED_MEMBER_SIZE != serialized_member_size_)
                {
                    extra_size -= 4; // Join NEXTINT and DHEADER.
                }
            }
            else
            {
                extra_size = 4; // EMHEADER;
            }
        }
        else if (CdrVersion::XCDRv1 == cdr_version_ &&
                (0 < calculated_size || EncodingAlgorithmFlag::PLAIN_CDR == current_encoding_))
        {
            extra_size = 4; // ShortMemberHeader

            if (0x3F00 < id.id || calculated_size > std::numeric_limits<uint16_t>::max())
            {
                extra_size += 8; // LongMemberHeader
            }

        }

        calculated_size += prev_size + extra_size;
        if (CdrVersion::XCDRv1 != cdr_version_)
        {
            current_alignment += extra_size;
        }


        return calculated_size;
    }

    Cdr_DllAPI size_t begin_calculate_type_serialized_size(
            EncodingAlgorithmFlag new_encoding,
            size_t& current_alignment);

    Cdr_DllAPI size_t end_calculate_type_serialized_size(
            EncodingAlgorithmFlag new_encoding,
            size_t& current_alignment);

private:

    CdrSizeCalculator() = delete;

    CdrVersion cdr_version_ {CdrVersion::XCDRv2};

    EncodingAlgorithmFlag current_encoding_ {EncodingAlgorithmFlag::PLAIN_CDR2};

    enum SerializedMemberSizeForNextInt
    {
        NO_SERIALIZED_MEMBER_SIZE,
        SERIALIZED_MEMBER_SIZE,
        SERIALIZED_MEMBER_SIZE_4,
        SERIALIZED_MEMBER_SIZE_8
    }
    serialized_member_size_ {NO_SERIALIZED_MEMBER_SIZE};

    size_t align64_ {4};

    inline size_t alignment(
            size_t current_alignment,
            size_t data_size) const
    {
        return (data_size - (current_alignment % data_size)) & (data_size - 1);
    }

    template<class _T, typename std::enable_if<std::is_enum<_T>::value ||
            std::is_arithmetic<_T>::value>::type* = nullptr>
    constexpr SerializedMemberSizeForNextInt get_serialized_member_size() const
    {
        return (1 == sizeof(_T) ? SERIALIZED_MEMBER_SIZE :
               (4 == sizeof(_T) ? SERIALIZED_MEMBER_SIZE_4 :
               (8 == sizeof(_T) ? SERIALIZED_MEMBER_SIZE_8 :  NO_SERIALIZED_MEMBER_SIZE)));
    }

};

}        // namespace fastcdr
}        // namespace eprosima

#endif // _FASTCDR_CDRSIZECALCULATOR_HPP_