/* * Valgrind testcase for PowerPC ISA 3.1 * * Copyright (C) 2019-2020 Will Schmidt * * 64bit build: * gcc -Winline -Wall -g -O -mregnames -maltivec -m64 */ /* * This program is free software; you can redistribute it and/or * modify it under the terms of the GNU General Public License as * published by the Free Software Foundation; either version 2 of the * License, or (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ #include #ifdef HAS_ISA_3_1 #include #include #include #include #include #include #include #include #include #include /* Condition Register fields. These are used to capture the condition register values immediately after the instruction under test is executed. This is done to help prevent other test overhead (switch statements, result compares, etc) from disturbing the test case results. */ unsigned long current_cr; unsigned long current_fpscr; struct test_list_t current_test; #include "isa_3_1_helpers.h" static void test_vcmpsq (void) { SET_CR_ZERO; __asm__ __volatile__ ("vcmpsq 3, %0, %1" :: "v" (vra), "v" (vrb) ); GET_CR(current_cr); SET_CR_ZERO; } static void test_vcmpuq (void) { SET_CR_ZERO; __asm__ __volatile__ ("vcmpuq 3, %0, %1" :: "v" (vra), "v" (vrb) ); GET_CR(current_cr); SET_CR_ZERO; } static void test_xvtlsbb (void) { SET_CR_ZERO; __asm__ __volatile__ ("xvtlsbb 3, %x0" :: "wa" (vec_xb) ); GET_CR(current_cr); SET_CR_ZERO; } static test_list_t testgroup_generic[] = { { &test_vcmpsq, "vcmpsq", "BF,VRA,VRB"}, /* bcs */ { &test_vcmpuq, "vcmpuq", "BF,VRA,VRB"}, /* bcs */ { &test_xvtlsbb, "xvtlsbb", "BF,XB"}, /* bcs */ { NULL, NULL }, }; /* Allow skipping of tests. */ unsigned long test_count=0xffff; unsigned long skip_count=0; unsigned long setup_only=0; /* Set up a setjmp/longjmp to gently handle our SIGILLs and SIGSEGVs. */ static jmp_buf mybuf; /* This (testfunction_generic) is meant to handle all of the instruction variations. The helpers set up the register and iterator values as is appropriate for the instruction being tested. */ static void testfunction_generic (const char* instruction_name, test_func_t test_function, unsigned int ignore_flags, char * cur_form) { identify_form_components (instruction_name , cur_form); debug_show_form (instruction_name, cur_form); set_up_iterators (); debug_show_iter_ranges (); initialize_buffer (0); debug_dump_buffer (); for (vrai = a_start; vrai < a_iters ; vrai+=a_inc) { for (vrbi = b_start; vrbi < b_iters ; vrbi+=b_inc) { for (vrci = c_start; vrci < c_iters ; vrci+=c_inc) { for (vrmi = m_start; (vrmi < m_iters) ; vrmi+=m_inc) { CHECK_OVERRIDES debug_show_current_iteration (); // Be sure to initialize the target registers first. initialize_target_registers (); initialize_source_registers (); printf ("%s", instruction_name); print_register_header (); printf( " =>"); fflush (stdout); if (!setup_only) { if (enable_setjmp) { if ( setjmp ( mybuf ) ) { printf("signal tripped. (FIXME)\n"); continue; } } (*test_function) (); } print_register_footer (); print_result_buffer (); printf ("\n"); } } } } } void mykillhandler ( int x ) { longjmp (mybuf, 1); } void mysegvhandler ( int x ) { longjmp (mybuf, 1); } static void do_tests ( void ) { int groupcount; char * cur_form; test_group_t group_function = &testfunction_generic; test_list_t *tests = testgroup_generic; struct sigaction kill_action, segv_action; struct sigaction old_kill_action, old_segv_action; if (enable_setjmp) { kill_action.sa_handler = mykillhandler; segv_action.sa_handler = mysegvhandler; sigemptyset ( &kill_action.sa_mask ); sigemptyset ( &segv_action.sa_mask ); kill_action.sa_flags = SA_NODEFER; segv_action.sa_flags = SA_NODEFER; sigaction ( SIGILL, &kill_action, &old_kill_action); sigaction ( SIGSEGV, &segv_action, &old_segv_action); } for (groupcount = 0; tests[groupcount].name != NULL; groupcount++) { cur_form = strdup(tests[groupcount].form); current_test = tests[groupcount]; if (groupcount < skip_count) continue; if (verbose) printf("Test #%d ,", groupcount); if (verbose > 1) printf(" instruction %s (v=%d)", current_test.name, verbose); (*group_function) (current_test.name, current_test.func, 0, cur_form ); printf ("\n"); if (groupcount >= (skip_count+test_count)) break; } if (debug_show_labels) printf("\n"); printf ("All done. Tested %d different instruction groups\n", groupcount); } static void usage (void) { fprintf(stderr, "Usage: test_isa_XXX [OPTIONS]\n" "\t-h: display this help and exit\n" "\t-v: increase verbosity\n" "\t-a : limit number of a-iterations to \n" "\t-b : limit number of b-iterations to \n" "\t-c : limit number of c-iterations to \n" "\t-n : limit to this number of tests.\n" "\t-r : run only test # \n" "\t\n" "\t-j :enable setjmp to recover from illegal insns. \n" "\t-m :(dev only?) lock VRM value to zero.\n" "\t-z :(dev only?) lock MC value to zero.\n" "\t-p :(dev only?) disable prefix instructions\n" "\t-s : skip tests \n" "\t-c : stop after running # of tests \n" "\t-f : Do the test setup but do not actually execute the test instruction. \n" ); } int main (int argc, char **argv) { int c; while ((c = getopt(argc, argv, "dhjvmpfzs:a:b:c:n:r:")) != -1) { switch (c) { case 'h': usage(); return 0; case 'v': verbose++; break; /* Options related to limiting the test iterations. */ case 'a': a_limit=atoi (optarg); printf ("limiting a-iters to %ld.\n", a_limit); break; case 'b': b_limit=atoi (optarg); printf ("limiting b-iters to %ld.\n", b_limit); break; case 'c': c_limit=atoi (optarg); printf ("limiting c-iters to %ld.\n", c_limit); break; case 'n': // run this number of tests. test_count=atoi (optarg); printf ("limiting to %ld tests\n", test_count); break; case 'r': // run just test #. skip_count=atoi (optarg); test_count=0; if (verbose) printf("Running only test number %ld\n", skip_count); break; case 's': // skip this number of tests. skip_count=atoi (optarg); printf ("skipping %ld tests\n", skip_count); break; /* debug options. */ case 'd': dump_tables=1; printf("DEBUG:dump_tables.\n"); break; case 'f': setup_only=1; printf("DEBUG:setup_only.\n"); break; case 'j': enable_setjmp=1; printf ("DEBUG:setjmp enabled.\n"); break; case 'm': vrm_override=1; printf ("DEBUG:vrm override enabled.\n"); break; case 'p': prefix_override=1; printf ("DEBUG:prefix override enabled.\n"); break; case 'z': mc_override=1; printf ("DEBUG:MC override enabled.\n"); break; default: usage(); fprintf(stderr, "Unknown argument: '%c'\n", c); } } generic_prologue (); build_vsx_table (); build_args_table (); build_float_vsx_tables (); if (dump_tables) { dump_float_vsx_tables (); dump_vsxargs (); } do_tests (); return 0; } #else // HAS_ISA_3_1 int main (int argc, char **argv) { printf("NO ISA 3.1 SUPPORT\n"); return 0; } #endif