Blame libipt/test/src/ptunit-time.c

Packit b1f7ae
/*
Packit b1f7ae
 * Copyright (c) 2014-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_time.h"
Packit b1f7ae
Packit b1f7ae
#include "intel-pt.h"
Packit b1f7ae
Packit b1f7ae
#include "ptunit.h"
Packit b1f7ae
Packit b1f7ae
Packit b1f7ae
/* A time unit test fixture. */
Packit b1f7ae
Packit b1f7ae
struct time_fixture {
Packit b1f7ae
	/* The configuration to use. */
Packit b1f7ae
	struct pt_config config;
Packit b1f7ae
Packit b1f7ae
	/* The calibration to use. */
Packit b1f7ae
	struct pt_time_cal tcal;
Packit b1f7ae
Packit b1f7ae
	/* The time struct to update. */
Packit b1f7ae
	struct pt_time time;
Packit b1f7ae
Packit b1f7ae
	/* The test fixture initialization and finalization functions. */
Packit b1f7ae
	struct ptunit_result (*init)(struct time_fixture *);
Packit b1f7ae
	struct ptunit_result (*fini)(struct time_fixture *);
Packit b1f7ae
};
Packit b1f7ae
Packit b1f7ae
static struct ptunit_result tfix_init(struct time_fixture *tfix)
Packit b1f7ae
{
Packit b1f7ae
	memset(&tfix->config, 0, sizeof(tfix->config));
Packit b1f7ae
	tfix->config.size = sizeof(tfix->config);
Packit b1f7ae
	tfix->config.cpuid_0x15_eax = 2;
Packit b1f7ae
	tfix->config.cpuid_0x15_ebx = 1;
Packit b1f7ae
	tfix->config.mtc_freq = 4;
Packit b1f7ae
Packit b1f7ae
	pt_tcal_init(&tfix->tcal);
Packit b1f7ae
	pt_tcal_set_fcr(&tfix->tcal, 0x2ull << pt_tcal_fcr_shr);
Packit b1f7ae
Packit b1f7ae
	pt_time_init(&tfix->time);
Packit b1f7ae
Packit b1f7ae
	return ptu_passed();
Packit b1f7ae
}
Packit b1f7ae
Packit b1f7ae
Packit b1f7ae
static struct ptunit_result tsc_null(struct time_fixture *tfix)
Packit b1f7ae
{
Packit b1f7ae
	struct pt_packet_tsc packet;
Packit b1f7ae
	int errcode;
Packit b1f7ae
Packit b1f7ae
	errcode = pt_time_update_tsc(NULL, &packet, &tfix->config);
Packit b1f7ae
	ptu_int_eq(errcode, -pte_internal);
Packit b1f7ae
Packit b1f7ae
	errcode = pt_time_update_tsc(&tfix->time, NULL, &tfix->config);
Packit b1f7ae
	ptu_int_eq(errcode, -pte_internal);
Packit b1f7ae
Packit b1f7ae
	return ptu_passed();
Packit b1f7ae
}
Packit b1f7ae
Packit b1f7ae
static struct ptunit_result cbr_null(struct time_fixture *tfix)
Packit b1f7ae
{
Packit b1f7ae
	struct pt_packet_cbr packet;
Packit b1f7ae
	int errcode;
Packit b1f7ae
Packit b1f7ae
	errcode = pt_time_update_cbr(NULL, &packet, &tfix->config);
Packit b1f7ae
	ptu_int_eq(errcode, -pte_internal);
Packit b1f7ae
Packit b1f7ae
	errcode = pt_time_update_cbr(&tfix->time, NULL, &tfix->config);
Packit b1f7ae
	ptu_int_eq(errcode, -pte_internal);
Packit b1f7ae
Packit b1f7ae
	return ptu_passed();
Packit b1f7ae
}
Packit b1f7ae
Packit b1f7ae
static struct ptunit_result tma_null(struct time_fixture *tfix)
Packit b1f7ae
{
Packit b1f7ae
	struct pt_packet_tma packet;
Packit b1f7ae
	int errcode;
Packit b1f7ae
Packit b1f7ae
	errcode = pt_time_update_tma(NULL, &packet, &tfix->config);
Packit b1f7ae
	ptu_int_eq(errcode, -pte_internal);
Packit b1f7ae
Packit b1f7ae
	errcode = pt_time_update_tma(&tfix->time, NULL, &tfix->config);
Packit b1f7ae
	ptu_int_eq(errcode, -pte_internal);
Packit b1f7ae
Packit b1f7ae
	errcode = pt_time_update_tma(&tfix->time, &packet, NULL);
Packit b1f7ae
	ptu_int_eq(errcode, -pte_internal);
Packit b1f7ae
Packit b1f7ae
	return ptu_passed();
Packit b1f7ae
}
Packit b1f7ae
Packit b1f7ae
static struct ptunit_result mtc_null(struct time_fixture *tfix)
Packit b1f7ae
{
Packit b1f7ae
	struct pt_packet_mtc packet;
Packit b1f7ae
	int errcode;
Packit b1f7ae
Packit b1f7ae
	errcode = pt_time_update_mtc(NULL, &packet, &tfix->config);
Packit b1f7ae
	ptu_int_eq(errcode, -pte_internal);
Packit b1f7ae
Packit b1f7ae
	errcode = pt_time_update_mtc(&tfix->time, NULL, &tfix->config);
Packit b1f7ae
	ptu_int_eq(errcode, -pte_internal);
Packit b1f7ae
Packit b1f7ae
	errcode = pt_time_update_mtc(&tfix->time, &packet, NULL);
Packit b1f7ae
	ptu_int_eq(errcode, -pte_internal);
Packit b1f7ae
Packit b1f7ae
	return ptu_passed();
Packit b1f7ae
}
Packit b1f7ae
Packit b1f7ae
static struct ptunit_result cyc_null(struct time_fixture *tfix)
Packit b1f7ae
{
Packit b1f7ae
	struct pt_packet_cyc packet;
Packit b1f7ae
	int errcode;
Packit b1f7ae
Packit b1f7ae
	errcode = pt_time_update_cyc(NULL, &packet, &tfix->config, 0ull);
Packit b1f7ae
	ptu_int_eq(errcode, -pte_internal);
Packit b1f7ae
Packit b1f7ae
	errcode = pt_time_update_cyc(&tfix->time, NULL, &tfix->config, 0ull);
Packit b1f7ae
	ptu_int_eq(errcode, -pte_internal);
Packit b1f7ae
Packit b1f7ae
	errcode = pt_time_update_cyc(&tfix->time, &packet, NULL, 0ull);
Packit b1f7ae
	ptu_int_eq(errcode, -pte_internal);
Packit b1f7ae
Packit b1f7ae
	return ptu_passed();
Packit b1f7ae
}
Packit b1f7ae
Packit b1f7ae
static struct ptunit_result query_tsc_null(struct time_fixture *tfix)
Packit b1f7ae
{
Packit b1f7ae
	uint64_t tsc;
Packit b1f7ae
	int errcode;
Packit b1f7ae
Packit b1f7ae
	errcode = pt_time_query_tsc(NULL, NULL, NULL, &tfix->time);
Packit b1f7ae
	ptu_int_eq(errcode, -pte_internal);
Packit b1f7ae
Packit b1f7ae
	errcode = pt_time_query_tsc(&tsc, NULL, NULL, NULL);
Packit b1f7ae
	ptu_int_eq(errcode, -pte_internal);
Packit b1f7ae
Packit b1f7ae
	return ptu_passed();
Packit b1f7ae
}
Packit b1f7ae
Packit b1f7ae
static struct ptunit_result query_tsc_none(struct time_fixture *tfix)
Packit b1f7ae
{
Packit b1f7ae
	uint64_t tsc;
Packit b1f7ae
	int errcode;
Packit b1f7ae
Packit b1f7ae
	errcode = pt_time_query_tsc(&tsc, NULL, NULL, &tfix->time);
Packit b1f7ae
	ptu_int_eq(errcode, -pte_no_time);
Packit b1f7ae
Packit b1f7ae
	return ptu_passed();
Packit b1f7ae
}
Packit b1f7ae
Packit b1f7ae
static struct ptunit_result query_cbr_null(struct time_fixture *tfix)
Packit b1f7ae
{
Packit b1f7ae
	uint32_t cbr;
Packit b1f7ae
	int errcode;
Packit b1f7ae
Packit b1f7ae
	errcode = pt_time_query_cbr(NULL, &tfix->time);
Packit b1f7ae
	ptu_int_eq(errcode, -pte_internal);
Packit b1f7ae
Packit b1f7ae
	errcode = pt_time_query_cbr(&cbr, NULL);
Packit b1f7ae
	ptu_int_eq(errcode, -pte_internal);
Packit b1f7ae
Packit b1f7ae
	return ptu_passed();
Packit b1f7ae
}
Packit b1f7ae
Packit b1f7ae
static struct ptunit_result query_cbr_none(struct time_fixture *tfix)
Packit b1f7ae
{
Packit b1f7ae
	uint32_t cbr;
Packit b1f7ae
	int errcode;
Packit b1f7ae
Packit b1f7ae
	errcode = pt_time_query_cbr(&cbr, &tfix->time);
Packit b1f7ae
	ptu_int_eq(errcode, -pte_no_cbr);
Packit b1f7ae
Packit b1f7ae
	return ptu_passed();
Packit b1f7ae
}
Packit b1f7ae
Packit b1f7ae
static struct ptunit_result tcal_cbr_null(struct time_fixture *tfix)
Packit b1f7ae
{
Packit b1f7ae
	struct pt_packet_cbr packet;
Packit b1f7ae
	int errcode;
Packit b1f7ae
Packit b1f7ae
	errcode = pt_tcal_update_cbr(NULL, &packet, &tfix->config);
Packit b1f7ae
	ptu_int_eq(errcode, -pte_internal);
Packit b1f7ae
Packit b1f7ae
	return ptu_passed();
Packit b1f7ae
}
Packit b1f7ae
Packit b1f7ae
static struct ptunit_result tcal_mtc_null(struct time_fixture *tfix)
Packit b1f7ae
{
Packit b1f7ae
	struct pt_packet_mtc packet;
Packit b1f7ae
	int errcode;
Packit b1f7ae
Packit b1f7ae
	errcode = pt_tcal_update_mtc(NULL, &packet, &tfix->config);
Packit b1f7ae
	ptu_int_eq(errcode, -pte_internal);
Packit b1f7ae
Packit b1f7ae
	errcode = pt_tcal_update_mtc(&tfix->tcal, NULL, &tfix->config);
Packit b1f7ae
	ptu_int_eq(errcode, -pte_internal);
Packit b1f7ae
Packit b1f7ae
	errcode = pt_tcal_update_mtc(&tfix->tcal, &packet, NULL);
Packit b1f7ae
	ptu_int_eq(errcode, -pte_internal);
Packit b1f7ae
Packit b1f7ae
	return ptu_passed();
Packit b1f7ae
}
Packit b1f7ae
Packit b1f7ae
static struct ptunit_result tcal_cyc_null(struct time_fixture *tfix)
Packit b1f7ae
{
Packit b1f7ae
	struct pt_packet_cyc packet;
Packit b1f7ae
	int errcode;
Packit b1f7ae
Packit b1f7ae
	errcode = pt_tcal_update_cyc(NULL, &packet, &tfix->config);
Packit b1f7ae
	ptu_int_eq(errcode, -pte_internal);
Packit b1f7ae
Packit b1f7ae
	errcode = pt_tcal_update_cyc(&tfix->tcal, NULL, &tfix->config);
Packit b1f7ae
	ptu_int_eq(errcode, -pte_internal);
Packit b1f7ae
Packit b1f7ae
	return ptu_passed();
Packit b1f7ae
}
Packit b1f7ae
Packit b1f7ae
static struct ptunit_result tsc(struct time_fixture *tfix)
Packit b1f7ae
{
Packit b1f7ae
	struct pt_packet_tsc packet;
Packit b1f7ae
	uint64_t tsc;
Packit b1f7ae
	uint32_t lost_mtc, lost_cyc;
Packit b1f7ae
	int errcode;
Packit b1f7ae
Packit b1f7ae
	packet.tsc = 0xdedededeull;
Packit b1f7ae
Packit b1f7ae
	errcode = pt_time_update_tsc(&tfix->time, &packet, &tfix->config);
Packit b1f7ae
	ptu_int_eq(errcode, 0);
Packit b1f7ae
Packit b1f7ae
	errcode = pt_time_query_tsc(&tsc, &lost_mtc, &lost_cyc, &tfix->time);
Packit b1f7ae
	ptu_int_eq(errcode, 0);
Packit b1f7ae
Packit b1f7ae
	ptu_uint_eq(tsc, 0xdedededeull);
Packit b1f7ae
	ptu_uint_eq(lost_mtc, 0);
Packit b1f7ae
	ptu_uint_eq(lost_cyc, 0);
Packit b1f7ae
Packit b1f7ae
	return ptu_passed();
Packit b1f7ae
}
Packit b1f7ae
Packit b1f7ae
static struct ptunit_result cbr(struct time_fixture *tfix)
Packit b1f7ae
{
Packit b1f7ae
	struct pt_packet_cbr packet;
Packit b1f7ae
	uint32_t cbr;
Packit b1f7ae
	int errcode;
Packit b1f7ae
Packit b1f7ae
	packet.ratio = 0x38;
Packit b1f7ae
Packit b1f7ae
	errcode = pt_time_update_cbr(&tfix->time, &packet, &tfix->config);
Packit b1f7ae
	ptu_int_eq(errcode, 0);
Packit b1f7ae
Packit b1f7ae
	errcode = pt_time_query_cbr(&cbr, &tfix->time);
Packit b1f7ae
	ptu_int_eq(errcode, 0);
Packit b1f7ae
Packit b1f7ae
	ptu_uint_eq(cbr, 0x38);
Packit b1f7ae
Packit b1f7ae
	return ptu_passed();
Packit b1f7ae
}
Packit b1f7ae
Packit b1f7ae
static struct ptunit_result tma(struct time_fixture *tfix)
Packit b1f7ae
{
Packit b1f7ae
	struct pt_packet_tma packet;
Packit b1f7ae
	int errcode;
Packit b1f7ae
Packit b1f7ae
	packet.ctc = 0xdc;
Packit b1f7ae
	packet.fc = 0xf;
Packit b1f7ae
Packit b1f7ae
	errcode = pt_time_update_tma(&tfix->time, &packet, &tfix->config);
Packit b1f7ae
	ptu_int_eq(errcode, -pte_bad_context);
Packit b1f7ae
Packit b1f7ae
	return ptu_passed();
Packit b1f7ae
}
Packit b1f7ae
Packit b1f7ae
static struct ptunit_result mtc(struct time_fixture *tfix)
Packit b1f7ae
{
Packit b1f7ae
	struct pt_packet_mtc packet;
Packit b1f7ae
	uint64_t tsc;
Packit b1f7ae
	int errcode;
Packit b1f7ae
Packit b1f7ae
	packet.ctc = 0xdc;
Packit b1f7ae
Packit b1f7ae
	errcode = pt_time_update_mtc(&tfix->time, &packet, &tfix->config);
Packit b1f7ae
	ptu_int_eq(errcode, 0);
Packit b1f7ae
Packit b1f7ae
	errcode = pt_time_query_tsc(&tsc, NULL, NULL, &tfix->time);
Packit b1f7ae
	ptu_int_eq(errcode, -pte_no_time);
Packit b1f7ae
Packit b1f7ae
	return ptu_passed();
Packit b1f7ae
}
Packit b1f7ae
Packit b1f7ae
static struct ptunit_result cyc(struct time_fixture *tfix)
Packit b1f7ae
{
Packit b1f7ae
	struct pt_packet_cyc packet;
Packit b1f7ae
	uint64_t fcr, tsc;
Packit b1f7ae
	int errcode;
Packit b1f7ae
Packit b1f7ae
	errcode = pt_tcal_fcr(&fcr, &tfix->tcal);
Packit b1f7ae
	ptu_int_eq(errcode, 0);
Packit b1f7ae
Packit b1f7ae
	packet.value = 0xdc;
Packit b1f7ae
Packit b1f7ae
	errcode = pt_time_update_cyc(&tfix->time, &packet, &tfix->config, fcr);
Packit b1f7ae
	ptu_int_eq(errcode, 0);
Packit b1f7ae
Packit b1f7ae
	errcode = pt_time_query_tsc(&tsc, NULL, NULL, &tfix->time);
Packit b1f7ae
	ptu_int_eq(errcode, -pte_no_time);
Packit b1f7ae
Packit b1f7ae
	return ptu_passed();
Packit b1f7ae
}
Packit b1f7ae
Packit b1f7ae
Packit b1f7ae
int main(int argc, char **argv)
Packit b1f7ae
{
Packit b1f7ae
	struct ptunit_suite suite;
Packit b1f7ae
	struct time_fixture tfix;
Packit b1f7ae
Packit b1f7ae
	suite = ptunit_mk_suite(argc, argv);
Packit b1f7ae
Packit b1f7ae
	tfix.init = tfix_init;
Packit b1f7ae
	tfix.fini = NULL;
Packit b1f7ae
Packit b1f7ae
	ptu_run_f(suite, tsc_null, tfix);
Packit b1f7ae
	ptu_run_f(suite, cbr_null, tfix);
Packit b1f7ae
	ptu_run_f(suite, tma_null, tfix);
Packit b1f7ae
	ptu_run_f(suite, mtc_null, tfix);
Packit b1f7ae
	ptu_run_f(suite, cyc_null, tfix);
Packit b1f7ae
Packit b1f7ae
	ptu_run_f(suite, query_tsc_null, tfix);
Packit b1f7ae
	ptu_run_f(suite, query_tsc_none, tfix);
Packit b1f7ae
	ptu_run_f(suite, query_cbr_null, tfix);
Packit b1f7ae
	ptu_run_f(suite, query_cbr_none, tfix);
Packit b1f7ae
Packit b1f7ae
	ptu_run_f(suite, tcal_cbr_null, tfix);
Packit b1f7ae
	ptu_run_f(suite, tcal_mtc_null, tfix);
Packit b1f7ae
	ptu_run_f(suite, tcal_cyc_null, tfix);
Packit b1f7ae
Packit b1f7ae
	ptu_run_f(suite, tsc, tfix);
Packit b1f7ae
	ptu_run_f(suite, cbr, tfix);
Packit b1f7ae
	ptu_run_f(suite, tma, tfix);
Packit b1f7ae
	ptu_run_f(suite, mtc, tfix);
Packit b1f7ae
	ptu_run_f(suite, cyc, tfix);
Packit b1f7ae
Packit b1f7ae
	/* The bulk is covered in ptt tests. */
Packit b1f7ae
Packit b1f7ae
	ptunit_report(&suite);
Packit b1f7ae
	return suite.nr_fails;
Packit b1f7ae
}