/* * FileIdentifier.h * * This source file is part of the FoundationDB open source project * * Copyright 2013-2018 Apple Inc. and the FoundationDB project authors * * 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. */ #pragma once #include #include using FileIdentifier = uint32_t; template struct HasFileIdentifierMember : std::false_type {}; template struct HasFileIdentifierMember : std::true_type {}; template struct CompositionDepthFor : std::integral_constant {}; template struct CompositionDepthFor : std::integral_constant {}; template struct FileIdentifierForBase; template struct FileIdentifierForBase {}; template struct FileIdentifierForBase { static constexpr FileIdentifier value = T::file_identifier; static_assert(CompositionDepthFor::value > 0 || T::file_identifier < (1 << 24), "non-composed file identifiers must be less than 2^24"); }; template struct FileIdentifierFor : FileIdentifierForBase::value> {}; template struct HasFileIdentifier : std::false_type {}; template struct HasFileIdentifier::value, 0)> : std::true_type {}; template ::value && CompositionDepthFor::value < 2)> struct ComposedIdentifier; // Manually specified file identifiers must be less than 2^24. // The first 8 bits are used to identify the wrapper classes: // The 5th-8th bits represent the inner wrapper class // The 1st-4th bits represent the outer wrapper class // Types with more than two level of composition do not get file identifiers. template struct ComposedIdentifier { static_assert(CompositionDepthFor::value < 2); static constexpr int composed_identifier_offset = (CompositionDepthFor::value == 1) ? 28 : 24; static_assert(B > 0 && B < 16, "Up to 15 types of composed identifiers allowed"); static_assert(FileIdentifierFor::value < (1 << composed_identifier_offset)); public: static constexpr int composition_depth = CompositionDepthFor::value + 1; static constexpr FileIdentifier file_identifier = (B << composed_identifier_offset) | FileIdentifierFor::value; }; template struct ComposedIdentifier {}; template ::value> struct ComposedIdentifierExternal; template struct ComposedIdentifierExternal {}; template struct ComposedIdentifierExternal { static constexpr FileIdentifier value = ComposedIdentifier::file_identifier; };