- IntegrationTests/ : HDL top files and simulations for chains of HLS modules and memories.
- TestBenches/ : Test bench code.
- TrackletAlgorithm/ : Algo source code.
- TrackQuality/ : BDT Track Quality specific source code.
- emData/ : .dat files with input/output test-bench data (corresponding to memory between algo steps) + .tab files of data for LUTs used internally by algos.
- project/ : .tcl scripts to create HLS project, compile & run code.
Most of the following directions will reference Vivado HLS and the corresponding
vivado_hls
command. You may use Vitis HLS instead by replacing all of thevivado_hls
commands withvitis_hls
.
An HLS project can be generated by running tcl file with Vivado/Vitis HLS in firmware-hls/project/ directory. e.g. To do so for the ProjectionRouter:
vivado_hls -f script_PR.tcl
This would create a project directory <project> ("projrouter" in case of the above example). The project name is defined in the tcl script. To open the project in GUI:
vivado_hls -p <project>
cd IntegrationTests/ReducedConfig/IRtoTB/script/
make -j $(nproc) Work
(makes HLS IP cores; run as many jobs as you have CPU cores).vivado -mode batch -source runSim.tcl
(runs Vivado simulation, which writes data output from chain to dataOut/*.txt).python ../../../common/script/CompareMemPrintsFW.py -p -s
(compares .txt files in emData and dataOut/ writing comparison to dataOut/*_cmp.txt. Uses Python 3.).make Work/Work.runs/synth_1
(runs synthesis, writes utilization & timing reports to current directory).make Work/Work.runs/impl_1
(runs implementation, writes utilization & timing reports to current directory). N.B. This step is optional, and not required for code validation.
These have test data corresponding to the contents of the memories between algo steps. Their data format is explained in https://twiki.cern.ch/twiki/bin/view/CMS/HybridDataFormat .
e.g. AllStubs*.dat contains one row per stub: "stub_number stub_coords_(binary)[r|z|phi|...] ditto_but_in_hex"; StubPairs*.dat contains one row per stub pair "pair_number stub_index_in_allstubs_mem_(binary)[innerLayer|outerLayer] ditto_but_in_hex.
File naming convention: "L3" or "D5" indicate barrel or disk number; "PHIC" indicates 3rd course phi division given layer of nonant.
Some of the files are large, so not stored directly in git. These are automatically downloaded when any of the scripts in the project/ directory are executed within Vivado/Vitis HLS.
These correspond to LUT used internally by the algo steps. The .tab file are in C++ and can be #include and compiled. The .dat files are a list of hex numbers which are easier to read in in hdl.
The files that are downloaded by emData/download.sh were created by the CMSSSW L1 track emulation, with the the following recipe (adapted from the L1TrackSoftware TWiki).
cmsrel CMSSW_12_0_0_pre4
cd CMSSW_12_0_0_pre4/src/
cmsenv
git cms-checkout-topic -u cms-L1TK:fw_synch_220523
A few configuration changes were made in order to output test vectors and lookup tables and adjust truncation. This required editing parameter values in L1Trigger/TrackFindingTracklet/interface/Settings.h to match the following excerpts:
…
//IR should be set to 108 to match the FW for the summer chain, but ultimately should be at 156
std::unordered_map<std::string, unsigned int> maxstep_{{"IR", 108}, //IR will run at a higher clock speed to handle
//input links running at 25 Gbits/s
//Set to 108 to match firmware project 240 MHz clock
…
//--- These used to create files needed by HLS code.
bool writeMem_{true}; //If true will print out content of memories (between algo steps) to files
bool writeTable_{true}; //If true will print out content of LUTs to files
bool writeConfig_{true}; //If true will print out the autogenerated configuration as files
…
bool writeHLSInvTable_{true}; //Write out tables of drinv and invt in tracklet calculator for HLS module
…
Then compilation was done with the usual command:
scram b -j8
The maximum number of events in L1Trigger/TrackFindingTracklet/test/L1TrackNtupleMaker_cfg.py was set to 100:
…
process.maxEvents = cms.untracked.PSet(input = cms.untracked.int32(100))
…
Also, in the same file, the algorithm was set to HYBRID_NEWKF:
…
# Set L1 tracking algorithm:
# 'HYBRID' (baseline, 4par fit) or 'HYBRID_DISPLACED' (extended, 5par fit).
# 'HYBRID_NEWKF' (baseline, 4par fit, with bit-accurate KF emulation),
# 'HYBRID_REDUCED' to use the "Summer Chain" configuration with reduced inputs.
# (Or legacy algos 'TMTT' or 'TRACKLET').
L1TRKALGO = 'HYBRID_NEWKF'
…
Finally, the emulation was run with:
cd L1Trigger/TrackFindingTracklet/test/
cmsRun L1TrackNtupleMaker_cfg.py
The files for the reduced configuration used for the summer/skinny chain were created by first running the full configuration as described above. Then, the project_generation_scripts repo was cloned, the wiring files for the full configuration were copied, the makeReducedConfig.py script was run, and the wiring files for the reduced configuration were copied back into the release:
cd $CMSSW_BASE/../
git clone https://github.com/cms-L1TK/project_generation_scripts.git
cd project_generation_scripts/
cp $CMSSW_BASE/src/L1Trigger/TrackFindingTracklet/data/LUTs/*.dat ./
./makeReducedConfig.py --no-graph
cp -fv reduced_*.dat $CMSSW_BASE/src/L1Trigger/TrackFindingTracklet/data/
cd -
The algorithm was changed to HYBRID_REDUCED:
…
# Set L1 tracking algorithm:
# 'HYBRID' (baseline, 4par fit) or 'HYBRID_DISPLACED' (extended, 5par fit).
# 'HYBRID_NEWKF' (baseline, 4par fit, with bit-accurate KF emulation),
# 'HYBRID_REDUCED' to use the "Summer Chain" configuration with reduced inputs.
# (Or legacy algos 'TMTT' or 'TRACKLET').
L1TRKALGO = 'HYBRID_REDUCED'
…
Finally, the emulation was rerun as for the full configuration:
cmsRun L1TrackNtupleMaker_cfg.py
The files for the configuration with combined modules (i.e., the
TrackletProcessor and MatchProcessor) were created with the same recipe as for
the full configuration described above, except the combined_
parameter was
also set to true in L1Trigger/TrackFindingTracklet/interface/Settings.h:
…
bool combined_{true}; // use combined TP (TE+TC) and MP (PR+ME+MC) configuration
…
Compilation was done again with the usual command:
cd $CMSSW_BASE/src/
scram b -j8
cd -
The algorithm was reset to HYBRID_NEWKF:
…
# Set L1 tracking algorithm:
# 'HYBRID' (baseline, 4par fit) or 'HYBRID_DISPLACED' (extended, 5par fit).
# 'HYBRID_NEWKF' (baseline, 4par fit, with bit-accurate KF emulation),
# 'HYBRID_REDUCED' to use the "Summer Chain" configuration with reduced inputs.
# (Or legacy algos 'TMTT' or 'TRACKLET').
L1TRKALGO = 'HYBRID_NEWKF'
…
Finally, the emulation was rerun as before:
cmsRun L1TrackNtupleMaker_cfg.py
The files for the barrel-only configuration were created by first running the full configuration as described above. Then, the head of the barrel_config branch of the emulation was checked out and compiled:
cd $CMSSW_BASE/src/
git ls-files -m | xargs git checkout --
git cms-checkout-topic -u cms-L1TK:e5047997b33dc6cfb1d7ce35aa34dfc56c0fe9bf
scram b -j8
cd -
The project_generation_scripts repo was cloned, the wiring files for the full configuration were copied, the makeBarrelConfig.py script was run, and the wiring files for the barrel-only configuration were copied back into the release:
cd $CMSSW_BASE/../
git clone https://github.com/cms-L1TK/project_generation_scripts.git
cd project_generation_scripts/
cp $CMSSW_BASE/src/L1Trigger/TrackFindingTracklet/data/LUTs/*.dat ./
./makeBarrelConfig.py
cp -fv barrel_*.dat $CMSSW_BASE/src/L1Trigger/TrackFindingTracklet/data/
cd -
The algorithm was reset to HYBRID_NEWKF:
…
# Set L1 tracking algorithm:
# 'HYBRID' (baseline, 4par fit) or 'HYBRID_DISPLACED' (extended, 5par fit).
# 'HYBRID_NEWKF' (baseline, 4par fit, with bit-accurate KF emulation),
# 'HYBRID_REDUCED' to use the "Summer Chain" configuration with reduced inputs.
# (Or legacy algos 'TMTT' or 'TRACKLET').
L1TRKALGO = 'HYBRID_NEWKF'
…
Finally, the emulation was rerun as before:
cmsRun L1TrackNtupleMaker_cfg.py
The steps needed to generate the configurations files for the reduced combined module chains is explained below.
Step 1: Produce the reduced configuration
Copy over the wires.dat, memorymodules.dat, and processingmodules.dat from the full configuration for the reduced modules to the area where you are runing the makeReducedConfig.py. I will assume that the namdes of these files are cm_wires.dat, cm_processingmodules.dat, and cm_memorymodules.dat.
We have three different reduced combined module configurations:
- The "CM_Reduced" configuration which is a skinny chain with on TP. To generate the configuation files for this configuration run the command:
./makeReducedConfig.py -w cm_wires.dat -p cm_processingmodules.dat -m cm_memorymodules.dat -s C -o CM_Reduced_ -t TP --no-graph
- The "CM_Reduced2" configuration which implementes all TP in L1L2 and all barrel MP is L3-L6. To generate this configuration run the command:
./makeReducedConfig.py -w cm_wires.dat -p cm_processingmodules.dat -m cm_memorymodules.dat -s All -o CM_Reduced2_ -t TP --no-graph
- The "CM_Barrel" configuration which has all barrel TP and MP. To generate the configuation files for this configuration run the commands (removes all disk related modules):
cat cm_memorymodules.dat | grep -v D1 | grep -v D2 | grep -v D3 | grep -v D4 | grep -v D5 > CM_Barrel_memorymodules.dat cat cm_processingmodules.dat | grep -v D1 | grep -v D2 | grep -v D3 | grep -v D4 | grep -v D5 > CM_Barrel_processingmodules.dat cat cm_wires.dat | grep -v D1 | grep -v D2 | grep -v D3 | grep -v D4 | grep -v D5 > CM_Barrel_wires.dat
This should produce the three files
- CM_{Reduced,Reduced2,Barrel}_wires.dat
- CM_{Reduced,Reduced2,Barrel}_memorymodules.dat
- CM_{Reduced,Reduced2,Barrel}_processingmodules.dat
respectively for the different configurations.
Step 2: Run CMSSW to pruduce the test vectors
NOTE - This has been tested in CMSSW_12_6_0_pre5 with the code in the PR #220
Copy the three *.dat files produced in Step 1 for the relevant configuration above to the 'data' director where you run CMSSW (TrackletTrackFinding/data)
Modify interface/Settings.h
Make IR process 108 stubs:
- std::unordered_map<std::string, unsigned int> maxstep_{{"IR", 156}, //IR will run at a higher clock speed to handle
- std::unordered_map<std::string, unsigned int> maxstep_{{"IR", 108}, //IR will run at a higher clock speed to handle
Print out memory and LUT files. (The writeConfig does not actually write out the config as it is not autogenerate, but read from a file.)
- bool writeMem_{false}; //If true will print out content of memories (between algo steps) to files
- bool writeTable_{false}; //If true will print out content of LUTs to files
- bool writeConfig_{false}; //If true will print out the autogenerated configuration as files
- bool writeMem_{true}; //If true will print out content of memories (between algo steps) to files
- bool writeTable_{true}; //If true will print out content of LUTs to files
- bool writeConfig_{true}; //If true will print out the autogenerated configuration as files
Turn off multiple matches in the MP
- bool doMultipleMatches_{true};
- bool doMultipleMatches_{false};
Use combined modules
- bool combined_{false};
- bool combined_{true};
Modify python/l1tTTTracksFromTrackletEmulation_cfi.py
- Reduced = cms.bool(False),
- Reduced = cms.bool(True),
and
- memoryModulesFile = cms.FileInPath('L1Trigger/TrackFindingTracklet/data/memorymodules_hourglassExtended.dat'),
- processingModulesFile = cms.FileInPath('L1Trigger/TrackFindingTracklet/data/processingmodules_hourglassExtended.dat'),
- wiresFile = cms.FileInPath('L1Trigger/TrackFindingTracklet/data/wires_hourglassExtended.dat'),
- memoryModulesFile = cms.FileInPath('L1Trigger/TrackFindingTracklet/data/CM_Reduced_memorymodules.dat'),
- processingModulesFile = cms.FileInPath('L1Trigger/TrackFindingTracklet/data/CM_Reduced_processingmodules.dat'),
- wiresFile = cms.FileInPath('L1Trigger/TrackFindingTracklet/data/CM_Reduced_wires.dat'),
(and similar for CM_Reduced2 and CM_Barrel configurations)
Modify test/L1TrackNtupleMaker_cfg.py:
-process.maxEvents = cms.untracked.PSet(input = cms.untracked.int32(10)) +process.maxEvents = cms.untracked.PSet(input = cms.untracked.int32(100))
After runing (cmsRun L1TrackNtupleMaker_cfg.py) in the TrackFindingTracklet/test director copy the configuration files:
cp CM_Reduced_wires.dat LUTs/wires.dat cp CM_Reduced_memorymodules.dat LUTs/memorymodules.dat cp CM_Reduced_processingmodules.dat LUTs/processingmodules.dat
Then create the tar balls:
tar czf MemPrints_CM_Reduced_230425.tgz MemPrints tar czf LUTs_CM_Reduced_230425.tgz LUTs
(and similar for CM_Reduced2 and CM_Barrel)
Purpose: Automatically run SW quality checks and build the HLS projects (csim, csynth, cosim, and export) for a PR to the master.
In order to keep the GitHub repository public we use GitHub Actions and GitLab CI/CD:
- GitHub Actions uses a public runner, the workflow is defined in .github/workflows/github_CI.yml
- GitHub Actions mirrors the repository to GitLab and runs GitLab CI/CD
- GitLab CI/CD uses a private runner (lnxfarm327.colorado.edu) and performs the SW quality checks and the HLS builds as defined in .gitlab-ci.yml
- SW quality checks are based on clang-tidy (llvm-toolset-7.0) and are defined in .clang-tidy and .clang-format very similar to CMSSW
- HLS builds are using Vivado HLS (or Vitis HLS) and are defined in the script files of the project folder
- Results (logs and artifacts) of the SW quality checks and HLS builds can be found here https://gitlab.cern.ch/cms-l1tk/firmware-hls/pipelines
- The default behavior blocks a stage (e.g. Hls-build) when a previous stage (e.g. Quality-check) failed
- GitHub Actions pulls the GitLab CI/CD status and the pass/fail outcome
- Add your branch name to the "on:" section of .github/workflows/GitLab_CI.yml
- In the "push:" subsection to trigger CI on each push, e.g. "branches: [feat_CI,<your_branch_name>]" and/or
- in the "pull_request:" subsection to trigger CI on each PR, e.g. "branches: [master,<your_branch_name>]"
Build the tracklet chain in EMP.
Currently the supported chain configurations for EMP builds are:
- Skinny Chain
- InputRouter to KalmanFilter
- Target: Apollo VU7P
- Path:
IntegrationTests/ReducedConfig/IRtoKF
- InputRouter to KalmanFilter
- Xilinx Vivado 2020.1 (HLS build) and 2021.2 (project build)
- ipbb: The IPbus Builder Tool. Tested with dev/2022g
- Python 3
- Questasim v2021.1_2 for Questa simulation
Step 1: Setup the work area
First source Xilinx Vivado 2020.1
ipbb init <project name>
cd <project name>
ipbb add git ssh://[email protected]:7999/p2-xware/firmware/emp-fwk.git -b v0.7.3
ipbb add git https://github.com/apollo-lhc/CM_FPGA_FW -b v1.2.2
ipbb add git https://gitlab.cern.ch/ttc/legacy_ttc.git -b v2.1
ipbb add git ssh://[email protected]:7999/cms-tcds/cms-tcds2-firmware.git -b v0_1_1
ipbb add git https://gitlab.cern.ch/HPTD/tclink.git -r fda0bcf
ipbb add git https://github.com/ipbus/ipbus-firmware -b v1.9
ipbb add git https://gitlab.cern.ch/dth_p1-v2/slinkrocket_ips.git -b v03.09
ipbb add git ssh://[email protected]:7999/dth_p1-v2/slinkrocket.git -b v03.10
ipbb add git https://gitlab.cern.ch/gbt-fpga/gbt-fpga.git -b gbt_fpga_6_1_0
ipbb add git https://gitlab.cern.ch/gbt-fpga/lpgbt-fpga.git -b v.2.1
ipbb add git https://:@gitlab.cern.ch:8443/gbtsc-fpga-support/gbt-sc.git -b gbt_sc_4_1
ipbb add git https://github.com/cms-L1TK/firmware-hls.git
Note: You need to be a member of the cms-tcds2-users
egroup in order to clone the cms-tcds2-firmware
repository. In order to add yourself to that egroup, go to the "Members" tab of this page, and click on the "Add me" button; you may need to wait ~ 24 hours to get access to the GitLab repo.
cd src/firmware-hls
make -C <EMP build path>/firmware
cd -
Source Xilinx Vivado 2021.2 for the following steps
Step 2: Create an ipbb project area
- For questa simulation testbench:
ipbb proj create sim qsim firmware-hls:<EMP build path> 'qsim.dep'
cd proj/qsim
- For vivado simulation testbench:
ipbb proj create vivado vsim firmware-hls:<EMP build path> 'vsim.dep'
cd proj/vsim
Step 3: Simulation
- For questa simulation testbench:
ipbb sim setup-simlib
ipbb sim ipcores
ipbb sim fli-udp
ipbb sim generate-project #(rerun this if you change VHDL)
./run_sim -c xil_defaultlib.top -Gsourcefile=<input.txt> -Gsinkfile=<out.txt> -Gplaylen=xyz -Gcaplen=xyz -do 'run 50.0us' -do quit
where xyz = number of events * 108
, where default is 9 events.
where input.txt
follows the standard EMP pattern file convention. An input file is provided in ../../src/firmware-hls/<EMP build path>/firmware/mem/in.txt
- For vivado simulation testbench
ipbb vivado generate-project
cd vsim
vivado vsim.xpr
and start the simulation from GUI (first time will take long).
Step 2: Create an ipbb project area
ipbb proj create vivado apollo firmware-hls:<EMP build path> 'apollo.dep'
cd proj/apollo
Step 3: Compile
Note: Note: For the following commands, you need to ensure that can find & use the gen_ipbus_addr_decode
script - e.g. for a standard uHAL installation:
export PATH=/opt/cactus/bin/uhal/tools:$PATH LD_LIBRARY_PATH=/opt/cactus/lib:$LD_LIBRARY_PATH
ipbb ipbus gendecoders
ipbb vivado generate-project synth -j8 impl -j8 package