|
Packit |
b1f7ae |
/*
|
|
Packit |
b1f7ae |
* Copyright (c) 2013-2017, Intel Corporation
|
|
Packit |
b1f7ae |
*
|
|
Packit |
b1f7ae |
* Redistribution and use in source and binary forms, with or without
|
|
Packit |
b1f7ae |
* modification, are permitted provided that the following conditions are met:
|
|
Packit |
b1f7ae |
*
|
|
Packit |
b1f7ae |
* * Redistributions of source code must retain the above copyright notice,
|
|
Packit |
b1f7ae |
* this list of conditions and the following disclaimer.
|
|
Packit |
b1f7ae |
* * Redistributions in binary form must reproduce the above copyright notice,
|
|
Packit |
b1f7ae |
* this list of conditions and the following disclaimer in the documentation
|
|
Packit |
b1f7ae |
* and/or other materials provided with the distribution.
|
|
Packit |
b1f7ae |
* * Neither the name of Intel Corporation nor the names of its contributors
|
|
Packit |
b1f7ae |
* may be used to endorse or promote products derived from this software
|
|
Packit |
b1f7ae |
* without specific prior written permission.
|
|
Packit |
b1f7ae |
*
|
|
Packit |
b1f7ae |
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
|
|
Packit |
b1f7ae |
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
|
Packit |
b1f7ae |
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
|
|
Packit |
b1f7ae |
* ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE
|
|
Packit |
b1f7ae |
* LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
|
|
Packit |
b1f7ae |
* CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
|
|
Packit |
b1f7ae |
* SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
|
|
Packit |
b1f7ae |
* INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
|
|
Packit |
b1f7ae |
* CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
|
|
Packit |
b1f7ae |
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
|
|
Packit |
b1f7ae |
* POSSIBILITY OF SUCH DAMAGE.
|
|
Packit |
b1f7ae |
*/
|
|
Packit |
b1f7ae |
|
|
Packit |
b1f7ae |
#include "pt_cpuid.h"
|
|
Packit |
b1f7ae |
|
|
Packit |
b1f7ae |
#include "intel-pt.h"
|
|
Packit |
b1f7ae |
|
|
Packit |
b1f7ae |
#include <limits.h>
|
|
Packit |
b1f7ae |
#include <stdlib.h>
|
|
Packit |
b1f7ae |
|
|
Packit |
b1f7ae |
|
|
Packit |
b1f7ae |
const char *cpu_vendors[] = {
|
|
Packit |
b1f7ae |
"",
|
|
Packit |
b1f7ae |
"GenuineIntel"
|
|
Packit |
b1f7ae |
};
|
|
Packit |
b1f7ae |
|
|
Packit |
b1f7ae |
enum {
|
|
Packit |
b1f7ae |
pt_cpuid_vendor_size = 12
|
|
Packit |
b1f7ae |
};
|
|
Packit |
b1f7ae |
|
|
Packit |
b1f7ae |
union cpu_vendor {
|
|
Packit |
b1f7ae |
/* The raw data returned from cpuid. */
|
|
Packit |
b1f7ae |
struct {
|
|
Packit |
b1f7ae |
uint32_t ebx;
|
|
Packit |
b1f7ae |
uint32_t edx;
|
|
Packit |
b1f7ae |
uint32_t ecx;
|
|
Packit |
b1f7ae |
} cpuid;
|
|
Packit |
b1f7ae |
|
|
Packit |
b1f7ae |
/* The resulting vendor string. */
|
|
Packit |
b1f7ae |
char vendor_string[pt_cpuid_vendor_size];
|
|
Packit |
b1f7ae |
};
|
|
Packit |
b1f7ae |
|
|
Packit |
b1f7ae |
static enum pt_cpu_vendor cpu_vendor(void)
|
|
Packit |
b1f7ae |
{
|
|
Packit |
b1f7ae |
union cpu_vendor vendor;
|
|
Packit |
b1f7ae |
uint32_t eax;
|
|
Packit |
b1f7ae |
size_t i;
|
|
Packit |
b1f7ae |
|
|
Packit |
b1f7ae |
memset(&vendor, 0, sizeof(vendor));
|
|
Packit |
b1f7ae |
eax = 0;
|
|
Packit |
b1f7ae |
|
|
Packit |
b1f7ae |
pt_cpuid(0u, &eax, &vendor.cpuid.ebx, &vendor.cpuid.ecx,
|
|
Packit |
b1f7ae |
&vendor.cpuid.edx);
|
|
Packit |
b1f7ae |
|
|
Packit |
b1f7ae |
for (i = 0; i < sizeof(cpu_vendors)/sizeof(*cpu_vendors); i++)
|
|
Packit |
b1f7ae |
if (strncmp(vendor.vendor_string,
|
|
Packit |
b1f7ae |
cpu_vendors[i], pt_cpuid_vendor_size) == 0)
|
|
Packit |
b1f7ae |
return (enum pt_cpu_vendor) i;
|
|
Packit |
b1f7ae |
|
|
Packit |
b1f7ae |
return pcv_unknown;
|
|
Packit |
b1f7ae |
}
|
|
Packit |
b1f7ae |
|
|
Packit |
b1f7ae |
static uint32_t cpu_info(void)
|
|
Packit |
b1f7ae |
{
|
|
Packit |
b1f7ae |
uint32_t eax, ebx, ecx, edx;
|
|
Packit |
b1f7ae |
|
|
Packit |
b1f7ae |
eax = 0;
|
|
Packit |
b1f7ae |
ebx = 0;
|
|
Packit |
b1f7ae |
ecx = 0;
|
|
Packit |
b1f7ae |
edx = 0;
|
|
Packit |
b1f7ae |
pt_cpuid(1u, &eax, &ebx, &ecx, &edx;;
|
|
Packit |
b1f7ae |
|
|
Packit |
b1f7ae |
return eax;
|
|
Packit |
b1f7ae |
}
|
|
Packit |
b1f7ae |
|
|
Packit |
b1f7ae |
int pt_cpu_parse(struct pt_cpu *cpu, const char *s)
|
|
Packit |
b1f7ae |
{
|
|
Packit |
b1f7ae |
const char sep = '/';
|
|
Packit |
b1f7ae |
char *endptr;
|
|
Packit |
b1f7ae |
long family, model, stepping;
|
|
Packit |
b1f7ae |
|
|
Packit |
b1f7ae |
if (!cpu || !s)
|
|
Packit |
b1f7ae |
return -pte_invalid;
|
|
Packit |
b1f7ae |
|
|
Packit |
b1f7ae |
family = strtol(s, &endptr, 0);
|
|
Packit |
b1f7ae |
if (s == endptr || *endptr == '\0' || *endptr != sep)
|
|
Packit |
b1f7ae |
return -pte_invalid;
|
|
Packit |
b1f7ae |
|
|
Packit |
b1f7ae |
if (family < 0 || family > USHRT_MAX)
|
|
Packit |
b1f7ae |
return -pte_invalid;
|
|
Packit |
b1f7ae |
|
|
Packit |
b1f7ae |
/* skip separator */
|
|
Packit |
b1f7ae |
s = endptr + 1;
|
|
Packit |
b1f7ae |
|
|
Packit |
b1f7ae |
model = strtol(s, &endptr, 0);
|
|
Packit |
b1f7ae |
if (s == endptr || (*endptr != '\0' && *endptr != sep))
|
|
Packit |
b1f7ae |
return -pte_invalid;
|
|
Packit |
b1f7ae |
|
|
Packit |
b1f7ae |
if (model < 0 || model > UCHAR_MAX)
|
|
Packit |
b1f7ae |
return -pte_invalid;
|
|
Packit |
b1f7ae |
|
|
Packit |
b1f7ae |
if (*endptr == '\0')
|
|
Packit |
b1f7ae |
/* stepping was omitted, it defaults to 0 */
|
|
Packit |
b1f7ae |
stepping = 0;
|
|
Packit |
b1f7ae |
else {
|
|
Packit |
b1f7ae |
/* skip separator */
|
|
Packit |
b1f7ae |
s = endptr + 1;
|
|
Packit |
b1f7ae |
|
|
Packit |
b1f7ae |
stepping = strtol(s, &endptr, 0);
|
|
Packit |
b1f7ae |
if (*endptr != '\0')
|
|
Packit |
b1f7ae |
return -pte_invalid;
|
|
Packit |
b1f7ae |
|
|
Packit |
b1f7ae |
if (stepping < 0 || stepping > UCHAR_MAX)
|
|
Packit |
b1f7ae |
return -pte_invalid;
|
|
Packit |
b1f7ae |
}
|
|
Packit |
b1f7ae |
|
|
Packit |
b1f7ae |
cpu->vendor = pcv_intel;
|
|
Packit |
b1f7ae |
cpu->family = (uint16_t) family;
|
|
Packit |
b1f7ae |
cpu->model = (uint8_t) model;
|
|
Packit |
b1f7ae |
cpu->stepping = (uint8_t) stepping;
|
|
Packit |
b1f7ae |
|
|
Packit |
b1f7ae |
return 0;
|
|
Packit |
b1f7ae |
}
|
|
Packit |
b1f7ae |
|
|
Packit |
b1f7ae |
int pt_cpu_read(struct pt_cpu *cpu)
|
|
Packit |
b1f7ae |
{
|
|
Packit |
b1f7ae |
uint32_t info;
|
|
Packit |
b1f7ae |
uint16_t family;
|
|
Packit |
b1f7ae |
|
|
Packit |
b1f7ae |
if (!cpu)
|
|
Packit |
b1f7ae |
return -pte_invalid;
|
|
Packit |
b1f7ae |
|
|
Packit |
b1f7ae |
cpu->vendor = cpu_vendor();
|
|
Packit |
b1f7ae |
|
|
Packit |
b1f7ae |
info = cpu_info();
|
|
Packit |
b1f7ae |
|
|
Packit |
b1f7ae |
cpu->family = family = (info>>8) & 0xf;
|
|
Packit |
b1f7ae |
if (family == 0xf)
|
|
Packit |
b1f7ae |
cpu->family += (info>>20) & 0xf;
|
|
Packit |
b1f7ae |
|
|
Packit |
b1f7ae |
cpu->model = (info>>4) & 0xf;
|
|
Packit |
b1f7ae |
if (family == 0x6 || family == 0xf)
|
|
Packit |
b1f7ae |
cpu->model += (info>>12) & 0xf0;
|
|
Packit |
b1f7ae |
|
|
Packit |
b1f7ae |
cpu->stepping = (info>>0) & 0xf;
|
|
Packit |
b1f7ae |
|
|
Packit |
b1f7ae |
return 0;
|
|
Packit |
b1f7ae |
}
|