| # SPDX-FileCopyrightText: 2020 Efabless Corporation |
| # |
| # Licensed under the Apache License, Version 2.0 (the "License"); |
| # you may not use this file except in compliance with the License. |
| # You may obtain a copy of the License at |
| # |
| # http://www.apache.org/licenses/LICENSE-2.0 |
| # |
| # Unless required by applicable law or agreed to in writing, software |
| # distributed under the License is distributed on an "AS IS" BASIS, |
| # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| # See the License for the specific language governing permissions and |
| # limitations under the License. |
| # |
| # SPDX-License-Identifier: Apache-2.0 |
| |
| # cannot commit files larger than 100 MB to GitHub |
| FILE_SIZE_LIMIT_MB = 100 |
| |
| # Commands to be used to compress/uncompress files |
| # they must operate **in place** (otherwise, modify the target to delete the |
| # intermediate file/archive) |
| COMPRESS ?= gzip -n --best |
| UNCOMPRESS ?= gzip -d |
| ARCHIVE_EXT ?= gz |
| |
| # The following variables are to build static pattern rules |
| |
| # Needed to rebuild archives that were previously split |
| SPLIT_FILES := $(shell find . -type f -name "*.$(ARCHIVE_EXT).00.split") |
| SPLIT_FILES_SOURCES := $(basename $(basename $(basename $(SPLIT_FILES)))) |
| |
| # Needed to uncompress the existing archives |
| ARCHIVES := $(shell find . -type f -name "*.$(ARCHIVE_EXT)") |
| ARCHIVE_SOURCES := $(basename $(ARCHIVES)) |
| |
| # Needed to compress and split files/archives that are too large |
| LARGE_FILES := $(shell find ./gds -type f -name "*.gds") |
| LARGE_FILES += $(shell find . -type f -size +$(FILE_SIZE_LIMIT_MB)M -not -path "./.git/*" -not -path "./gds/*" -not -path "./openlane/*") |
| LARGE_FILES_GZ := $(addsuffix .$(ARCHIVE_EXT), $(LARGE_FILES)) |
| LARGE_FILES_GZ_SPLIT := $(addsuffix .$(ARCHIVE_EXT).00.split, $(LARGE_FILES)) |
| # consider splitting existing archives |
| LARGE_FILES_GZ_SPLIT += $(addsuffix .00.split, $(ARCHIVES)) |
| |
| MCW_ROOT?=$(PWD)/mgmt_core_wrapper |
| MCW ?=LITEX_VEXRISCV |
| |
| # Install lite version of caravel, (1): caravel-lite, (0): caravel |
| MCW_LITE?=1 |
| |
| ifeq ($(MCW),LITEX_VEXRISCV) |
| MCW_NAME := mcw-litex-vexriscv |
| MCW_REPO := https://github.com/efabless/caravel_mgmt_soc_litex |
| MCW_BRANCH := main |
| else |
| MCW_NAME := mcw-pico |
| MCW_REPO := https://github.com/efabless/caravel_pico |
| MCW_BRANCH := main |
| endif |
| |
| # Install caravel as submodule, (1): submodule, (0): clone |
| SUBMODULE?=0 |
| |
| # Caravel Root (Default: pwd) |
| # Need to be overwritten if running the makefile from UPRJ_ROOT, |
| # If caravel is sub-moduled in the user project, run export CARAVEL_ROOT=$(pwd)/caravel |
| CARAVEL_ROOT ?= $(shell pwd) |
| |
| # User project root |
| UPRJ_ROOT ?= $(shell pwd) |
| |
| # MANAGEMENT AREA ROOT |
| MGMT_AREA_ROOT ?= $(shell pwd)/mgmt_core_wrapper |
| |
| # Build tasks such as make ship, make generate_fill, make set_user_id, make final run in the foreground (1) or background (0) |
| FOREGROUND ?= 1 |
| |
| # PDK setup configs |
| THREADS ?= $(shell nproc) |
| STD_CELL_LIBRARY ?= sky130_fd_sc_hd |
| SPECIAL_VOLTAGE_LIBRARY ?= sky130_fd_sc_hvl |
| IO_LIBRARY ?= sky130_fd_io |
| PRIMITIVES_LIBRARY ?= sky130_fd_pr |
| SKYWATER_COMMIT ?= c094b6e83a4f9298e47f696ec5a7fd53535ec5eb |
| OPEN_PDKS_COMMIT ?= 89f6ff4d9359a1ef9717b839e3e59dfdf33aaea6 |
| INSTALL_SRAM ?= no # = yes to enable |
| |
| .DEFAULT_GOAL := ship |
| # We need portable GDS_FILE pointers... |
| .PHONY: ship |
| ship: check-env uncompress uncompress-caravel |
| ifeq ($(FOREGROUND),1) |
| @echo "Running make ship in the foreground..." |
| $(MAKE) -f $(CARAVEL_ROOT)/Makefile __ship |
| @echo "Make ship completed." 2>&1 | tee -a ./signoff/build/make_ship.out |
| else |
| @echo "Running make ship in the background..." |
| nohup $(MAKE) -f $(CARAVEL_ROOT)/Makefile __ship >/dev/null 2>&1 & |
| tail -f signoff/build/make_ship.out |
| @echo "Make ship completed." 2>&1 | tee -a ./signoff/build/make_ship.out |
| endif |
| |
| __ship: |
| @echo "###############################################" |
| @echo "Generating Caravel GDS (sources are in the 'gds' directory)" |
| @sleep 1 |
| #### Runs from the CARAVEL_ROOT mag directory |
| @echo "\ |
| random seed `$(CARAVEL_ROOT)/scripts/set_user_id.py -report`; \ |
| drc off; \ |
| crashbackups stop; \ |
| addpath hexdigits; \ |
| addpath $(CARAVEL_ROOT)/mag; \ |
| addpath $(UPRJ_ROOT)/mag; \ |
| load user_project_wrapper; \ |
| property LEFview true; \ |
| property GDS_FILE $(UPRJ_ROOT)/gds/user_project_wrapper.gds; \ |
| property GDS_START 0; \ |
| load mgmt_core_wrapper; \ |
| property LEFview true; \ |
| property GDS_FILE ../mgmt_core_wrapper/gds/mgmt_core_wrapper.gds; \ |
| property GDS_START 0; \ |
| load $(UPRJ_ROOT)/mag/user_id_programming; \ |
| load $(UPRJ_ROOT)/mag/user_id_textblock; \ |
| load ../maglef/simple_por; \ |
| load $(UPRJ_ROOT)/mag/caravel -dereference; \ |
| select top cell; \ |
| expand; \ |
| cif *hier write disable; \ |
| cif *array write disable; \ |
| gds write $(UPRJ_ROOT)/gds/caravel.gds; \ |
| quit -noprompt;" > $(UPRJ_ROOT)/mag/mag2gds_caravel.tcl |
| ### Runs from CARAVEL_ROOT |
| @mkdir -p ./signoff/build |
| #@cd $(CARAVEL_ROOT)/mag && PDKPATH=${PDK_ROOT}/sky130A MAGTYPE=mag magic -noc -dnull -rcfile ${PDK_ROOT}/sky130A/libs.tech/magic/sky130A.magicrc $(UPRJ_ROOT)/mag/mag2gds_caravel.tcl 2>&1 | tee $(UPRJ_ROOT)/signoff/build/make_ship.out |
| @cd $(CARAVEL_ROOT)/mag && PDKPATH=${PDK_ROOT}/sky130A MAGTYPE=mag magic -noc -dnull -rcfile ./.magicrc $(UPRJ_ROOT)/mag/mag2gds_caravel.tcl 2>&1 | tee $(UPRJ_ROOT)/signoff/build/make_ship.out |
| ### @rm $(UPRJ_ROOT)/mag/mag2gds_caravel.tcl |
| |
| truck: check-env uncompress uncompress-caravel |
| ifeq ($(FOREGROUND),1) |
| @echo "Running make truck in the foreground..." |
| mkdir -p ./signoff |
| mkdir -p ./build |
| $(MAKE) -f $(CARAVEL_ROOT)/Makefile __truck |
| @echo "Make truck completed." 2>&1 | tee -a ./signoff/build/make_truck.out |
| else |
| @echo "Running make truck in the background..." |
| mkdir -p ./signoff |
| mkdir -p ./build |
| nohup $(MAKE) -f $(CARAVEL_ROOT)/Makefile __truck >/dev/null 2>&1 & |
| tail -f signoff/build/make_truck.out |
| @echo "Make truck completed." 2>&1 | tee -a ./signoff/build/make_truck.out |
| endif |
| |
| __truck: |
| @echo "###############################################" |
| @echo "Generating Caravan GDS (sources are in the 'gds' directory)" |
| @sleep 1 |
| #### Runs from the CARAVEL_ROOT mag directory |
| @echo "\ |
| random seed `$(CARAVEL_ROOT)/scripts/set_user_id.py -report`; \ |
| drc off; \ |
| crashbackups stop; \ |
| addpath hexdigits; \ |
| addpath $(CARAVEL_ROOT)/mag; \ |
| addpath $(UPRJ_ROOT)/mag; \ |
| load user_analog_project_wrapper; \ |
| property LEFview true; \ |
| property GDS_FILE $(UPRJ_ROOT)/gds/user_analog_project_wrapper.gds; \ |
| property GDS_START 0; \ |
| load mgmt_core_wrapper; \ |
| property LEFview true; \ |
| property GDS_FILE ../mgmt_core_wrapper/gds/mgmt_core_wrapper.gds; \ |
| property GDS_START 0; \ |
| load $(UPRJ_ROOT)/mag/user_id_programming; \ |
| load $(UPRJ_ROOT)/mag/user_id_textblock; \ |
| load ../maglef/simple_por; \ |
| load $(UPRJ_ROOT)/mag/caravan -dereference; \ |
| select top cell; \ |
| expand; \ |
| cif *hier write disable; \ |
| cif *array write disable; \ |
| gds write $(UPRJ_ROOT)/gds/caravan.gds; \ |
| quit -noprompt;" > $(UPRJ_ROOT)/mag/mag2gds_caravan.tcl |
| ### Runs from CARAVEL_ROOT |
| @mkdir -p ./signoff/build |
| #@cd $(CARAVEL_ROOT)/mag && PDKPATH=${PDK_ROOT}/sky130A MAGTYPE=mag magic -noc -dnull -rcfile ${PDK_ROOT}/sky130A/libs.tech/magic/sky130A.magicrc $(UPRJ_ROOT)/mag/mag2gds_caravan.tcl 2>&1 | tee $(UPRJ_ROOT)/signoff/build/make_truck.out |
| @cd $(CARAVEL_ROOT)/mag && PDKPATH=${PDK_ROOT}/sky130A MAGTYPE=mag magic -noc -dnull -rcfile ./.magicrc $(UPRJ_ROOT)/mag/mag2gds_caravan.tcl 2>&1 | tee $(UPRJ_ROOT)/signoff/build/make_truck.out |
| ### @rm $(UPRJ_ROOT)/mag/mag2gds_caravan.tcl |
| |
| .PHONY: clean |
| clean: |
| cd $(CARAVEL_ROOT)/verilog/dv/caravel/mgmt_soc/ && \ |
| $(MAKE) -j$(THREADS) clean |
| cd $(CARAVEL_ROOT)/verilog/dv/wb_utests/ && \ |
| $(MAKE) -j$(THREADS) clean |
| |
| |
| .PHONY: verify |
| verify: |
| cd $(CARAVEL_ROOT)/verilog/dv/caravel/mgmt_soc/ && \ |
| $(MAKE) -j$(THREADS) all |
| cd $(CARAVEL_ROOT)/verilog/dv/wb_utests/ && \ |
| $(MAKE) -j$(THREADS) all |
| |
| |
| |
| ##### |
| $(LARGE_FILES_GZ): %.$(ARCHIVE_EXT): % |
| @if ! [ $(suffix $<) = ".$(ARCHIVE_EXT)" ]; then\ |
| $(COMPRESS) $< > /dev/null &&\ |
| echo "$< -> $@";\ |
| fi |
| |
| $(LARGE_FILES_GZ_SPLIT): %.$(ARCHIVE_EXT).00.split: %.$(ARCHIVE_EXT) |
| @if [ -n "$$(find "$<" -prune -size +$(FILE_SIZE_LIMIT_MB)M)" ]; then\ |
| split $< -b $(FILE_SIZE_LIMIT_MB)M $<. -d &&\ |
| rm $< &&\ |
| for file in $$(ls $<.*); do mv "$$file" "$$file.split"; done &&\ |
| echo -n "$< -> $$(ls $<.*.split)" | tr '\n' ' ' && echo "";\ |
| fi |
| |
| # This target compresses all files larger than $(FILE_SIZE_LIMIT_MB) MB |
| .PHONY: compress |
| compress: $(LARGE_FILES_GZ) $(LARGE_FILES_GZ_SPLIT) |
| @echo "Files larger than $(FILE_SIZE_LIMIT_MB) MBytes are compressed!" |
| |
| |
| |
| ##### |
| $(ARCHIVE_SOURCES): %: %.$(ARCHIVE_EXT) |
| @$(UNCOMPRESS) $< |
| @echo "$< -> $@" |
| |
| .SECONDEXPANSION: |
| $(SPLIT_FILES_SOURCES): %: $$(sort $$(wildcard %.$(ARCHIVE_EXT).*.split)) |
| @cat $? > $@.$(ARCHIVE_EXT) |
| @rm $? |
| @echo "$? -> $@.$(ARCHIVE_EXT)" |
| @$(UNCOMPRESS) $@.$(ARCHIVE_EXT) |
| @echo "$@.$(ARCHIVE_EXT) -> $@" |
| |
| |
| .PHONY: uncompress |
| uncompress: $(SPLIT_FILES_SOURCES) $(ARCHIVE_SOURCES) |
| @echo "All files are uncompressed!" |
| |
| # Needed for targets that are run from UPRJ_ROOT for which caravel isn't submoduled. |
| .PHONY: uncompress-caravel |
| uncompress-caravel: |
| cd $(CARAVEL_ROOT) && \ |
| $(MAKE) uncompress |
| |
| # Digital Wrapper |
| # verify that the wrapper was respected |
| xor-wrapper: uncompress uncompress-caravel |
| ### first erase the user's user_project_wrapper.gds |
| sh $(CARAVEL_ROOT)/utils/erase_box.sh gds/user_project_wrapper.gds 0 0 2920 3520 |
| ### do the same for the empty wrapper |
| sh $(CARAVEL_ROOT)/utils/erase_box.sh $(CARAVEL_ROOT)/gds/user_project_wrapper_empty.gds 0 0 2920 3520 |
| mkdir -p signoff/user_project_wrapper_xor |
| ### XOR the two resulting layouts |
| sh $(CARAVEL_ROOT)/utils/xor.sh \ |
| $(CARAVEL_ROOT)/gds/user_project_wrapper_empty_erased.gds gds/user_project_wrapper_erased.gds \ |
| user_project_wrapper user_project_wrapper.xor.xml |
| sh $(CARAVEL_ROOT)/utils/xor.sh \ |
| $(CARAVEL_ROOT)/gds/user_project_wrapper_empty_erased.gds gds/user_project_wrapper_erased.gds \ |
| user_project_wrapper gds/user_project_wrapper.xor.gds > signoff/user_project_wrapper_xor/xor.log |
| rm $(CARAVEL_ROOT)/gds/user_project_wrapper_empty_erased.gds gds/user_project_wrapper_erased.gds |
| mv gds/user_project_wrapper.xor.gds gds/user_project_wrapper.xor.xml signoff/user_project_wrapper_xor |
| python $(CARAVEL_ROOT)/utils/parse_klayout_xor_log.py \ |
| -l signoff/user_project_wrapper_xor/xor.log \ |
| -o signoff/user_project_wrapper_xor/total.txt |
| ### screenshot the result for convenience |
| sh $(CARAVEL_ROOT)/utils/scrotLayout.sh \ |
| $(PDK_ROOT)/sky130A/libs.tech/klayout/sky130A.lyt \ |
| signoff/user_project_wrapper_xor/user_project_wrapper.xor.gds |
| @cat signoff/user_project_wrapper_xor/total.txt |
| |
| # Analog Wrapper |
| # verify that the wrapper was respected |
| xor-analog-wrapper: uncompress uncompress-caravel |
| ### first erase the user's user_project_wrapper.gds |
| sh $(CARAVEL_ROOT)/utils/erase_box.sh gds/user_analog_project_wrapper.gds 0 0 2920 3520 -8 -8 |
| ### do the same for the empty wrapper |
| sh $(CARAVEL_ROOT)/utils/erase_box.sh $(CARAVEL_ROOT)/gds/user_analog_project_wrapper_empty.gds 0 0 2920 3520 -8 -8 |
| mkdir -p signoff/user_analog_project_wrapper_xor |
| ### XOR the two resulting layouts |
| sh $(CARAVEL_ROOT)/utils/xor.sh \ |
| $(CARAVEL_ROOT)/gds/user_analog_project_wrapper_empty_erased.gds gds/user_analog_project_wrapper_erased.gds \ |
| user_analog_project_wrapper user_analog_project_wrapper.xor.xml |
| sh $(CARAVEL_ROOT)/utils/xor.sh \ |
| $(CARAVEL_ROOT)/gds/user_analog_project_wrapper_empty_erased.gds gds/user_analog_project_wrapper_erased.gds \ |
| user_analog_project_wrapper gds/user_analog_project_wrapper.xor.gds > signoff/user_analog_project_wrapper_xor/xor.log |
| rm $(CARAVEL_ROOT)/gds/user_analog_project_wrapper_empty_erased.gds gds/user_analog_project_wrapper_erased.gds |
| mv gds/user_analog_project_wrapper.xor.gds gds/user_analog_project_wrapper.xor.xml signoff/user_analog_project_wrapper_xor |
| python $(CARAVEL_ROOT)/utils/parse_klayout_xor_log.py \ |
| -l signoff/user_analog_project_wrapper_xor/xor.log \ |
| -o signoff/user_analog_project_wrapper_xor/total.txt |
| ### screenshot the result for convenience |
| sh $(CARAVEL_ROOT)/utils/scrotLayout.sh \ |
| $(PDK_ROOT)/sky130A/libs.tech/klayout/sky130A.lyt \ |
| signoff/user_analog_project_wrapper_xor/user_analog_project_wrapper.xor.gds |
| @cat signoff/user_analog_project_wrapper_xor/total.txt |
| |
| # LVS |
| BLOCKS = $(shell cd openlane && find * -maxdepth 0 -type d) |
| LVS_BLOCKS = $(foreach block, $(BLOCKS), lvs-$(block)) |
| $(LVS_BLOCKS): lvs-% : ./mag/%.mag ./verilog/gl/%.v |
| echo "Extracting $*" |
| mkdir -p ./mag/tmp |
| echo "addpath $(CARAVEL_ROOT)/mag/hexdigits;\ |
| addpath $(CARAVEL_ROOT)/subcells/simple_por/mag;\ |
| addpath \$$PDKPATH/libs.ref/sky130_ml_xx_hd/mag;\ |
| load $* -dereference;\ |
| select top cell;\ |
| foreach cell [cellname list children] {\ |
| load \$$cell -dereference;\ |
| property LEFview TRUE;\ |
| };\ |
| load $* -dereference;\ |
| select top cell;\ |
| extract no all;\ |
| extract do local;\ |
| extract unique;\ |
| extract;\ |
| ext2spice lvs;\ |
| ext2spice $*.ext;\ |
| feedback save extract_$*.log;\ |
| exit;" > ./mag/extract_$*.tcl |
| cd mag && \ |
| export MAGTYPE=maglef; \ |
| magic -rcfile ${PDK_ROOT}/sky130A/libs.tech/magic/sky130A.magicrc -noc -dnull extract_$*.tcl < /dev/null |
| mv ./mag/$*.spice ./spi/lvs |
| rm ./mag/*.ext |
| mv -f ./mag/extract_$*.tcl ./mag/tmp |
| mv -f ./mag/extract_$*.log ./mag/tmp |
| #### |
| mkdir -p ./spi/lvs/tmp |
| sh $(CARAVEL_ROOT)/spi/lvs/run_lvs.sh ./spi/lvs/$*.spice ./verilog/gl/$*.v $* |
| @echo "" |
| python3 $(CARAVEL_ROOT)/scripts/count_lvs.py -f ./verilog/gl/$*.v_comp.json | tee ./spi/lvs/tmp/$*.lvs.summary.log |
| mv -f ./verilog/gl/*.out ./spi/lvs/tmp 2> /dev/null || true |
| mv -f ./verilog/gl/*.json ./spi/lvs/tmp 2> /dev/null || true |
| mv -f ./verilog/gl/*.log ./spi/lvs/tmp 2> /dev/null || true |
| @echo "" |
| @echo "LVS: ./spi/lvs/$*.spice vs. ./verilog/gl/$*.v" |
| @echo "Comparison result: ./spi/lvs/tmp/$*.v_comp.out" |
| @awk '/^NET mismatches/,0' ./spi/lvs/tmp/$*.v_comp.out |
| |
| |
| LVS_GDS_BLOCKS = $(foreach block, $(BLOCKS), lvs-gds-$(block)) |
| $(LVS_GDS_BLOCKS): lvs-gds-% : ./gds/%.gds ./verilog/gl/%.v |
| echo "Extracting $*" |
| mkdir -p ./gds/tmp |
| echo " gds flatglob \"*_example_*\";\ |
| gds flatten true;\ |
| gds read ./$*.gds;\ |
| load $* -dereference;\ |
| select top cell;\ |
| extract no all;\ |
| extract do local;\ |
| extract;\ |
| ext2spice lvs;\ |
| ext2spice $*.ext;\ |
| feedback save extract_$*.log;\ |
| exit;" > ./gds/extract_$*.tcl |
| cd gds && \ |
| magic -rcfile ${PDK_ROOT}/sky130A/libs.tech/magic/sky130A.magicrc -noc -dnull extract_$*.tcl < /dev/null |
| mv ./gds/$*.spice ./spi/lvs |
| rm ./gds/*.ext |
| mv -f ./gds/extract_$*.tcl ./gds/tmp |
| mv -f ./gds/extract_$*.log ./gds/tmp |
| #### |
| mkdir -p ./spi/lvs/tmp |
| MAGIC_EXT_USE_GDS=1 sh $(CARAVEL_ROOT)/spi/lvs/run_lvs.sh ./spi/lvs/$*.spice ./verilog/gl/$*.v $* |
| @echo "" |
| python3 $(CARAVEL_ROOT)/scripts/count_lvs.py -f ./verilog/gl/$*.v_comp.json | tee ./spi/lvs/tmp/$*.lvs.summary.log |
| mv -f ./verilog/gl/*.out ./spi/lvs/tmp 2> /dev/null || true |
| mv -f ./verilog/gl/*.json ./spi/lvs/tmp 2> /dev/null || true |
| mv -f ./verilog/gl/*.log ./spi/lvs/tmp 2> /dev/null || true |
| @echo "" |
| @echo "LVS: ./spi/lvs/$*.spice vs. ./verilog/gl/$*.v" |
| @echo "Comparison result: ./spi/lvs/tmp/$*.v_comp.out" |
| @awk '/^NET mismatches/,0' ./spi/lvs/tmp/$*.v_comp.out |
| |
| |
| # connect-by-label is enabled here! |
| LVS_MAGLEF_BLOCKS = $(foreach block, $(BLOCKS), lvs-maglef-$(block)) |
| $(LVS_MAGLEF_BLOCKS): lvs-maglef-% : ./mag/%.mag ./verilog/gl/%.v |
| echo "Extracting $*" |
| mkdir -p ./maglef/tmp |
| echo "load $* -dereference;\ |
| select top cell;\ |
| foreach cell [cellname list children] {\ |
| load \$$cell -dereference;\ |
| property LEFview TRUE;\ |
| };\ |
| load $* -dereference;\ |
| select top cell;\ |
| extract no all;\ |
| extract do local;\ |
| extract;\ |
| ext2spice lvs;\ |
| ext2spice $*.ext;\ |
| feedback save extract_$*.log;\ |
| exit;" > ./mag/extract_$*.tcl |
| cd mag && export MAGTYPE=maglef; magic -noc -dnull extract_$*.tcl < /dev/null |
| mv ./mag/$*.spice ./spi/lvs |
| rm ./maglef/*.ext |
| mv -f ./mag/extract_$*.tcl ./maglef/tmp |
| mv -f ./mag/extract_$*.log ./maglef/tmp |
| #### |
| mkdir -p ./spi/lvs/tmp |
| sh $(CARAVEL_ROOT)/spi/lvs/run_lvs.sh ./spi/lvs/$*.spice ./verilog/gl/$*.v $* |
| @echo "" |
| python3 $(CARAVEL_ROOT)/scripts/count_lvs.py -f ./verilog/gl/$*.v_comp.json | tee ./spi/lvs/tmp/$*.maglef.lvs.summary.log |
| mv -f ./verilog/gl/*.out ./spi/lvs/tmp 2> /dev/null || true |
| mv -f ./verilog/gl/*.json ./spi/lvs/tmp 2> /dev/null || true |
| mv -f ./verilog/gl/*.log ./spi/lvs/tmp 2> /dev/null || true |
| @echo "" |
| @echo "LVS: ./spi/lvs/$*.spice vs. ./verilog/gl/$*.v" |
| @echo "Comparison result: ./spi/lvs/tmp/$*.v_comp.out" |
| @awk '/^NET mismatches/,0' ./spi/lvs/tmp/$*.v_comp.out |
| |
| # DRC |
| BLOCKS = $(shell cd openlane && find * -maxdepth 0 -type d) |
| DRC_BLOCKS = $(foreach block, $(BLOCKS), drc-$(block)) |
| $(DRC_BLOCKS): drc-% : ./gds/%.gds |
| echo "Running DRC on $*" |
| mkdir -p ./gds/tmp |
| cd gds && export DESIGN_IN_DRC=$* && export MAGTYPE=mag; magic -rcfile ${PDK_ROOT}/sky130A/libs.tech/magic/sky130A.magicrc -noc -dnull $(CARAVEL_ROOT)/gds/drc_on_gds.tcl < /dev/null |
| @echo "DRC result: ./gds/tmp/$*.drc" |
| |
| # Antenna |
| BLOCKS = $(shell cd openlane && find * -maxdepth 0 -type d) |
| ANTENNA_BLOCKS = $(foreach block, $(BLOCKS), antenna-$(block)) |
| $(ANTENNA_BLOCKS): antenna-% : ./gds/%.gds |
| echo "Running Antenna Checks on $*" |
| mkdir -p ./gds/tmp |
| cd gds && export DESIGN_IN_ANTENNA=$* && export MAGTYPE=mag; magic -rcfile ${PDK_ROOT}/sky130A/libs.tech/magic/sky130A.magicrc -noc -dnull $(CARAVEL_ROOT)/gds/antenna_on_gds.tcl < /dev/null 2>&1 | tee ./tmp/$*.antenna |
| mv -f ./gds/*.ext ./gds/tmp/ |
| @echo "Antenna result: ./gds/tmp/$*.antenna" |
| |
| # MAG2GDS |
| BLOCKS = $(shell cd openlane && find * -maxdepth 0 -type d) |
| MAG_BLOCKS = $(foreach block, $(BLOCKS), mag2gds-$(block)) |
| $(MAG_BLOCKS): mag2gds-% : ./mag/%.mag uncompress |
| echo "Converting mag file $* to GDS..." |
| echo "addpath $(CARAVEL_ROOT)/mag/hexdigits;\ |
| addpath ${PDKPATH}/libs.ref/sky130_ml_xx_hd/mag;\ |
| addpath $(CARAVEL_ROOT)/mag/primitives;\ |
| drc off;\ |
| gds rescale false;\ |
| load $* -dereference;\ |
| select top cell;\ |
| expand;\ |
| gds write $*.gds;\ |
| exit;" > ./mag/mag2gds_$*.tcl |
| cd ./mag && magic -rcfile ${PDK_ROOT}/sky130A/libs.tech/magic/sky130A.magicrc -noc -dnull mag2gds_$*.tcl < /dev/null |
| rm ./mag/mag2gds_$*.tcl |
| mv -f ./mag/$*.gds ./gds/ |
| |
| # MAG2LEF |
| BLOCKS = $(shell cd openlane && find * -maxdepth 0 -type d) |
| MAG_BLOCKS = $(foreach block, $(BLOCKS), mag2lef-$(block)) |
| $(MAG_BLOCKS): mag2lef-% : ./mag/%.mag uncompress |
| echo "Converting mag file $* to LEF..." |
| echo "addpath $(CARAVEL_ROOT)/mag/hexdigits;\ |
| addpath ${PDKPATH}/libs.ref/sky130_ml_xx_hd/mag;\ |
| addpath $(CARAVEL_ROOT)/mag/primitives;\ |
| drc off;\ |
| load $*;\ |
| lef write $*.lef;\ |
| exit;" > ./mag/mag2lef_$*.tcl |
| cd ./mag && magic -rcfile ${PDK_ROOT}/sky130A/libs.tech/magic/sky130A.magicrc -noc -dnull mag2lef_$*.tcl < /dev/null |
| rm ./mag/mag2lef_$*.tcl |
| mv -f ./mag/$*.lef ./lef/ |
| |
| # MAG2DEF |
| # BLOCKS = $(shell cd openlane && find * -maxdepth 0 -type d) |
| # MAG_BLOCKS = $(foreach block, $(BLOCKS), mag2lef-$(block)) |
| # $(MAG_BLOCKS): mag2lef-% : ./mag/%.mag uncompress |
| # echo "Converting mag file $* to DEF..." |
| # echo "addpath $(CARAVEL_ROOT)/mag/hexdigits;\ |
| # addpath ${PDKPATH}/libs.ref/sky130_ml_xx_hd/mag;\ |
| # addpath $(CARAVEL_ROOT)/mag/primitives;\ |
| # drc off;\ |
| # load $*;\ |
| # lef write $*.lef;\ |
| # exit;" > ./mag/mag2lef_$*.tcl |
| # cd ./mag && magic -rcfile ${PDK_ROOT}/sky130A/libs.tech/magic/sky130A.magicrc -noc -dnull mag2lef_$*.tcl < /dev/null |
| # rm ./mag/mag2lef_$*.tcl |
| # mv -f ./mag/$*.lef ./lef/ |
| |
| .PHONY: help |
| help: |
| @$(MAKE) -pRrq -f $(lastword $(MAKEFILE_LIST)) : 2>/dev/null | awk -v RS= -F: '/^# File/,/^# Finished Make data base/ {if ($$1 !~ "^[#.]") {print $$1}}' | sort | egrep -v -e '^[^[:alnum:]]' -e '^$@$$' |
| |
| # RCX Extraction |
| BLOCKS = $(shell cd openlane && find * -maxdepth 0 -type d) |
| RCX_BLOCKS = $(foreach block, $(BLOCKS), rcx-$(block)) |
| OPENLANE_IMAGE_NAME=efabless/openlane:2021.11.23_01.42.34 |
| $(RCX_BLOCKS): rcx-% : ./def/%.def check-mcw |
| echo "Running RC Extraction on $*" |
| mkdir -p ./def/tmp |
| # merge techlef and standard cell lef files |
| python3 $(OPENLANE_ROOT)/scripts/mergeLef.py -i $(PDK_ROOT)/sky130A/libs.ref/$(STD_CELL_LIBRARY)/techlef/$(STD_CELL_LIBRARY).tlef $(PDK_ROOT)/sky130A/libs.ref/$(STD_CELL_LIBRARY)/lef/*.lef \ |
| $(PDK_ROOT)/sky130A/libs.ref/$(SPECIAL_VOLTAGE_LIBRARY)/techlef/$(SPECIAL_VOLTAGE_LIBRARY).tlef $(PDK_ROOT)/sky130A/libs.ref/$(SPECIAL_VOLTAGE_LIBRARY)/lef/*.lef \ |
| $(PDK_ROOT)/sky130A/libs.ref/$(IO_LIBRARY)/lef/*.lef $(PDK_ROOT)/sky130A/libs.ref/$(PRIMITIVES_LIBRARY)/lef/*.lef \ |
| -o ./def/tmp/merged.lef |
| |
| echo "\ |
| read_liberty $(PDK_ROOT)/sky130A/libs.ref/$(STD_CELL_LIBRARY)/lib/$(STD_CELL_LIBRARY)__tt_025C_1v80.lib;\ |
| read_liberty $(PDK_ROOT)/sky130A/libs.ref/$(SPECIAL_VOLTAGE_LIBRARY)/lib/$(SPECIAL_VOLTAGE_LIBRARY)__tt_025C_3v30.lib;\ |
| read_liberty $(PDK_ROOT)/sky130A/libs.ref/$(SPECIAL_VOLTAGE_LIBRARY)/lib/$(SPECIAL_VOLTAGE_LIBRARY)__tt_025C_3v30_lv1v80.lib;\ |
| set std_cell_lef ./def/tmp/merged.lef;\ |
| set mgmt_area_lef $(MCW_ROOT)/lef/mgmt_core_wrapper.lef;\ |
| if {[catch {read_lef \$$std_cell_lef} errmsg]} {\ |
| puts stderr \$$errmsg;\ |
| exit 1;\ |
| };\ |
| foreach lef_file [glob ./lef/*.lef] {\ |
| if {[catch {read_lef \$$lef_file} errmsg]} {\ |
| puts stderr \$$errmsg;\ |
| exit 1;\ |
| }\ |
| };\ |
| if {[catch {read_lef \$$mgmt_area_lef} errmsg]} {\ |
| puts stderr \$$errmsg;\ |
| exit 1;\ |
| };\ |
| if {[catch {read_def -order_wires ./def/$*.def} errmsg]} {\ |
| puts stderr \$$errmsg;\ |
| exit 1;\ |
| };\ |
| read_sdc ./sdc/$*.sdc;\ |
| set_propagated_clock [all_clocks];\ |
| set rc_values \"mcon 9.249146E-3,via 4.5E-3,via2 3.368786E-3,via3 0.376635E-3,via4 0.00580E-3\";\ |
| set vias_rc [split \$$rc_values ","];\ |
| foreach via_rc \$$vias_rc {\ |
| set layer_name [lindex \$$via_rc 0];\ |
| set resistance [lindex \$$via_rc 1];\ |
| set_layer_rc -via \$$layer_name -resistance \$$resistance;\ |
| };\ |
| set_wire_rc -signal -layer met2;\ |
| set_wire_rc -clock -layer met5;\ |
| define_process_corner -ext_model_index 0 X;\ |
| extract_parasitics -ext_model_file ${PDK_ROOT}/sky130A/libs.tech/openlane/rcx_rules.info -corner_cnt 1 -max_res 50 -coupling_threshold 0.1 -cc_model 10 -context_depth 5;\ |
| write_spef ./spef/$*.spef" > ./def/tmp/rcx_$*.tcl |
| ## Generate Spef file |
| docker run -it -v $(OPENLANE_ROOT):/openLANE_flow -v $(PDK_ROOT):$(PDK_ROOT) -v $(PWD):/caravel -v $(MCW_ROOT):$(MCW_ROOT) -e PDK_ROOT=$(PDK_ROOT) -u $(shell id -u $(USER)):$(shell id -g $(USER)) $(OPENLANE_IMAGE_NAME) \ |
| sh -c " cd /caravel; openroad -exit ./def/tmp/rcx_$*.tcl |& tee ./def/tmp/rcx_$*.log" |
| ## Run OpenSTA |
| echo "\ |
| set std_cell_lef ./def/tmp/merged.lef;\ |
| set mgmt_area_lef $(MCW_ROOT)/lef/mgmt_core_wrapper.lef;\ |
| if {[catch {read_lef \$$std_cell_lef} errmsg]} {\ |
| puts stderr \$$errmsg;\ |
| exit 1;\ |
| };\ |
| foreach lef_file [glob ./lef/*.lef] {\ |
| if {[catch {read_lef \$$lef_file} errmsg]} {\ |
| puts stderr \$$errmsg;\ |
| exit 1;\ |
| }\ |
| };\ |
| if {[catch {read_lef \$$mgmt_area_lef} errmsg]} {\ |
| puts stderr \$$errmsg;\ |
| exit 1;\ |
| };\ |
| set_cmd_units -time ns -capacitance pF -current mA -voltage V -resistance kOhm -distance um;\ |
| read_liberty $(PDK_ROOT)/sky130A/libs.ref/$(STD_CELL_LIBRARY)/lib/$(STD_CELL_LIBRARY)__tt_025C_1v80.lib;\ |
| read_liberty $(PDK_ROOT)/sky130A/libs.ref/$(SPECIAL_VOLTAGE_LIBRARY)/lib/$(SPECIAL_VOLTAGE_LIBRARY)__tt_025C_3v30.lib;\ |
| read_liberty $(PDK_ROOT)/sky130A/libs.ref/$(SPECIAL_VOLTAGE_LIBRARY)/lib/$(SPECIAL_VOLTAGE_LIBRARY)__tt_025C_3v30_lv1v80.lib;\ |
| read_def ./def/$*.def;\ |
| read_spef ./spef/$*.spef;\ |
| read_sdc -echo ./sdc/$*.sdc;\ |
| write_sdf ./sdf/$*.sdf;\ |
| report_checks -fields {capacitance slew input_pins nets fanout} -path_delay min_max -group_count 1000;\ |
| report_check_types -max_slew -max_capacitance -max_fanout -violators;\ |
| report_checks -to [all_outputs] -group_count 1000;\ |
| report_checks -to [all_outputs] -unconstrained -group_count 1000;\ |
| " > ./def/tmp/or_sta_$*.tcl |
| docker run -it -v $(OPENLANE_ROOT):/openLANE_flow -v $(PDK_ROOT):$(PDK_ROOT) -v $(PWD):/caravel -v $(MCW_ROOT):$(MCW_ROOT) -e PDK_ROOT=$(PDK_ROOT) -u $(shell id -u $(USER)):$(shell id -g $(USER)) $(OPENLANE_IMAGE_NAME) \ |
| sh -c "cd /caravel; openroad -exit ./def/tmp/or_sta_$*.tcl |& tee ./def/tmp/or_sta_$*.log" |
| |
| |
| caravel_timing: ./def/caravel.def ./sdc/caravel.sdc ./verilog/gl/caravel.v check-mcw |
| mkdir -p ./def/tmp |
| ## Run OpenSTA |
| echo "\ |
| read_liberty $(PDK_ROOT)/sky130A/libs.ref/$(STD_CELL_LIBRARY)/lib/$(STD_CELL_LIBRARY)__tt_025C_1v80.lib;\ |
| read_liberty $(PDK_ROOT)/sky130A/libs.ref/sky130_sram_macros/lib/sky130_sram_2kbyte_1rw1r_32x512_8_TT_1p8V_25C.lib;\ |
| read_liberty $(PDK_ROOT)/sky130A/libs.ref/$(SPECIAL_VOLTAGE_LIBRARY)/lib/$(SPECIAL_VOLTAGE_LIBRARY)__tt_025C_3v30.lib;\ |
| read_liberty $(PDK_ROOT)/sky130A/libs.ref/$(SPECIAL_VOLTAGE_LIBRARY)/lib/$(SPECIAL_VOLTAGE_LIBRARY)__tt_025C_3v30_lv1v80.lib;\ |
| read_liberty $(PDK_ROOT)/sky130A/libs.ref/$(IO_LIBRARY)/lib/sky130_fd_io__top_gpiov2_tt_tt_025C_1v80_3v30.lib;\ |
| read_liberty $(PDK_ROOT)/sky130A/libs.ref/$(IO_LIBRARY)/lib/sky130_fd_io__top_ground_hvc_wpad_tt_025C_1v80_3v30_3v30.lib;\ |
| read_liberty $(PDK_ROOT)/sky130A/libs.ref/$(IO_LIBRARY)/lib/sky130_fd_io__top_ground_lvc_wpad_tt_025C_1v80_3v30.lib;\ |
| read_liberty $(PDK_ROOT)/sky130A/libs.ref/$(IO_LIBRARY)/lib/sky130_fd_io__top_ground_lvc_wpad_tt_100C_1v80_3v30.lib;\ |
| read_liberty $(PDK_ROOT)/sky130A/libs.ref/$(IO_LIBRARY)/lib/sky130_fd_io__top_power_lvc_wpad_tt_025C_1v80_3v30_3v30.lib;\ |
| read_liberty $(PDK_ROOT)/sky130A/libs.ref/$(IO_LIBRARY)/lib/sky130_fd_io__top_xres4v2_tt_tt_025C_1v80_3v30.lib;\ |
| read_liberty $(PDK_ROOT)/sky130A/libs.ref/$(IO_LIBRARY)/lib/sky130_ef_io__gpiov2_pad_tt_tt_025C_1v80_3v30.lib;\ |
| read_liberty $(PDK_ROOT)/sky130A/libs.ref/$(IO_LIBRARY)/lib/sky130_ef_io__vccd_lvc_clamped_pad_tt_025C_1v80_3v30_3v30.lib;\ |
| read_liberty $(PDK_ROOT)/sky130A/libs.ref/$(IO_LIBRARY)/lib/sky130_ef_io__vdda_hvc_clamped_pad_tt_025C_1v80_3v30_3v30.lib;\ |
| read_liberty $(PDK_ROOT)/sky130A/libs.ref/$(IO_LIBRARY)/lib/sky130_ef_io__vssa_hvc_clamped_pad_tt_025C_1v80_3v30_3v30.lib;\ |
| read_liberty $(PDK_ROOT)/sky130A/libs.ref/$(IO_LIBRARY)/lib/sky130_ef_io__vssd_lvc_clamped3_pad_tt_025C_1v80_3v30.lib;\ |
| read_liberty $(PDK_ROOT)/sky130A/libs.ref/$(IO_LIBRARY)/lib/sky130_ef_io__vccd_lvc_clamped3_pad_tt_025C_1v80_3v30_3v30.lib;\ |
| read_liberty $(PDK_ROOT)/sky130A/libs.ref/$(IO_LIBRARY)/lib/sky130_ef_io__vssd_lvc_clamped_pad_tt_025C_1v80_3v30.lib;\ |
| read_verilog $(MCW_ROOT)/verilog/gl/mgmt_core.v;\ |
| read_verilog $(MCW_ROOT)/verilog/gl/DFFRAM.v;\ |
| read_verilog $(MCW_ROOT)/verilog/gl/mgmt_core_wrapper.v;\ |
| read_verilog ./verilog/gl/caravel_clocking.v;\ |
| read_verilog ./verilog/gl/digital_pll.v;\ |
| read_verilog ./verilog/gl/housekeeping.v;\ |
| read_verilog ./verilog/gl/gpio_logic_high.v;\ |
| read_verilog ./verilog/gl/gpio_control_block.v;\ |
| read_verilog ./verilog/gl/gpio_defaults_block.v;\ |
| read_verilog ./verilog/gl/gpio_defaults_block_0403.v;\ |
| read_verilog ./verilog/gl/gpio_defaults_block_1803.v;\ |
| read_verilog ./verilog/gl/mgmt_protect_hv.v;\ |
| read_verilog ./verilog/gl/mprj_logic_high.v;\ |
| read_verilog ./verilog/gl/mprj2_logic_high.v;\ |
| read_verilog ./verilog/gl/mgmt_protect.v;\ |
| read_verilog ./verilog/gl/user_id_programming.v;\ |
| read_verilog ./verilog/gl/xres_buf.v;\ |
| read_verilog ./verilog/gl/spare_logic_block.v;\ |
| read_verilog ./verilog/gl/chip_io.v;\ |
| read_verilog ./verilog/gl/caravel.v;\ |
| link_design caravel;\ |
| read_spef -path soc/DFFRAM_0 $(MCW_ROOT)/spef/DFFRAM.spef;\ |
| read_spef -path soc/core $(MCW_ROOT)/spef/mgmt_core.spef;\ |
| read_spef -path soc $(MCW_ROOT)/spef/mgmt_core_wrapper.spef;\ |
| read_spef -path padframe ./spef/chip_io.spef;\ |
| read_spef -path rstb_level ./spef/xres_buf.spef;\ |
| read_spef -path pll ./spef/digital_pll.spef;\ |
| read_spef -path housekeeping ./spef/housekeeping.spef;\ |
| read_spef -path mgmt_buffers/powergood_check ./spef/mgmt_protect_hv.spef;\ |
| read_spef -path mgmt_buffers/mprj_logic_high_inst ./spef/mprj_logic_high.spef;\ |
| read_spef -path mgmt_buffers/mprj2_logic_high_inst ./spef/mprj2_logic_high.spef;\ |
| read_spef -path mgmt_buffers ./spef/mgmt_protect.spef;\ |
| read_spef -path \gpio_control_bidir_1[0] ./spef/gpio_control_block.spef;\ |
| read_spef -path \gpio_control_bidir_1[1] ./spef/gpio_control_block.spef;\ |
| read_spef -path \gpio_control_bidir_2[1] ./spef/gpio_control_block.spef;\ |
| read_spef -path \gpio_control_bidir_2[2] ./spef/gpio_control_block.spef;\ |
| read_spef -path \gpio_control_in_1[0] ./spef/gpio_control_block.spef;\ |
| read_spef -path \gpio_control_in_1[10] ./spef/gpio_control_block.spef;\ |
| read_spef -path \gpio_control_in_1[1] ./spef/gpio_control_block.spef;\ |
| read_spef -path \gpio_control_in_1[2] ./spef/gpio_control_block.spef;\ |
| read_spef -path \gpio_control_in_1[3] ./spef/gpio_control_block.spef;\ |
| read_spef -path \gpio_control_in_1[4] ./spef/gpio_control_block.spef;\ |
| read_spef -path \gpio_control_in_1[5] ./spef/gpio_control_block.spef;\ |
| read_spef -path \gpio_control_in_1[6] ./spef/gpio_control_block.spef;\ |
| read_spef -path \gpio_control_in_1[7] ./spef/gpio_control_block.spef;\ |
| read_spef -path \gpio_control_in_1[8] ./spef/gpio_control_block.spef;\ |
| read_spef -path \gpio_control_in_1[9] ./spef/gpio_control_block.spef;\ |
| read_spef -path \gpio_control_in_1a[0] ./spef/gpio_control_block.spef;\ |
| read_spef -path \gpio_control_in_1a[1] ./spef/gpio_control_block.spef;\ |
| read_spef -path \gpio_control_in_1a[2] ./spef/gpio_control_block.spef;\ |
| read_spef -path \gpio_control_in_1a[3] ./spef/gpio_control_block.spef;\ |
| read_spef -path \gpio_control_in_1a[4] ./spef/gpio_control_block.spef;\ |
| read_spef -path \gpio_control_in_1a[5] ./spef/gpio_control_block.spef;\ |
| read_spef -path \gpio_control_in_2[0] ./spef/gpio_control_block.spef;\ |
| read_spef -path \gpio_control_in_2[10] ./spef/gpio_control_block.spef;\ |
| read_spef -path \gpio_control_in_2[11] ./spef/gpio_control_block.spef;\ |
| read_spef -path \gpio_control_in_2[12] ./spef/gpio_control_block.spef;\ |
| read_spef -path \gpio_control_in_2[13] ./spef/gpio_control_block.spef;\ |
| read_spef -path \gpio_control_in_2[14] ./spef/gpio_control_block.spef;\ |
| read_spef -path \gpio_control_in_2[15] ./spef/gpio_control_block.spef;\ |
| read_spef -path \gpio_control_in_2[1] ./spef/gpio_control_block.spef;\ |
| read_spef -path \gpio_control_in_2[2] ./spef/gpio_control_block.spef;\ |
| read_spef -path \gpio_control_in_2[3] ./spef/gpio_control_block.spef;\ |
| read_spef -path \gpio_control_in_2[4] ./spef/gpio_control_block.spef;\ |
| read_spef -path \gpio_control_in_2[5] ./spef/gpio_control_block.spef;\ |
| read_spef -path \gpio_control_in_2[6] ./spef/gpio_control_block.spef;\ |
| read_spef -path \gpio_control_in_2[7] ./spef/gpio_control_block.spef;\ |
| read_spef -path \gpio_control_in_2[8] ./spef/gpio_control_block.spef;\ |
| read_spef -path \gpio_control_in_2[9] ./spef/gpio_control_block.spef;\ |
| read_spef -path gpio_defaults_block_0 ./spef/gpio_defaults_block_1803.spef;\ |
| read_spef -path gpio_defaults_block_1 ./spef/gpio_defaults_block_1803.spef;\ |
| read_spef -path gpio_defaults_block_2 ./spef/gpio_defaults_block_0403.spef;\ |
| read_spef -path gpio_defaults_block_3 ./spef/gpio_defaults_block_0403.spef;\ |
| read_spef -path gpio_defaults_block_4 ./spef/gpio_defaults_block_0403.spef;\ |
| read_spef -path gpio_defaults_block_5 ./spef/gpio_defaults_block.spef;\ |
| read_spef -path gpio_defaults_block_6 ./spef/gpio_defaults_block.spef;\ |
| read_spef -path gpio_defaults_block_7 ./spef/gpio_defaults_block.spef;\ |
| read_spef -path gpio_defaults_block_8 ./spef/gpio_defaults_block.spef;\ |
| read_spef -path gpio_defaults_block_9 ./spef/gpio_defaults_block.spef;\ |
| read_spef -path gpio_defaults_block_10 ./spef/gpio_defaults_block.spef;\ |
| read_spef -path gpio_defaults_block_11 ./spef/gpio_defaults_block.spef;\ |
| read_spef -path gpio_defaults_block_12 ./spef/gpio_defaults_block.spef;\ |
| read_spef -path gpio_defaults_block_13 ./spef/gpio_defaults_block.spef;\ |
| read_spef -path gpio_defaults_block_14 ./spef/gpio_defaults_block.spef;\ |
| read_spef -path gpio_defaults_block_15 ./spef/gpio_defaults_block.spef;\ |
| read_spef -path gpio_defaults_block_16 ./spef/gpio_defaults_block.spef;\ |
| read_spef -path gpio_defaults_block_17 ./spef/gpio_defaults_block.spef;\ |
| read_spef -path gpio_defaults_block_18 ./spef/gpio_defaults_block.spef;\ |
| read_spef -path gpio_defaults_block_19 ./spef/gpio_defaults_block.spef;\ |
| read_spef -path gpio_defaults_block_20 ./spef/gpio_defaults_block.spef;\ |
| read_spef -path gpio_defaults_block_21 ./spef/gpio_defaults_block.spef;\ |
| read_spef -path gpio_defaults_block_22 ./spef/gpio_defaults_block.spef;\ |
| read_spef -path gpio_defaults_block_23 ./spef/gpio_defaults_block.spef;\ |
| read_spef -path gpio_defaults_block_24 ./spef/gpio_defaults_block.spef;\ |
| read_spef -path gpio_defaults_block_25 ./spef/gpio_defaults_block.spef;\ |
| read_spef -path gpio_defaults_block_26 ./spef/gpio_defaults_block.spef;\ |
| read_spef -path gpio_defaults_block_27 ./spef/gpio_defaults_block.spef;\ |
| read_spef -path gpio_defaults_block_28 ./spef/gpio_defaults_block.spef;\ |
| read_spef -path gpio_defaults_block_29 ./spef/gpio_defaults_block.spef;\ |
| read_spef -path gpio_defaults_block_30 ./spef/gpio_defaults_block.spef;\ |
| read_spef -path gpio_defaults_block_31 ./spef/gpio_defaults_block.spef;\ |
| read_spef -path gpio_defaults_block_32 ./spef/gpio_defaults_block.spef;\ |
| read_spef -path gpio_defaults_block_33 ./spef/gpio_defaults_block.spef;\ |
| read_spef -path gpio_defaults_block_34 ./spef/gpio_defaults_block.spef;\ |
| read_spef -path gpio_defaults_block_35 ./spef/gpio_defaults_block.spef;\ |
| read_spef -path gpio_defaults_block_36 ./spef/gpio_defaults_block.spef;\ |
| read_spef -path gpio_defaults_block_37 ./spef/gpio_defaults_block.spef;\ |
| read_sdc -echo ./sdc/caravel.sdc;\ |
| report_checks -path_delay min -fields {slew cap input nets fanout} -format full_clock_expanded -group_count 50;\ |
| report_checks -path_delay max -fields {slew cap input nets fanout} -format full_clock_expanded -group_count 50;\ |
| report_worst_slack -max ;\ |
| report_worst_slack -min ;\ |
| report_checks -path_delay min -fields {slew cap input nets fanout} -format full_clock_expanded -slack_max 0.18 -group_count 10;\ |
| " > ./def/tmp/caravel_timing.tcl |
| sta -exit ./def/tmp/caravel_timing.tcl | tee ./signoff/caravel/caravel_timing.log |
| |
| |
| ########################################################################### |
| .PHONY: generate_fill |
| generate_fill: check-env check-uid check-project uncompress |
| ifeq ($(FOREGROUND),1) |
| @echo "Running generate_fill in the foreground..." |
| $(MAKE) -f $(CARAVEL_ROOT)/Makefile __generate_fill |
| @echo "Generate fill completed." 2>&1 | tee -a ./signoff/build/generate_fill.out |
| else |
| @echo "Running generate_fill in the background..." |
| @nohup $(MAKE) -f $(CARAVEL_ROOT)/Makefile __generate_fill >/dev/null 2>&1 & |
| tail -f signoff/build/generate_fill.out |
| @echo "Generate fill completed." | tee -a signoff/build/generate_fill.out |
| endif |
| |
| __generate_fill: |
| @mkdir -p ./signoff/build |
| @cp -r $(CARAVEL_ROOT)/mag/.magicrc $(shell pwd)/mag |
| python3 $(CARAVEL_ROOT)/scripts/generate_fill.py $(USER_ID) $(PROJECT) $(shell pwd) -dist 2>&1 | tee ./signoff/build/generate_fill.out |
| #python3 $(CARAVEL_ROOT)/scripts/generate_fill.py $(USER_ID) $(PROJECT) $(shell pwd) -keep 2>&1 | tee ./signoff/build/generate_fill.out |
| |
| |
| .PHONY: final |
| final: check-env check-uid check-project uncompress uncompress-caravel |
| ifeq ($(FOREGROUND),1) |
| $(MAKE) -f $(CARAVEL_ROOT)/Makefile __final |
| @echo "Final build completed." 2>&1 | tee -a ./signoff/build/final_build.out |
| else |
| $(MAKE) -f $(CARAVEL_ROOT)/Makefile __final >/dev/null 2>&1 & |
| tail -f signoff/build/final_build.out |
| @echo "Final build completed." 2>&1 | tee -a ./signoff/build/final_build.out |
| endif |
| |
| __final: |
| python3 $(CARAVEL_ROOT)/scripts/compositor.py $(USER_ID) $(PROJECT) $(shell pwd) $(CARAVEL_ROOT)/mag $(shell pwd)/gds -keep |
| #mv $(CARAVEL_ROOT)/mag/caravel_$(USER_ID).mag ./mag/ |
| @rm -rf ./mag/tmp |
| |
| .PHONY: set_user_id |
| set_user_id: check-env check-uid uncompress uncompress-caravel |
| ifeq ($(FOREGROUND),1) |
| $(MAKE) -f $(CARAVEL_ROOT)/Makefile __set_user_id |
| @echo "Set user ID completed." 2>&1 | tee -a ./signoff/build/set_user_id.out |
| else |
| $(MAKE) -f $(CARAVEL_ROOT)/Makefile __set_user_id >/dev/null 2>&1 & |
| tail -f signoff/build/set_user_id.out |
| @echo "Set user ID completed." 2>&1 | tee -a ./signoff/build/set_user_id.out |
| endif |
| |
| __set_user_id: |
| mkdir -p ./signoff/build |
| # Update info.yaml |
| # sed -r "s/^(\s*project_id\s*:\s*).*/\1${USER_ID}/" -i info.yaml |
| cp $(CARAVEL_ROOT)/gds/user_id_programming.gds ./gds/user_id_programming.gds |
| cp $(CARAVEL_ROOT)/mag/user_id_programming.mag ./mag/user_id_programming.mag |
| cp $(CARAVEL_ROOT)/mag/user_id_textblock.mag ./mag/user_id_textblock.mag |
| cp $(CARAVEL_ROOT)/verilog/rtl/caravel.v ./verilog/rtl/caravel.v |
| python3 $(CARAVEL_ROOT)/scripts/set_user_id.py $(USER_ID) $(shell pwd) 2>&1 | tee ./signoff/build/set_user_id.out |
| |
| .PHONY: gpio_defaults |
| gpio_defaults: check-env uncompress uncompress-caravel |
| ifeq ($(FOREGROUND),1) |
| $(MAKE) -f $(CARAVEL_ROOT)/Makefile __gpio_defaults |
| @echo "GPIO defaults completed." 2>&1 | tee -a ./signoff/build/__gpio_defaults.out |
| else |
| $(MAKE) -f $(CARAVEL_ROOT)/Makefile __gpio_defaults >/dev/null 2>&1 & |
| tail -f signoff/build/gpio_defaults.out |
| @echo "GPIO defaults completed." 2>&1 | tee -a ./signoff/build/__gpio_defaults.out |
| endif |
| |
| __gpio_defaults: |
| mkdir -p ./signoff/build |
| mkdir -p ./verilog/gl |
| python3 $(CARAVEL_ROOT)/scripts/gen_gpio_defaults.py $(shell pwd) 2>&1 | tee ./signoff/build/gpio_defaults.out |
| |
| .PHONY: update_caravel |
| update_caravel: |
| cd caravel/ && \ |
| git checkout master && \ |
| git pull |
| |
| ########################################################################### |
| |
| # Install Mgmt Core Wrapper |
| .PHONY: install_mcw |
| install_mcw: |
| ifeq ($(SUBMODULE),1) |
| @echo "Installing $(MCW_NAME) as a submodule.." |
| # Convert MCW_ROOT to relative path because .gitmodules doesn't accept '/' |
| $(eval MCW_PATH := $(shell realpath --relative-to=$(shell pwd) $(MCW_ROOT))) |
| @if [ ! -d $(MCW_ROOT) ]; then git submodule add --name $(MCW_NAME) $(MCW_REPO) $(MCW_PATH); fi |
| @git submodule update --init |
| @cd $(MCW_ROOT); git checkout $(MCW_BRANCH) |
| $(MAKE) simlink |
| else |
| @echo "Installing $(MCW_NAME).." |
| @git clone $(MCW_REPO) $(MCW_ROOT) |
| @cd $(MCW_ROOT); git checkout $(MCW_BRANCH) |
| endif |
| |
| # Update Mgmt Core Wrapper |
| .PHONY: update_mcw |
| update_mcw: check-mcw |
| ifeq ($(SUBMODULE),1) |
| @git submodule update --init --recursive |
| cd $(MCW_ROOT) && \ |
| git checkout $(MCW_BRANCH) && \ |
| git pull |
| else |
| cd $(MCW_ROOT)/ && \ |
| git checkout $(MCW_BRANCH) && \ |
| git pull |
| endif |
| |
| # Uninstall Mgmt Core Wrapper |
| .PHONY: uninstall_mcw |
| uninstall_mcw: |
| ifeq ($(SUBMODULE),1) |
| git config -f .gitmodules --remove-section "submodule.$(MCW_NAME)" |
| git add .gitmodules |
| git submodule deinit -f $(MCW_ROOT) |
| git rm --cached $(MCW_ROOT) |
| rm -rf .git/modules/$(MCW_NAME) |
| rm -rf $(MCW_ROOT) |
| else |
| rm -rf $(MCW_ROOT)/* |
| endif |
| |
| ########################################################################### |
| .PHONY: pdk |
| pdk: skywater-pdk skywater-library skywater-timing open_pdks build-pdk gen-sources |
| |
| $(PDK_ROOT)/skywater-pdk: |
| git clone https://github.com/google/skywater-pdk.git $(PDK_ROOT)/skywater-pdk |
| |
| .PHONY: skywater-pdk |
| skywater-pdk: check-env $(PDK_ROOT)/skywater-pdk |
| cd $(PDK_ROOT)/skywater-pdk && \ |
| git checkout main && git pull && \ |
| git checkout -qf $(SKYWATER_COMMIT) |
| |
| .PHONY: skywater-library |
| skywater-library: check-env $(PDK_ROOT)/skywater-pdk |
| cd $(PDK_ROOT)/skywater-pdk && \ |
| git submodule update --init libraries/$(STD_CELL_LIBRARY)/latest && \ |
| git submodule update --init libraries/$(IO_LIBRARY)/latest && \ |
| git submodule update --init libraries/$(SPECIAL_VOLTAGE_LIBRARY)/latest && \ |
| git submodule update --init libraries/$(PRIMITIVES_LIBRARY)/latest |
| |
| gen-sources: $(PDK_ROOT)/sky130A |
| touch $(PDK_ROOT)/sky130A/SOURCES |
| echo -ne "skywater-pdk " >> $(PDK_ROOT)/sky130A/SOURCES |
| cd $(PDK_ROOT)/skywater-pdk && git rev-parse HEAD >> $(PDK_ROOT)/sky130A/SOURCES |
| echo -ne "open_pdks " >> $(PDK_ROOT)/sky130A/SOURCES |
| cd $(PDK_ROOT)/open_pdks && git rev-parse HEAD >> $(PDK_ROOT)/sky130A/SOURCES |
| |
| skywater-timing: check-env $(PDK_ROOT)/skywater-pdk |
| cd $(PDK_ROOT)/skywater-pdk && \ |
| $(MAKE) timing |
| ### OPEN_PDKS |
| $(PDK_ROOT)/open_pdks: |
| git clone git://opencircuitdesign.com/open_pdks $(PDK_ROOT)/open_pdks |
| |
| .PHONY: open_pdks |
| open_pdks: check-env $(PDK_ROOT)/open_pdks |
| cd $(PDK_ROOT)/open_pdks && \ |
| git checkout master && git pull && \ |
| git checkout -qf $(OPEN_PDKS_COMMIT) |
| |
| .PHONY: build-pdk |
| build-pdk: check-env $(PDK_ROOT)/open_pdks $(PDK_ROOT)/skywater-pdk |
| [ -d $(PDK_ROOT)/sky130A ] && \ |
| (echo "Warning: A sky130A build already exists under $(PDK_ROOT). It will be deleted first!" && \ |
| sleep 5 && \ |
| rm -rf $(PDK_ROOT)/sky130A) || \ |
| true |
| cd $(PDK_ROOT)/open_pdks && \ |
| ./configure --enable-sky130-pdk=$(PDK_ROOT)/skywater-pdk/libraries --with-sky130-local-path=$(PDK_ROOT) --enable-sram-sky130=$(INSTALL_SRAM) && \ |
| cd sky130 && \ |
| $(MAKE) veryclean && \ |
| $(MAKE) && \ |
| make SHARED_PDKS_PATH=$(PDK_ROOT) install && \ |
| $(MAKE) clean |
| |
| .RECIPE: manifest |
| manifest: mag/ maglef/ verilog/rtl/ Makefile |
| touch manifest && \ |
| find verilog/rtl/* -type f ! -name "caravel_netlists.v" ! -name "user_*.v" ! -name "README" ! -name "defines.v" -exec shasum {} \; > manifest && \ |
| shasum scripts/set_user_id.py scripts/generate_fill.py scripts/compositor.py >> manifest |
| # shasum lef/user_project_wrapper_empty.lef >> manifest |
| # find maglef/*.mag -type f ! -name "user_project_wrapper.mag" -exec shasum {} \; >> manifest && \ |
| # shasum mag/caravel.mag mag/.magicrc >> manifest |
| |
| .RECIPE: master_manifest |
| master_manifest: |
| find verilog/rtl/* -type f -exec shasum {} \; > master_manifest && \ |
| find verilog/gl/* -type f -exec shasum {} \; >> master_manifest && \ |
| shasum scripts/set_user_id.py scripts/generate_fill.py scripts/compositor.py >> master_manifest && \ |
| find lef/*.lef -type f -exec shasum {} \; >> master_manifest && \ |
| find def/*.def -type f -exec shasum {} \; >> master_manifest && \ |
| find mag/*.mag -type f -exec shasum {} \; >> master_manifest && \ |
| find maglef/*.mag -type f -exec shasum {} \; >> master_manifest && \ |
| find spi/lvs/*.spice -type f -exec shasum {} \; >> master_manifest && \ |
| find gds/*.gds -type f -exec shasum {} \; >> master_manifest |
| |
| check-env: |
| ifndef PDK_ROOT |
| $(error PDK_ROOT is undefined, please export it before running make) |
| endif |
| |
| check-uid: |
| ifndef USER_ID |
| $(error USER_ID is undefined, please export it before running make set_user_id) |
| else |
| @echo USER_ID is set to $(USER_ID) |
| endif |
| |
| check-project: |
| ifndef PROJECT |
| $(error PROJECT is undefined, please export it before running make generate_fill or make final) |
| else |
| @echo PROJECT is set to $(PROJECT) |
| endif |
| |
| check-mcw: |
| @if [ ! -d "$(MCW_ROOT)" ]; then \ |
| echo "MCW Root: "$(MCW_ROOT)" doesn't exists, please export the correct path before running make. "; \ |
| exit 1; \ |
| fi |
| |
| # Make README.rst |
| README.rst: README.src.rst docs/source/getting-started.rst docs/source/tool-versioning.rst openlane/README.src.rst docs/source/caravel-with-openlane.rst Makefile |
| pip -q install rst_include && \ |
| rm -f README.rst && \ |
| rst_include include README.src.rst - | \ |
| sed \ |
| -e's@\.\/\_static@\/docs\/source\/\_static@g' \ |
| -e's@:doc:`tool-versioning`@`tool-versioning.rst <./docs/source/tool-versioning.rst>`__@g' \ |
| -e's@.. note::@**NOTE:**@g' \ |
| -e's@.. warning::@**WARNING:**@g' \ |
| > README.rst && \ |
| rst_include include openlane/README.src.rst - | \ |
| sed \ |
| -e's@https://github.com/efabless/caravel/blob/master/verilog@../verilog@g' \ |
| -e's@:ref:`getting-started`@`README.rst <../README.rst>`__@g' \ |
| -e's@https://github.com/efabless/caravel/blob/master/openlane/@./@g' \ |
| -e's@.. note::@**NOTE:**@g' \ |
| -e's@.. warning::@**WARNING:**@g' \ |
| > openlane/README.rst |