aboutsummaryrefslogtreecommitdiffstats
path: root/contrib/libs/apache/orc/c++/src/CpuInfoUtil.hh
blob: 5637053e6d60d6c6013697efd035d5e636bbbd81 (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
/**
 * Licensed to the Apache Software Foundation (ASF) under one
 * or more contributor license agreements.  See the NOTICE file
 * distributed with this work for additional information
 * regarding copyright ownership.  The ASF licenses this file
 * to you 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.
 */

/**
 * @file CpuInfoUtil.hh is from Apache Arrow as of 2023-03-21
 */

#ifndef ORC_CPUINFOUTIL_HH
#define ORC_CPUINFOUTIL_HH

#include <cstdint>
#include <memory>
#include <string>

namespace orc {

  /**
   * CpuInfo is an interface to query for cpu information at runtime.  The caller can
   * ask for the sizes of the caches and what hardware features are supported.
   * On Linux, this information is pulled from a couple of sys files (/proc/cpuinfo and
   * /sys/devices)
   */
  class CpuInfo {
   public:
    ~CpuInfo();

    // x86 features
    static constexpr int64_t SSSE3 = (1LL << 0);
    static constexpr int64_t SSE4_1 = (1LL << 1);
    static constexpr int64_t SSE4_2 = (1LL << 2);
    static constexpr int64_t POPCNT = (1LL << 3);
    static constexpr int64_t AVX = (1LL << 4);
    static constexpr int64_t AVX2 = (1LL << 5);
    static constexpr int64_t AVX512F = (1LL << 6);
    static constexpr int64_t AVX512CD = (1LL << 7);
    static constexpr int64_t AVX512VL = (1LL << 8);
    static constexpr int64_t AVX512DQ = (1LL << 9);
    static constexpr int64_t AVX512BW = (1LL << 10);
    static constexpr int64_t AVX512 = AVX512F | AVX512CD | AVX512VL | AVX512DQ | AVX512BW;
    static constexpr int64_t BMI1 = (1LL << 11);
    static constexpr int64_t BMI2 = (1LL << 12);

    /// Arm features
    static constexpr int64_t ASIMD = (1LL << 32);

    // Cache enums for L1 (data), L2 and L3
    enum class CacheLevel { L1 = 0, L2, L3, Last = L3 };

    // CPU vendors
    enum class Vendor { Unknown, Intel, AMD };

    static const CpuInfo* getInstance();

    // Returns all the flags for this cpu
    int64_t hardwareFlags() const;

    // Returns the number of cores (including hyper-threaded) on this machine.
    int numCores() const;

    // Returns the vendor of the cpu.
    Vendor vendor() const;

    // Returns the model name of the cpu (e.g. Intel i7-2600)
    const std::string& modelName() const;

    // Returns the size of the cache in KB at this cache level
    int64_t cacheSize(CacheLevel level) const;

    /**
     * Returns whether or not the given feature is enabled.
     * isSupported() is true if isDetected() is also true and the feature
     * wasn't disabled by the user (for example by setting the ORC_USER_SIMD_LEVEL
     * environment variable).
     */
    bool isSupported(int64_t flags) const;

    // Returns whether or not the given feature is available on the CPU.
    bool isDetected(int64_t flags) const;

    // Determine if the CPU meets the minimum CPU requirements and if not, issue an error
    // and terminate.
    void verifyCpuRequirements() const;

    bool hasEfficientBmi2() const {
      // BMI2 (pext, pdep) is only efficient on Intel X86 processors.
      return vendor() == Vendor::Intel && isSupported(BMI2);
    }

   private:
    CpuInfo();

    struct Impl;
    std::unique_ptr<Impl> impl_;
  };

}  // namespace orc

#endif