Skip to content

Commit

Permalink
simd: bootstrap the SIMD abstraction layer
Browse files Browse the repository at this point in the history
First, add some useful instruction set-specific definitions.

Two types:
- RegisterType: underlying representation of the future Register class.
- MaskType: same as RegisterType, but for the Mask class.

Three useful constants:
- INSTRUCTION_SET: useful for discrimination.
- ALIGNMENT: useful for memory allocation.
- REG_BITSZ: useful for iteration.

Refs: scality#222
  • Loading branch information
slaperche-scality committed Oct 15, 2018
1 parent f642316 commit 7c121f0
Show file tree
Hide file tree
Showing 2 changed files with 183 additions and 0 deletions.
124 changes: 124 additions & 0 deletions src/simd/definitions.h
Original file line number Diff line number Diff line change
@@ -0,0 +1,124 @@
/*
* Copyright 2017-2018 Scality
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are met:
*
* 1. Redistributions of source code must retain the above copyright notice,
* this list of conditions and the following disclaimer.
*
* 2. Redistributions in binary form must reproduce the above copyright notice,
* this list of conditions and the following disclaimer in the documentation
* and/or other materials provided with the distribution.
*
* 3. Neither the name of the copyright holder nor the names of its contributors
* may be used to endorse or promote products derived from this software
* without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
* ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
* LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
* CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
* SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
* INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
* CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
* POSSIBILITY OF SUCH DAMAGE.
*/

/** @file definitions.h
*
* Provide includes and constants that are platform-specific.
*/

#ifndef __QUAD_SIMD_SIMD_DEFINITIONS_H__
#define __QUAD_SIMD_SIMD_DEFINITIONS_H__

#include <climits>
#include <cstdint>

// SIMD-specific includes {{{

#ifdef QUADIRON_USE_SIMD

// SIMD for x86 and x86_64
#if defined(__i386__) || defined(__x86_64__)
#include <x86intrin.h>
#endif

// SIMD for ARM
#if defined(__ARM_NEON)
#include <arm_neon.h>
#endif

#endif

// }}}

namespace quadiron {
namespace simd {

/// Supported instruction set.
enum class InstructionSet {
NONE, ///< No SIMD instruction (fallback).
SSE, ///< SSE4.1
AVX, ///< AVX2
};

// Definitions for Intel AVX-256 {{{

// We required AVX2 because we relies on some instructions (such as
// `_mm256_add_epi16` and others) that aren't available in the first version of
// AVX.
#if defined(__AVX__) && defined(__AVX2__)

using RegisterType = __m256;
using MaskType = __m256i;

static constexpr InstructionSet INSTRUCTION_SET = InstructionSet::AVX;

// }}}
// Definitions for Intel SSE {{{

#elif defined(__SSE__) && defined(__SSE4_1__)

using RegisterType = __m128;
using MaskType = __m128i;

static constexpr InstructionSet INSTRUCTION_SET = InstructionSet::SSE;

// }}}
// Definitions for scalar fallback {{{

#else

// Use The Widest Integer Type Available.
#if __LP64__ == 1
using RegisterType = uint64_t;
using MaskType = uint64_t;
#else
using RegisterType = uint32_t;
using MaskType = uint32_t;
#endif

static constexpr InstructionSet INSTRUCTION_SET = InstructionSet::NONE;

#endif

// }}}
// Portable definitions {{{

/// Alignment constraint (in bytes).
static constexpr std::size_t ALIGNMENT = alignof(RegisterType);

/// Register size (in bits).
static constexpr std::size_t REG_BITSZ = sizeof(RegisterType) * CHAR_BIT;

// }}}

} // namespace simd
} // namespace quadiron

#endif
59 changes: 59 additions & 0 deletions src/simd/simd.h
Original file line number Diff line number Diff line change
@@ -0,0 +1,59 @@
/*
* Copyright 2017-2018 Scality
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are met:
*
* 1. Redistributions of source code must retain the above copyright notice,
* this list of conditions and the following disclaimer.
*
* 2. Redistributions in binary form must reproduce the above copyright notice,
* this list of conditions and the following disclaimer in the documentation
* and/or other materials provided with the distribution.
*
* 3. Neither the name of the copyright holder nor the names of its contributors
* may be used to endorse or promote products derived from this software
* without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
* ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
* LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
* CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
* SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
* INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
* CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
* POSSIBILITY OF SUCH DAMAGE.
*/

#ifndef __QUAD_SIMD_SIMD_H__
#define __QUAD_SIMD_SIMD_H__

#include "simd/definitions.h"

namespace quadiron {

/** SIMD abstraction layer.
*
* This module provides a class, Register, that allows to write code that
* leverage SIMD acceleration regardless of the supported instruction set.
*/
namespace simd {

/// Return the number of element of type T that can fit into a SIMD register.
template <typename T>
static constexpr std::size_t countof()
{
// Without SIMD, there is no parallelism.
if (INSTRUCTION_SET == InstructionSet::NONE) {
return 1;
}
return REG_BITSZ / (sizeof(T) * CHAR_BIT);
}

} // namespace simd
} // namespace quadiron

#endif

0 comments on commit 7c121f0

Please sign in to comment.