/*
 * Copyright (c) 2018-2020 Atmosphère-NX
 *
 * This program is free software; you can redistribute it and/or modify it
 * under the terms and conditions of the GNU General Public License,
 * version 2, as published by the Free Software Foundation.
 *
 * This program is distributed in the hope it will be useful, but WITHOUT
 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
 * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License for
 * more details.
 *
 * You should have received a copy of the GNU General Public License
 * along with this program.  If not, see <http://www.gnu.org/licenses/>.
 */
#include <stratosphere.hpp>

namespace ams::util {

    namespace {

        struct UuidImpl {
            util::BitPack32 data[4];

            using TimeLow               = util::BitPack32::Field<0,                           BITSIZEOF(u32), u32>;

            using TimeMid               = util::BitPack32::Field<0,                           BITSIZEOF(u16), u16>;
            using TimeHighAndVersion    = util::BitPack32::Field<TimeMid::Next,               BITSIZEOF(u16), u16>;
            using Version               = util::BitPack32::Field<TimeMid::Next + 12,                       4, u16>;

            static_assert(TimeHighAndVersion::Next == Version::Next);

            using ClockSeqHiAndReserved = util::BitPack32::Field<0,                           BITSIZEOF(u8),  u8>;
            using Reserved              = util::BitPack32::Field<6,                           2,              u8>;
            using ClockSeqLow           = util::BitPack32::Field<ClockSeqHiAndReserved::Next, BITSIZEOF(u8),  u8>;
            using NodeLow               = util::BitPack32::Field<ClockSeqLow::Next,           BITSIZEOF(u16), u16>;

            static_assert(ClockSeqHiAndReserved::Next == Reserved::Next);

            using NodeHigh              = util::BitPack32::Field<0,                           BITSIZEOF(u32), u32>;


            inline Uuid Convert() const {
                /* Convert the fields from native endian to big endian. */
                util::BitPack32 converted[4] = {util::BitPack32{0}, util::BitPack32{0}, util::BitPack32{0}, util::BitPack32{0}};

                converted[0].Set<TimeLow>(util::ConvertToBigEndian(this->data[0].Get<TimeLow>()));

                converted[1].Set<TimeMid>(util::ConvertToBigEndian(this->data[1].Get<TimeMid>()));
                converted[1].Set<TimeHighAndVersion>(util::ConvertToBigEndian(this->data[1].Get<TimeHighAndVersion>()));


                converted[2].Set<ClockSeqHiAndReserved>(util::ConvertToBigEndian(this->data[2].Get<ClockSeqHiAndReserved>()));
                converted[2].Set<ClockSeqLow>(util::ConvertToBigEndian(this->data[2].Get<ClockSeqLow>()));

                u64 node_lo = static_cast<u64>(this->data[2].Get<NodeLow>());
                u64 node_hi = static_cast<u64>(this->data[3].Get<NodeHigh>());
                u64 node    = util::ConvertToBigEndian48(static_cast<u64>((node_hi << BITSIZEOF(u16)) | (node_lo)));

                constexpr u64 NodeLoMask = (UINT64_C(1) << BITSIZEOF(u16)) - 1u;
                constexpr u64 NodeHiMask = (UINT64_C(1) << BITSIZEOF(u32)) - 1u;

                converted[2].Set<NodeLow>(static_cast<u16>(node & NodeLoMask));
                converted[3].Set<NodeHigh>(static_cast<u32>((node >> BITSIZEOF(u16)) & NodeHiMask));

                Uuid uuid;
                std::memcpy(uuid.data, converted, sizeof(uuid.data));
                return uuid;
            }
        };
        static_assert(sizeof(UuidImpl) == sizeof(Uuid));

        ALWAYS_INLINE Uuid GenerateUuidVersion4() {
            constexpr u16 Version  = 0x4;
            constexpr u8  Reserved = 0x1;

            /* Generate a random uuid. */
            UuidImpl uuid = {util::BitPack32{0}, util::BitPack32{0}, util::BitPack32{0}, util::BitPack32{0}};
            os::GenerateRandomBytes(uuid.data, sizeof(uuid.data));

            /* Set version and reserved. */
            uuid.data[1].Set<UuidImpl::Version>(Version);
            uuid.data[2].Set<UuidImpl::Reserved>(Reserved);

            /* Return the uuid. */
            return uuid.Convert();
        }

    }

    Uuid GenerateUuid() {
        return GenerateUuidVersion4();
    }

    Uuid GenerateUuidVersion5(const void *sha1_hash) {
        constexpr u16 Version  = 0x5;
        constexpr u8  Reserved = 0x1;

        /* Generate a uuid from a SHA1 hash. */
        UuidImpl uuid = {util::BitPack32{0}, util::BitPack32{0}, util::BitPack32{0}, util::BitPack32{0}};
        std::memcpy(uuid.data, sha1_hash, sizeof(uuid.data));

        /* Set version and reserved. */
        uuid.data[1].Set<UuidImpl::Version>(Version);
        uuid.data[2].Set<UuidImpl::Reserved>(Reserved);

        /* Return the uuid. */
        return uuid.Convert();
    }

}