From 30eabe90c2e9ae62994a7f27f3e0f9541b90cbff Mon Sep 17 00:00:00 2001 From: Paul Mathieu Date: Tue, 10 May 2022 11:41:10 -0700 Subject: [PATCH] arm: add vivado helper scripts --- arm/vivado/make_mmi_file.tcl | 201 +++++++++++++++++++++++++++++++++ arm/vivado/make_prog_files.bat | 29 +++++ arm/vivado/make_prog_files.tcl | 101 +++++++++++++++++ 3 files changed, 331 insertions(+) create mode 100644 arm/vivado/make_mmi_file.tcl create mode 100644 arm/vivado/make_prog_files.bat create mode 100644 arm/vivado/make_prog_files.tcl diff --git a/arm/vivado/make_mmi_file.tcl b/arm/vivado/make_mmi_file.tcl new file mode 100644 index 0000000..cc3044b --- /dev/null +++ b/arm/vivado/make_mmi_file.tcl @@ -0,0 +1,201 @@ +# ----------------------------------------------------------------------------- +# The confidential and proprietary information contained in this file may +# only be used by a person authorised under and to the extent permitted +# by a subsisting licensing agreement from ARM limited. +# +# (C) COPYRIGHT 2018 ARM limited. +# ALL RIGHTS RESERVED +# +# This entire notice must be reproduced on all copies of this file +# and copies of this file may only be made by a person if such person is +# permitted to do so under the terms of a subsisting license agreement +# from ARM limited. +# +# SVN Information +# +# Checked In : $Date$ +# +# Revision : $Revision$ +# +# Release Information : Cortex-M1 DesignStart-r0p1-00rel0 +# +# ----------------------------------------------------------------------------- +# Project : Cortex-M1 Arty A7 Example design with V2C-DAPLink adaptor board +# +# Purpose : Script to get ITCM BRAM locations +# MMI format from following two articles +# +# https://www.xilinx.com/support/answers/63041.html +# https://forums.xilinx.com/t5/Vivado-TCL-Community/export-BRAM-locations-into-MMI-file/td-p/771221 +# ----------------------------------------------------------------------------- + +# Set MMI output file name +set mmi_file "m1.mmi" +set part "xc7a35tftg256-1" + +# Function to swap bits +proc swap_bits { bit } { + + if { $bit > 23 } {return [expr {24 + (31 - $bit)}]} + if { $bit > 15 } {return [expr {16 + (23 - $bit)}]} + if { $bit > 7 } {return [expr {8 + (15 - $bit)}]} + return [expr {7 - $bit}] +} + +# If run from batch file, will need to open project, then open the run +# open_run impl_1 + +# Find all the ITCM RAMs, place in a list +set itcm_ram [get_cells -hier -regexp {.*itcm.*ram_block_reg.*} -filter {REF_NAME =~ RAMB36E1}] + +# Vivado appears to read the memories in their actual bit order +# However update_mem amongst its very many failings doesn't support endianness, even though you specify it in the file! +# It also pays no attention to the bit_lane definition, it does the conversion based on the order memories are defined +# in the file! Not clear what the MMI file does achieve! + +# So go through and reverse each block of 4 memories +if { [expr {[llength $itcm_ram] % 4}] != 0 } { + puts "Error - Number of memories not divisible by 4" + return -1 +} + +# Number of RAMs details memory size. Each RAM is 32kb, so 4kB. +set itcm_size_bytes [expr {4096*[llength $itcm_ram]}] +puts "Instruction memory size $itcm_size_bytes" + +# Currently only support memory sizes between 16kB, (one byte per mem), and 128kB, (one bit per mem) +if { ($itcm_size_bytes < (4*4096)) || ($itcm_size_bytes > (32*4096)) } { + puts "Error - Memory size of $itcm_size_bytes out of range" + puts " Script only supports memory sizes between 16kB and 128kB" + return -1 +} + +# Create and open target mmi file +set fp [open $mmi_file {WRONLY CREAT TRUNC}] +if { $fp == 0 } { + puts "Error - Unable to open $mmi_file for writing" + return -1 +} + +# Write the file header +puts $fp "" +puts $fp "" +puts $fp " " +puts $fp " " +puts $fp " " + +# Create an array to put the location and top memory index into +array set mem_array {} + +# Calculate the expected number of bits per memory +set mem_bits [expr {32/[llength $itcm_ram]}] + +# set itcm_ram_reordered [list] + +for {set i 0} {$i < [llength $itcm_ram]/4} {incr i} { + set start [expr {$i*4}] + set end [expr {($i*4)+3}] + set new_list [lreverse [lrange $itcm_ram [expr {$i*4}] [expr {($i*4)+3}]]] +# puts "$start $end\n$new_list" + # lreplace $itcm_ram [expr {$i*4}] [expr {($i*4)+3}] [lreverse [lrange $itcm_ram [expr {$i*4}] [expr {($i*4)+3}]]] + + +# puts $itcm_ram_reordered + + # For each entry display the location + foreach ram $new_list { + # Get the RAM location + set loc_val [get_property LOC [get_cells $ram]] + regexp -- {(RAMB36_)([0-9XY]+)} $loc_val full ram_name loc_xy + + # Get the nets driven by the D0 pins + set data_bus [get_nets -of_objects [get_pins -filter {REF_PIN_NAME =~ DO*DO*} -of [get_cells $ram]]] + + # Check number of bits is the same as that expected + if { [llength $data_bus] != $mem_bits } { + puts "Error - Number of data pins read, [llength $data_bus], does not equal expected memory bits, $mem_bits" + return -1 + } + + # Number of pins connected to the memory sets the memory depth. + set memory_depth [expr {(32768/[llength $data_bus])-1}] + + set idx_list [list] + foreach entry $data_bus { + # Filter the data_bus down to just the two index numbers + set index [regexp -inline -- {[0-9]+} [regexp -inline -- {\[.*} [lindex $entry 0]]] + lappend idx_list $index + } + + # Sort the index list from highest to lowest + set idx_list [lsort -decreasing -integer $idx_list] + + # Assign the highest and lowest bits for the range variables + set index_low [lindex $idx_list end] + set index_high [lindex $idx_list 0] + + # Debug + # puts $data_bus + # puts $idx_list + # puts "$index_high downto $index_low pos $loc_val" + + array set mem_array [list $index_high $loc_xy] + }; # foreach +}; # for + + # Sort array into index order + array set mem_array_sorted {} + + foreach entry [lsort [array names mem_array]] { + # puts "$entry : $mem_array($entry)" + array set mem_array_sorted [list $entry $mem_array($entry)] + } + + foreach entry [array names mem_array_sorted] { + # puts "$entry : $mem_array_sorted($entry)" + } + + # MMI file needs to be in little endian format because update_mem doesn't actually use the endianness field + # So first index is 7, next is 15, 23, 31. + # Number of entries to write is 8/mem_bits. Lower index is index_high - (mem_bits - 1) + for {set top_idx 7} {$top_idx < 32} {incr top_idx 8} { + for {set idx_high $top_idx} {$idx_high > ($top_idx-8)} {incr idx_high -$mem_bits} { + # puts $idx_high + set loc $mem_array_sorted($idx_high) + set idx_low [expr {$idx_high - $mem_bits + 1}] + + if { $loc == "" } { + puts "Error - No location entry for index $idx_high" + return -1 + } + + # Write relevant XML + puts $fp " " + puts $fp " " + puts $fp " " + puts $fp " " + puts $fp " " + } + } + + + +# Write the file tail +puts $fp " " +puts $fp " " +puts $fp " " +puts $fp " " +puts $fp " " +puts $fp " " +puts $fp " " +puts $fp " " +puts $fp "" + +# Close the output file +close $fp + + +# Useful facilities +# Get the bus as a list +# get_nets -hierarchical -regexp {.*itcm.*doutA.*} diff --git a/arm/vivado/make_prog_files.bat b/arm/vivado/make_prog_files.bat new file mode 100644 index 0000000..161dfe2 --- /dev/null +++ b/arm/vivado/make_prog_files.bat @@ -0,0 +1,29 @@ +@REM ----------------------------------------------------------------------------- +@REM The confidential and proprietary information contained in this file may +@REM only be used by a person authorised under and to the extent permitted +@REM by a subsisting licensing agreement from ARM limited. +@REM +@REM (C) COPYRIGHT 2018 ARM limited. +@REM ALL RIGHTS RESERVED +@REM +@REM This entire notice must be reproduced on all copies of this file +@REM and copies of this file may only be made by a person if such person is +@REM permitted to do so under the terms of a subsisting license agreement +@REM from ARM limited. +@REM +@REM SVN Information +@REM +@REM Checked In : $Date$ +@REM +@REM Revision : $Revision$ +@REM +@REM Release Information : Cortex-M1 DesignStart-r0p1-00rel0 +@REM +@REM ----------------------------------------------------------------------------- +@REM Create MCS file for base board QSPI flash memory +@REM Requires existing bit file to convert +@REM Has to call Vivado in batch mode, then run TCL script + +set PATH=F:\xilinx\Vivado\2022.1\bin;%PATH% + +call vivado -mode batch -source make_prog_files.tcl -notrace diff --git a/arm/vivado/make_prog_files.tcl b/arm/vivado/make_prog_files.tcl new file mode 100644 index 0000000..d6ca36b --- /dev/null +++ b/arm/vivado/make_prog_files.tcl @@ -0,0 +1,101 @@ +# ----------------------------------------------------------------------------- +# The confidential and proprietary information contained in this file may +# only be used by a person authorised under and to the extent permitted +# by a subsisting licensing agreement from ARM limited. +# +# (C) COPYRIGHT 2018 ARM limited. +# ALL RIGHTS RESERVED +# +# This entire notice must be reproduced on all copies of this file +# and copies of this file may only be made by a person if such person is +# permitted to do so under the terms of a subsisting license agreement +# from ARM limited. +# +# SVN Information +# +# Checked In : $Date$ +# +# Revision : $Revision$ +# +# Release Information : Cortex-M1 DesignStart-r0p1-00rel0 +# +# ----------------------------------------------------------------------------- +# Project : Cortex-M1 Arty A7 Example design with V2C-DAPLink adaptor board +# +# Purpose : Script to create bit and mcs files for Arty A7 board +# +# Combines the original bit file, mmi file, and software elf to create +# the full bitstream +# Then converts full bitstream to mcs for download to the onboard flash +# +# Can be run either in Vivado GUI TCL console, or else in batch mode +# from command line +# If run in Vivado TCL console, pwd needs to be set to root of project, +# in the same location as the bit file +# ----------------------------------------------------------------------------- + +# Input files +set mmi_file "./m1.mmi" +set elf_file "../bsp/bootloader.elf" +set source_bit_file "./aum1.runs/impl_1/au_top.bit" +set reference_bit_file "./aum1_reference.bit" + +# Output files +set output_bit_file "aum1.bit" +set output_bin_file "aum1.bin" + +# Enable to turn on debug +set updatemem_debug 0 + +# Assemble bit file that can be downloaded to device directly +# Combine the original bit file, mmi file, and software elf to create the full bitstream + +# Delete target file +file delete -force $output_bit_file +file delete -force $output_bin_file + +# Determine if the user has built the project and has the target source file +# If not, then use the reference bit file shipped with the project +if { ![file exists $source_bit_file] } { + puts "\n********************************************" + puts "INFO - File $source_bit_file doesn't exist as project has not been built" + puts " Using $reference_bit_file instead\n" + puts "********************************************/n" + set source_bit_file $reference_bit_file +} + +# Banner message to console as there is no output for a few seconds +puts " Running updatemem ..." + +if { $updatemem_debug } { + set error [catch {exec updatemem --debug --force --meminfo $mmi_file --data $elf_file --bit $source_bit_file --proc dummy --out $output_bit_file} result] +} else { + set error [catch {exec updatemem --force --meminfo $mmi_file --data $elf_file --bit $source_bit_file --proc dummy --out $output_bit_file} result] +} + +# Print the stdout from updatemem +puts $result + +# Updatemem returns 0 even when there is an error, so cannot trap on error. Having deleted output file to start, then +# detect if it now exists, else exit. +if { ![file exists $output_bit_file] } { + puts "ERROR - $output_bit_file not made" + return -1 +} else { + puts "\n********************************************" + puts " $output_bit_file correctly generated" + puts "********************************************\n" +} + +# Create BIN file for base board QSPI flash memory +write_cfgmem -force -format BIN -size 2 -interface SPIx1 -loadbit " up 0 $output_bit_file" $output_bin_file + +# Check BIN was correctly made +if { ![file exists $output_bin_file] } { + puts "ERROR - $output_bit_file not made" + return -1 +} else { + puts "\n********************************************" + puts " $output_bin_file correctly generated" + puts "********************************************\n" +} \ No newline at end of file