/*
* Copyright (c) 2015-2019, Intel Corporation
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are met:
*
* * Redistributions of source code must retain the above copyright notice,
* this list of conditions and the following disclaimer.
* * 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.
* * Neither the name of Intel Corporation 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 OWNER 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.
*/
#include "ptunit.h"
#include "pt_config.h"
#include "pt_opcodes.h"
#include "intel-pt.h"
#include <stddef.h>
/* A global fake buffer to pacify static analyzers. */
static uint8_t buffer[8];
static struct ptunit_result from_user_null(void)
{
struct pt_config config;
int errcode;
errcode = pt_config_from_user(NULL, &config);
ptu_int_eq(errcode, -pte_internal);
errcode = pt_config_from_user(&config, NULL);
ptu_int_eq(errcode, -pte_invalid);
return ptu_passed();
}
static struct ptunit_result from_user_too_small(void)
{
struct pt_config config, user;
int errcode;
user.size = sizeof(config.size);
errcode = pt_config_from_user(&config, &user);
ptu_int_eq(errcode, -pte_bad_config);
return ptu_passed();
}
static struct ptunit_result from_user_bad_buffer(void)
{
struct pt_config config, user;
int errcode;
pt_config_init(&user);
errcode = pt_config_from_user(&config, &user);
ptu_int_eq(errcode, -pte_bad_config);
user.begin = buffer;
errcode = pt_config_from_user(&config, &user);
ptu_int_eq(errcode, -pte_bad_config);
user.begin = NULL;
user.end = buffer;
errcode = pt_config_from_user(&config, &user);
ptu_int_eq(errcode, -pte_bad_config);
user.begin = &buffer[1];
user.end = buffer;
errcode = pt_config_from_user(&config, &user);
ptu_int_eq(errcode, -pte_bad_config);
return ptu_passed();
}
static struct ptunit_result from_user(void)
{
struct pt_config config, user;
int errcode;
user.size = sizeof(user);
user.begin = buffer;
user.end = &buffer[sizeof(buffer)];
user.cpu.vendor = pcv_intel;
user.errata.bdm70 = 1;
errcode = pt_config_from_user(&config, &user);
ptu_int_eq(errcode, 0);
ptu_uint_eq(config.size, sizeof(config));
ptu_ptr_eq(config.begin, buffer);
ptu_ptr_eq(config.end, &buffer[sizeof(buffer)]);
ptu_int_eq(config.cpu.vendor, pcv_intel);
ptu_uint_eq(config.errata.bdm70, 1);
return ptu_passed();
}
static struct ptunit_result from_user_small(void)
{
struct pt_config config, user;
int errcode;
memset(&config, 0xcd, sizeof(config));
user.size = offsetof(struct pt_config, cpu);
user.begin = buffer;
user.end = &buffer[sizeof(buffer)];
errcode = pt_config_from_user(&config, &user);
ptu_int_eq(errcode, 0);
ptu_uint_eq(config.size, offsetof(struct pt_config, cpu));
ptu_ptr_eq(config.begin, buffer);
ptu_ptr_eq(config.end, &buffer[sizeof(buffer)]);
ptu_int_eq(config.cpu.vendor, pcv_unknown);
ptu_uint_eq(config.errata.bdm70, 0);
return ptu_passed();
}
static struct ptunit_result from_user_big(void)
{
struct pt_config config, user;
int errcode;
user.size = sizeof(user) + 4;
user.begin = buffer;
user.end = &buffer[sizeof(buffer)];
user.cpu.vendor = pcv_intel;
user.errata.bdm70 = 1;
errcode = pt_config_from_user(&config, &user);
ptu_int_eq(errcode, 0);
ptu_uint_eq(config.size, sizeof(config));
ptu_ptr_eq(config.begin, buffer);
ptu_ptr_eq(config.end, &buffer[sizeof(buffer)]);
ptu_int_eq(config.cpu.vendor, pcv_intel);
ptu_uint_eq(config.errata.bdm70, 1);
return ptu_passed();
}
static struct ptunit_result size(void)
{
ptu_uint_eq(sizeof(struct pt_errata), 16 * 4);
return ptu_passed();
}
static struct ptunit_result addr_filter_size(void)
{
struct pt_conf_addr_filter conf;
ptu_uint_eq(sizeof(conf.config), 8);
return ptu_passed();
}
static struct ptunit_result addr_filter_none(void)
{
struct pt_config config;
uint8_t filter;
pt_config_init(&config);
ptu_uint_eq(config.addr_filter.config.addr_cfg, 0ull);
for (filter = 0; filter < 4; ++filter) {
uint32_t addr_cfg;
addr_cfg = pt_filter_addr_cfg(&config.addr_filter, filter);
ptu_uint_eq(addr_cfg, pt_addr_cfg_disabled);
}
return ptu_passed();
}
static struct ptunit_result addr_filter_0(void)
{
struct pt_config config;
uint64_t addr_a, addr_b;
uint32_t addr_cfg;
uint8_t filter;
pt_config_init(&config);
config.addr_filter.config.ctl.addr0_cfg = pt_addr_cfg_filter;
config.addr_filter.addr0_a = 0xa000ull;
config.addr_filter.addr0_b = 0xb000ull;
ptu_uint_ne(config.addr_filter.config.addr_cfg, 0ull);
addr_cfg = pt_filter_addr_cfg(&config.addr_filter, 0);
ptu_uint_eq(addr_cfg, pt_addr_cfg_filter);
addr_a = pt_filter_addr_a(&config.addr_filter, 0);
ptu_uint_eq(addr_a, 0xa000ull);
addr_b = pt_filter_addr_b(&config.addr_filter, 0);
ptu_uint_eq(addr_b, 0xb000ull);
for (filter = 1; filter < 4; ++filter) {
addr_cfg = pt_filter_addr_cfg(&config.addr_filter, filter);
ptu_uint_eq(addr_cfg, pt_addr_cfg_disabled);
}
return ptu_passed();
}
static struct ptunit_result addr_filter_1_3(void)
{
struct pt_config config;
uint64_t addr_a, addr_b;
uint32_t addr_cfg;
pt_config_init(&config);
config.addr_filter.config.ctl.addr1_cfg = pt_addr_cfg_filter;
config.addr_filter.addr1_a = 0xa000ull;
config.addr_filter.addr1_b = 0xb000ull;
config.addr_filter.config.ctl.addr3_cfg = pt_addr_cfg_stop;
config.addr_filter.addr3_a = 0x100a000ull;
config.addr_filter.addr3_b = 0x100b000ull;
ptu_uint_ne(config.addr_filter.config.addr_cfg, 0ull);
addr_cfg = pt_filter_addr_cfg(&config.addr_filter, 0);
ptu_uint_eq(addr_cfg, pt_addr_cfg_disabled);
addr_cfg = pt_filter_addr_cfg(&config.addr_filter, 1);
ptu_uint_eq(addr_cfg, pt_addr_cfg_filter);
addr_a = pt_filter_addr_a(&config.addr_filter, 1);
ptu_uint_eq(addr_a, 0xa000ull);
addr_b = pt_filter_addr_b(&config.addr_filter, 1);
ptu_uint_eq(addr_b, 0xb000ull);
addr_cfg = pt_filter_addr_cfg(&config.addr_filter, 2);
ptu_uint_eq(addr_cfg, pt_addr_cfg_disabled);
addr_cfg = pt_filter_addr_cfg(&config.addr_filter, 3);
ptu_uint_eq(addr_cfg, pt_addr_cfg_stop);
addr_a = pt_filter_addr_a(&config.addr_filter, 3);
ptu_uint_eq(addr_a, 0x100a000ull);
addr_b = pt_filter_addr_b(&config.addr_filter, 3);
ptu_uint_eq(addr_b, 0x100b000ull);
return ptu_passed();
}
static struct ptunit_result addr_filter_oob(uint8_t filter)
{
struct pt_config config;
uint64_t addr_a, addr_b;
uint32_t addr_cfg;
pt_config_init(&config);
memset(&config.addr_filter, 0xcc, sizeof(config.addr_filter));
addr_cfg = pt_filter_addr_cfg(&config.addr_filter, filter);
ptu_uint_eq(addr_cfg, pt_addr_cfg_disabled);
addr_a = pt_filter_addr_a(&config.addr_filter, filter);
ptu_uint_eq(addr_a, 0ull);
addr_b = pt_filter_addr_b(&config.addr_filter, filter);
ptu_uint_eq(addr_b, 0ull);
return ptu_passed();
}
static struct ptunit_result addr_filter_ip_in(void)
{
struct pt_config config;
int status;
pt_config_init(&config);
config.addr_filter.config.ctl.addr1_cfg = pt_addr_cfg_filter;
config.addr_filter.addr1_a = 0xa000;
config.addr_filter.addr1_b = 0xb000;
config.addr_filter.config.ctl.addr3_cfg = pt_addr_cfg_filter;
config.addr_filter.addr3_a = 0x10a000;
config.addr_filter.addr3_b = 0x10b000;
status = pt_filter_addr_check(&config.addr_filter, 0xa000);
ptu_int_eq(status, 1);
status = pt_filter_addr_check(&config.addr_filter, 0xaf00);
ptu_int_eq(status, 1);
status = pt_filter_addr_check(&config.addr_filter, 0xb000);
ptu_int_eq(status, 1);
status = pt_filter_addr_check(&config.addr_filter, 0x10a000);
ptu_int_eq(status, 1);
status = pt_filter_addr_check(&config.addr_filter, 0x10af00);
ptu_int_eq(status, 1);
status = pt_filter_addr_check(&config.addr_filter, 0x10b000);
ptu_int_eq(status, 1);
return ptu_passed();
}
static struct ptunit_result addr_filter_ip_out(void)
{
struct pt_config config;
int status;
pt_config_init(&config);
config.addr_filter.config.ctl.addr1_cfg = pt_addr_cfg_filter;
config.addr_filter.addr1_a = 0xa000;
config.addr_filter.addr1_b = 0xb000;
config.addr_filter.config.ctl.addr3_cfg = pt_addr_cfg_filter;
config.addr_filter.addr3_a = 0x10a000;
config.addr_filter.addr3_b = 0x10b000;
status = pt_filter_addr_check(&config.addr_filter, 0xfff);
ptu_int_eq(status, 0);
status = pt_filter_addr_check(&config.addr_filter, 0xb001);
ptu_int_eq(status, 0);
status = pt_filter_addr_check(&config.addr_filter, 0x100fff);
ptu_int_eq(status, 0);
status = pt_filter_addr_check(&config.addr_filter, 0x10b001);
ptu_int_eq(status, 0);
return ptu_passed();
}
static struct ptunit_result addr_filter_stop_in(void)
{
struct pt_config config;
int status;
pt_config_init(&config);
config.addr_filter.config.ctl.addr1_cfg = pt_addr_cfg_stop;
config.addr_filter.addr1_a = 0xa000;
config.addr_filter.addr1_b = 0xb000;
config.addr_filter.config.ctl.addr3_cfg = pt_addr_cfg_stop;
config.addr_filter.addr3_a = 0x10a000;
config.addr_filter.addr3_b = 0x10b000;
status = pt_filter_addr_check(&config.addr_filter, 0xa000);
ptu_int_eq(status, 0);
status = pt_filter_addr_check(&config.addr_filter, 0xaf00);
ptu_int_eq(status, 0);
status = pt_filter_addr_check(&config.addr_filter, 0xb000);
ptu_int_eq(status, 0);
status = pt_filter_addr_check(&config.addr_filter, 0x10a000);
ptu_int_eq(status, 0);
status = pt_filter_addr_check(&config.addr_filter, 0x10af00);
ptu_int_eq(status, 0);
status = pt_filter_addr_check(&config.addr_filter, 0x10b000);
ptu_int_eq(status, 0);
return ptu_passed();
}
static struct ptunit_result addr_filter_stop_out(void)
{
struct pt_config config;
int status;
pt_config_init(&config);
config.addr_filter.config.ctl.addr1_cfg = pt_addr_cfg_stop;
config.addr_filter.addr1_a = 0xa000;
config.addr_filter.addr1_b = 0xb000;
config.addr_filter.config.ctl.addr3_cfg = pt_addr_cfg_stop;
config.addr_filter.addr3_a = 0x10a000;
config.addr_filter.addr3_b = 0x10b000;
status = pt_filter_addr_check(&config.addr_filter, 0xfff);
ptu_int_eq(status, 1);
status = pt_filter_addr_check(&config.addr_filter, 0xb001);
ptu_int_eq(status, 1);
status = pt_filter_addr_check(&config.addr_filter, 0x100fff);
ptu_int_eq(status, 1);
status = pt_filter_addr_check(&config.addr_filter, 0x10b001);
ptu_int_eq(status, 1);
return ptu_passed();
}
static struct ptunit_result addr_filter_ip_out_stop_in(void)
{
struct pt_config config;
int status;
pt_config_init(&config);
config.addr_filter.config.ctl.addr1_cfg = pt_addr_cfg_filter;
config.addr_filter.addr1_a = 0x100f00;
config.addr_filter.addr1_b = 0x10af00;
config.addr_filter.config.ctl.addr3_cfg = pt_addr_cfg_stop;
config.addr_filter.addr3_a = 0x10a000;
config.addr_filter.addr3_b = 0x10b000;
status = pt_filter_addr_check(&config.addr_filter, 0x10af01);
ptu_int_eq(status, 0);
status = pt_filter_addr_check(&config.addr_filter, 0x10b000);
ptu_int_eq(status, 0);
return ptu_passed();
}
static struct ptunit_result addr_filter_ip_in_stop_in(void)
{
struct pt_config config;
int status;
pt_config_init(&config);
config.addr_filter.config.ctl.addr1_cfg = pt_addr_cfg_filter;
config.addr_filter.addr1_a = 0x100f00;
config.addr_filter.addr1_b = 0x10af00;
config.addr_filter.config.ctl.addr3_cfg = pt_addr_cfg_stop;
config.addr_filter.addr3_a = 0x10a000;
config.addr_filter.addr3_b = 0x10b000;
status = pt_filter_addr_check(&config.addr_filter, 0x10af00);
ptu_int_eq(status, 0);
status = pt_filter_addr_check(&config.addr_filter, 0x10a0ff);
ptu_int_eq(status, 0);
return ptu_passed();
}
static struct ptunit_result cpu_errata_null(void)
{
struct pt_errata errata;
struct pt_cpu cpu;
int errcode;
errcode = pt_cpu_errata(&errata, NULL);
ptu_int_eq(errcode, -pte_invalid);
errcode = pt_cpu_errata(NULL, &cpu);
ptu_int_eq(errcode, -pte_invalid);
return ptu_passed();
}
static struct ptunit_result cpu_errata_unknown(void)
{
struct pt_errata errata;
struct pt_cpu cpu;
int errcode;
memset(&cpu, 0, sizeof(cpu));
errcode = pt_cpu_errata(&errata, &cpu);
ptu_int_eq(errcode, -pte_bad_cpu);
return ptu_passed();
}
static struct ptunit_result cpu_errata_bad_vendor(void)
{
struct pt_errata errata;
struct pt_cpu cpu;
int errcode;
memset(&cpu, 0, sizeof(cpu));
cpu.vendor = (enum pt_cpu_vendor) 0xffff;
errcode = pt_cpu_errata(&errata, &cpu);
ptu_int_eq(errcode, -pte_bad_cpu);
return ptu_passed();
}
static struct ptunit_result cpu_errata_bad_cpuid(void)
{
struct pt_errata errata;
struct pt_cpu cpu;
int errcode;
memset(&cpu, 0, sizeof(cpu));
cpu.vendor = pcv_intel;
cpu.family = 6;
cpu.model = 63;
errcode = pt_cpu_errata(&errata, &cpu);
ptu_int_eq(errcode, -pte_bad_cpu);
return ptu_passed();
}
int main(int argc, char **argv)
{
struct ptunit_suite suite;
suite = ptunit_mk_suite(argc, argv);
ptu_run(suite, from_user_null);
ptu_run(suite, from_user_too_small);
ptu_run(suite, from_user_bad_buffer);
ptu_run(suite, from_user);
ptu_run(suite, from_user_small);
ptu_run(suite, from_user_big);
ptu_run(suite, size);
ptu_run(suite, addr_filter_size);
ptu_run(suite, addr_filter_none);
ptu_run(suite, addr_filter_0);
ptu_run(suite, addr_filter_1_3);
ptu_run_p(suite, addr_filter_oob, 255);
ptu_run_p(suite, addr_filter_oob, 8);
ptu_run(suite, addr_filter_ip_in);
ptu_run(suite, addr_filter_ip_out);
ptu_run(suite, addr_filter_stop_in);
ptu_run(suite, addr_filter_stop_out);
ptu_run(suite, addr_filter_ip_out_stop_in);
ptu_run(suite, addr_filter_ip_in_stop_in);
ptu_run(suite, cpu_errata_null);
ptu_run(suite, cpu_errata_unknown);
ptu_run(suite, cpu_errata_bad_vendor);
ptu_run(suite, cpu_errata_bad_cpuid);
return ptunit_report(&suite);
}