// The Construct // // Copyright (C) The Construct Developers, Authors & Contributors // Copyright (C) 2016-2020 Jason Volk // // Permission to use, copy, modify, and/or distribute this software for any // purpose with or without fee is hereby granted, provided that the above // copyright notice and this permission notice is present in all copies. The // full license for this software is available in the LICENSE file. #pragma once #define HAVE_IRCD_SIMD_ROR_H // xmmx rotate-right interface namespace ircd::simd { template T _ror(const T) noexcept; template typename std::enable_if::type ror(const T a) noexcept; template typename std::enable_if::type ror(const T a) noexcept; template typename std::enable_if::type ror(const T a) noexcept; } template [[using gnu: always_inline, gnu_inline, artificial]] extern inline T ircd::simd::_ror(const T a) noexcept { static_assert ( sizeof(T) == sizeof(V) ); static_assert ( b % 8 == 0, "[emulated] xmmx register only rotates left at bytewise resolution." ); constexpr int B { b / 8 }; constexpr auto S { sizeof(V) - B }; size_t i(0); V arg(a), ret; for(; i < S; ++i) ret[i] = arg[i + B]; for(; i < sizeof(V); ++i) ret[i] = arg[i - S]; return T(ret); } template [[using gnu: always_inline, gnu_inline, artificial]] extern inline typename std::enable_if::type ircd::simd::ror(const T a) noexcept { return _ror(a); } template [[using gnu: always_inline, gnu_inline, artificial]] extern inline typename std::enable_if::type ircd::simd::ror(const T a) noexcept { return _ror(a); } template [[using gnu: always_inline, gnu_inline, artificial]] extern inline typename std::enable_if::type ircd::simd::ror(const T a) noexcept { return _ror(a); }