1428d7b3dSmrg/* 2428d7b3dSmrg * Copyright (c) 2013 Intel Corporation 3428d7b3dSmrg * 4428d7b3dSmrg * Permission is hereby granted, free of charge, to any person obtaining a 5428d7b3dSmrg * copy of this software and associated documentation files (the "Software"), 6428d7b3dSmrg * to deal in the Software without restriction, including without limitation 7428d7b3dSmrg * the rights to use, copy, modify, merge, publish, distribute, sublicense, 8428d7b3dSmrg * and/or sell copies of the Software, and to permit persons to whom the 9428d7b3dSmrg * Software is furnished to do so, subject to the following conditions: 10428d7b3dSmrg * 11428d7b3dSmrg * The above copyright notice and this permission notice (including the next 12428d7b3dSmrg * paragraph) shall be included in all copies or substantial portions of the 13428d7b3dSmrg * Software. 14428d7b3dSmrg * 15428d7b3dSmrg * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR 16428d7b3dSmrg * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, 17428d7b3dSmrg * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL 18428d7b3dSmrg * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER 19428d7b3dSmrg * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, 20428d7b3dSmrg * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE 21428d7b3dSmrg * SOFTWARE. 22428d7b3dSmrg * 23428d7b3dSmrg * Authors: 24428d7b3dSmrg * Chris Wilson <chris@chris-wilson.co.uk> 25428d7b3dSmrg * 26428d7b3dSmrg */ 27428d7b3dSmrg 28428d7b3dSmrg#ifdef HAVE_CONFIG_H 29428d7b3dSmrg#include "config.h" 30428d7b3dSmrg#endif 31428d7b3dSmrg 32428d7b3dSmrg#include "sna.h" 33428d7b3dSmrg#include "sna_cpuid.h" 34428d7b3dSmrg 35428d7b3dSmrg#define xgetbv(index,eax,edx) \ 36428d7b3dSmrg __asm__ ("xgetbv" : "=a"(eax), "=d"(edx) : "c" (index)) 37428d7b3dSmrg 38428d7b3dSmrg#define has_YMM 0x1 39428d7b3dSmrg 40428d7b3dSmrgunsigned sna_cpu_detect(void) 41428d7b3dSmrg{ 42428d7b3dSmrg unsigned max = __get_cpuid_max(BASIC_CPUID, NULL); 43428d7b3dSmrg unsigned eax, ebx, ecx, edx; 44428d7b3dSmrg unsigned features = 0; 45428d7b3dSmrg unsigned extra = 0; 46428d7b3dSmrg 47428d7b3dSmrg if (max >= 1) { 48428d7b3dSmrg __cpuid(1, eax, ebx, ecx, edx); 49428d7b3dSmrg 50428d7b3dSmrg if (ecx & bit_SSE3) 51428d7b3dSmrg features |= SSE3; 52428d7b3dSmrg 53428d7b3dSmrg if (ecx & bit_SSSE3) 54428d7b3dSmrg features |= SSSE3; 55428d7b3dSmrg 56428d7b3dSmrg if (ecx & bit_SSE4_1) 57428d7b3dSmrg features |= SSE4_1; 58428d7b3dSmrg 59428d7b3dSmrg if (ecx & bit_SSE4_2) 60428d7b3dSmrg features |= SSE4_2; 61428d7b3dSmrg 62428d7b3dSmrg if (ecx & bit_OSXSAVE) { 63428d7b3dSmrg unsigned int bv_eax, bv_ecx; 64428d7b3dSmrg xgetbv(0, bv_eax, bv_ecx); 65428d7b3dSmrg if ((bv_eax & 6) == 6) 66428d7b3dSmrg extra |= has_YMM; 67428d7b3dSmrg } 68428d7b3dSmrg 69428d7b3dSmrg if ((extra & has_YMM) && (ecx & bit_AVX)) 70428d7b3dSmrg features |= AVX; 71428d7b3dSmrg 72428d7b3dSmrg if (edx & bit_MMX) 73428d7b3dSmrg features |= MMX; 74428d7b3dSmrg 75428d7b3dSmrg if (edx & bit_SSE) 76428d7b3dSmrg features |= SSE; 77428d7b3dSmrg 78428d7b3dSmrg if (edx & bit_SSE2) 79428d7b3dSmrg features |= SSE2; 80428d7b3dSmrg } 81428d7b3dSmrg 82428d7b3dSmrg if (max >= 7) { 83428d7b3dSmrg __cpuid_count(7, 0, eax, ebx, ecx, edx); 84428d7b3dSmrg 85428d7b3dSmrg if ((extra & has_YMM) && (ebx & bit_AVX2)) 86428d7b3dSmrg features |= AVX2; 87428d7b3dSmrg } 88428d7b3dSmrg 89428d7b3dSmrg return features; 90428d7b3dSmrg} 91428d7b3dSmrg 92428d7b3dSmrgchar *sna_cpu_features_to_string(unsigned features, char *line) 93428d7b3dSmrg{ 94428d7b3dSmrg char *ret = line; 95428d7b3dSmrg 96428d7b3dSmrg#ifdef __x86_64__ 97428d7b3dSmrg line += sprintf (line, "x86-64"); 98428d7b3dSmrg#else 99428d7b3dSmrg line += sprintf (line, "x86"); 100428d7b3dSmrg#endif 101428d7b3dSmrg 102428d7b3dSmrg if (features & SSE2) 103428d7b3dSmrg line += sprintf (line, ", sse2"); 104428d7b3dSmrg if (features & SSE3) 105428d7b3dSmrg line += sprintf (line, ", sse3"); 106428d7b3dSmrg if (features & SSSE3) 107428d7b3dSmrg line += sprintf (line, ", ssse3"); 108428d7b3dSmrg if (features & SSE4_1) 109428d7b3dSmrg line += sprintf (line, ", sse4.1"); 110428d7b3dSmrg if (features & SSE4_2) 111428d7b3dSmrg line += sprintf (line, ", sse4.2"); 112428d7b3dSmrg if (features & AVX) 113428d7b3dSmrg line += sprintf (line, ", avx"); 114428d7b3dSmrg if (features & AVX2) 115428d7b3dSmrg line += sprintf (line, ", avx2"); 116428d7b3dSmrg 117428d7b3dSmrg return ret; 118428d7b3dSmrg} 119