summaryrefslogtreecommitdiff
path: root/libipt/test/src/ptunit-config.c
diff options
context:
space:
mode:
Diffstat (limited to 'libipt/test/src/ptunit-config.c')
-rw-r--r--libipt/test/src/ptunit-config.c562
1 files changed, 562 insertions, 0 deletions
diff --git a/libipt/test/src/ptunit-config.c b/libipt/test/src/ptunit-config.c
new file mode 100644
index 0000000000000..1cf7dbffb0cfd
--- /dev/null
+++ b/libipt/test/src/ptunit-config.c
@@ -0,0 +1,562 @@
+/*
+ * 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);
+}