| # -*- Makefile -*- |
| # Each plugin shall have a directory named 'test' that contains test cases |
| # directories and a Makefile which includes this Makefile template. |
| # The test Makefile specifies which tests to execute and how to verify them. |
| # The test to execute should be explicitly specified in the TESTS variable. |
| # Each test needs a verification step define in the name_of_test_verify variable. |
| # A simple diff verification template have been defined in the template Makefile |
| # diff_test performs a simple diff and takes name of file and its extension |
| # Example of a test Makefile is given below: |
| # |
| # include $(shell pwd)/../../Makefile_test.common |
| # TESTS = test1 test2 |
| # test1_verify = $(call diff_test,test1,ext) && test $$(grep "PASS" test1/test1.txt | wc -l) -eq 2 |
| # test2_verify = $(call diff_test,test2,ext) |
| # |
| |
| SHELL := /usr/bin/env bash |
| |
| # Either find yosys in system and use its path or use the given path |
| YOSYS_PATH ?= $(realpath $(dir $(shell command -v yosys))/..) |
| |
| # Find yosys-config, throw an error if not found |
| YOSYS_CONFIG = $(YOSYS_PATH)/bin/yosys-config |
| ifeq (,$(wildcard $(YOSYS_CONFIG))) |
| $(error "Didn't find 'yosys-config' under '$(YOSYS_PATH)'") |
| endif |
| |
| GTEST_DIR ?= $(abspath ../../third_party/googletest) |
| CXX ?= $(shell $(YOSYS_CONFIG) --cxx) |
| CXXFLAGS ?= $(shell $(YOSYS_CONFIG) --cxxflags) -I.. -I$(GTEST_DIR)/googletest/include |
| LDLIBS ?= $(shell $(YOSYS_CONFIG) --ldlibs) -L$(GTEST_DIR)/build/lib -lgtest -lgtest_main -lpthread |
| LDFLAGS ?= $(shell $(YOSYS_CONFIG) --ldflags) |
| TEST_UTILS ?= $(abspath ../../test-utils/test-utils.tcl) |
| |
| define test_tpl = |
| $(1): $(1)/ok |
| @set +e; \ |
| $$($$(subst /,-,$(1)_verify)); \ |
| if [ $$$$? -eq 0 ]; then \ |
| printf "Test %-20s \e[32mPASSED\e[0m @ %s\n" $(1) $(CURDIR); \ |
| touch $$<; \ |
| true; \ |
| else \ |
| printf "Test %-20s \e[31;1mFAILED\e[0m @ %s\n" $(1) $(CURDIR); \ |
| false; \ |
| fi |
| |
| $(1)/ok: $(1)/$$(notdir $(1).v) |
| @set +e; \ |
| cd $(1); \ |
| echo "source $(TEST_UTILS)" > run-$$(notdir $(1)).tcl ;\ |
| echo "source $$(notdir $(1)).tcl" >> run-$$(notdir $(1)).tcl ;\ |
| DESIGN_TOP=$$(notdir $(1)) TEST_OUTPUT_PREFIX=./ \ |
| yosys -c "run-$$(notdir $(1)).tcl" -q -q -l $$(notdir $(1)).log; \ |
| RETVAL=$$$$?; \ |
| rm -f run-$$(notdir $(1)).tcl; \ |
| if [ ! -z "$$($(1)_negative)" ] && [ $$($(1)_negative) -eq 1 ]; then \ |
| if [ $$$$RETVAL -ne 0 ]; then \ |
| printf "Negative test %-20s \e[32mPASSED\e[0m @ %s\n" $(1) $(CURDIR); \ |
| true; \ |
| else \ |
| printf "Negative test %-20s \e[31;1mFAILED\e[0m @ %s\n" $(1) $(CURDIR); \ |
| false; \ |
| fi \ |
| else \ |
| if [ $$$$RETVAL -ne 0 ]; then \ |
| echo "Unexpected runtime error"; \ |
| printf "Test %-20s \e[31;1mFAILED\e[0m @ %s\n" $(1) $(CURDIR); \ |
| false; \ |
| fi \ |
| fi |
| |
| endef |
| |
| DEV = $(shell echo $(1) | cut -d "/" -f 1) |
| SIM_LIBS = $(shell find ../$(DEV) -name "*.v" -not -name "*map.v" -not -name "cells_sim.v") |
| define test_sim_tpl = |
| $(1): $(1)/ok |
| @printf "Test %-18s \e[32mPASSED\e[0m @ %s\n" $(1) $(CURDIR); |
| |
| $(1)/$$(notdir $(1).vvp): $(1)/$$(notdir $(1).v) |
| @iverilog -vvvv -g2005 -o $$@ $$< $(SIM_LIBS) -I../ -DVCD_FILE=\"$(1)/$$(notdir $(1).vcd)\" >$(1)/$$(notdir $(1).vvp.log) 2>&1; \ |
| if [ $$$$? -ne 0 ]; then \ |
| printf "Test %-18s \e[31;1mFAILED\e[0m @ %s\n" $(1) $(CURDIR); \ |
| false; \ |
| fi |
| |
| $(1)/ok: $(1)/$$(notdir $(1).vvp) $(1)/$$(notdir $(1).v) |
| @vvp -vvvv $$< >$(1)/$$(notdir $(1).log) 2>&1; \ |
| if [ $$$$? -ne 0 ]; then \ |
| printf "Test %-18s \e[31;1mFAILED\e[0m @ %s\n" $(1) $(CURDIR); \ |
| false; \ |
| else \ |
| touch $$@; \ |
| true; \ |
| fi |
| |
| endef |
| |
| define test_post_synth_sim_tpl = |
| $(1): $(1)/ok |
| @printf "Test %-18s \e[32mPASSED\e[0m @ %s\n" $(1) $(CURDIR); |
| |
| $(1)/ok: $(1)/synth |
| @make -C $(1)/sim sim; \ |
| if [ $$$$? -ne 0 ]; then \ |
| printf "Test %-18s \e[31;1mFAILED\e[0m @ %s\n" $(1) $(CURDIR); \ |
| false; \ |
| else \ |
| touch $$@; \ |
| true; \ |
| fi |
| |
| $(1)/synth: $(1)/$$(notdir $(1).v) |
| @set +e; \ |
| cd $(1); \ |
| echo "source $(TEST_UTILS)" > run-$$(notdir $(1)).tcl ;\ |
| echo "source $$(notdir $(1)).tcl" >> run-$$(notdir $(1)).tcl ;\ |
| DESIGN_TOP=$$(notdir $(1)) TEST_OUTPUT_PREFIX=./ \ |
| yosys -c "run-$$(notdir $(1)).tcl" -q -q -l $$(notdir $(1)).log; \ |
| RETVAL=$$$$?; \ |
| rm -f run-$$(notdir $(1)).tcl; \ |
| if [ ! -z "$$($(1)_negative)" ] && [ $$($(1)_negative) -eq 1 ]; then \ |
| if [ $$$$RETVAL -ne 0 ]; then \ |
| printf "Negative test %-20s \e[32mPASSED\e[0m @ %s\n" $(1) $(CURDIR); \ |
| true; \ |
| else \ |
| printf "Negative test %-20s \e[31;1mFAILED\e[0m @ %s\n" $(1) $(CURDIR); \ |
| false; \ |
| fi \ |
| else \ |
| if [ $$$$RETVAL -ne 0 ]; then \ |
| echo "Unexpected runtime error"; \ |
| printf "Test %-20s \e[31;1mFAILED\e[0m @ %s\n" $(1) $(CURDIR); \ |
| false; \ |
| fi \ |
| fi |
| |
| endef |
| |
| define unit_test_tpl = |
| $(1): $(1)/$(1).test |
| @$$< |
| |
| $(1)/$(1).test: $(1)/$(1).test.o $$(GTEST_DIR)/build/lib/libgtest.a |
| @$(CXX) $(LDFLAGS) -o $$@ $$< $(LDLIBS) |
| |
| $(1)/$(1).test.o: $(1)/$(1).test.cc |
| @$(CXX) $(CXXFLAGS) $(LDFLAGS) -c $$< -o $$@ |
| |
| endef |
| |
| diff_test = diff $(1)/$(1).golden.$(2) $(1)/$(1).$(2) |
| |
| all: $(TESTS) $(SIM_TESTS) $(POST_SYNTH_SIM_TESTS) $(UNIT_TESTS) |
| |
| $(GTEST_DIR)/build/lib/libgtest.a $(GTEST_DIR)/build/lib/libgtest_main.a: |
| @mkdir -p $(GTEST_DIR)/build |
| @cd $(GTEST_DIR)/build; \ |
| cmake ..; \ |
| make |
| |
| .PHONY: all clean $(TESTS) $(SIM_TESTS) $(UNIT_TESTS) |
| |
| $(foreach test,$(TESTS),$(eval $(call test_tpl,$(test)))) |
| $(foreach test,$(SIM_TESTS),$(eval $(call test_sim_tpl,$(test)))) |
| $(foreach test,$(POST_SYNTH_SIM_TESTS),$(eval $(call test_post_synth_sim_tpl,$(test)))) |
| $(foreach test,$(UNIT_TESTS),$(eval $(call unit_test_tpl,$(test)))) |
| |
| clean: |
| @rm -rf $(foreach test,$(TESTS),$(test)/$(test).sdc $(test)/$(test)_[0-9].sdc $(test)/$(test).txt $(test)/$(test).eblif $(test)/$(test).json) |
| @rm -rf $(foreach test,$(SIM_TESTS),$(test)/*.vvp $(test)/*.vcd) |
| @rm -rf $(foreach test,$(POST_SYNTH_SIM_TESTS),$(test)/sim/*.vvp $(test)/sim/*.vcd $(test)/sim/*post_synth.v) |
| @rm -rf $(foreach test,$(UNIT_TESTS),$(test)/$(test).test.o $(test)/$(test).test.d $(test)/$(test).test) |
| @find . -name "ok" -or -name "*.log" | xargs rm -rf |