

# BAG tutorial for schematic generation and simulation

Zhongkai Wang  
01/30/2018

# About BAG



- BAG is an interface between Python and CAD tools (e.g. Cadence)
- Writing parameterized schematic/layout generators
- Main purpose is for layout



# BAG related tools/softwares

- Python
- Git(Hub)
  - Version control software
- Pycharm
  - Python IDE



# Materials for BAG

- Course page last year
  - <http://inst.eecs.berkeley.edu/~ee240b/sp18/homework.html>
  - Several tutorials/slides by Eric
- Bootcamp examples
  - Embedded in the workspace
  - Run ./start\_tutorial.sh, look at module: schematic\_generators

A screenshot of a Jupyter Notebook interface. The title bar says "jupyter". Below it, there are tabs for "Files", "Running", and "Clusters". A "Logout" button is in the top right. The main area shows a file list with the following items:

| Name                            | Last Modified       |
|---------------------------------|---------------------|
| bootcamp_pcls                   | 2 hours ago         |
| solutions                       | 2 hours ago         |
| 1_low_demo.ipynb                | 2 hours ago         |
| 2_xbase_routing.ipynb           | 2 hours ago         |
| 3_analogbase.ipynb              | 2 hours ago         |
| 4_schematic_generators.ipynb    | Running 2 hours ago |
| 5_hierarchical_generators.ipynb | 2 hours ago         |
| 6_MOSDBDiscrete.ipynb           | 2 hours ago         |

A screenshot of a Jupyter Notebook cell titled "4\_schematic\_generators (autosaved)". The cell contains the following text:

**Schematic Generators**  
This module covers the basics of writing a schematic generator.

**Schematic Generation Flow**  
The schematic generator design flow is slightly different than the layout generator design flow. As described in Module 1, BAG copies an existing schematic template in Virtuoso and perform modifications on it in order to generate human-readable schematic instances. The schematic generation flow follows the steps below:

1. Create schematic in Virtuoso.
2. Import schematic information from Virtuoso to Python.
3. Implement schematic generator in Python.
4. Use BAG to create new instances of the schematic.

**CS Amplifier Schematic Generator Example**  
Let's walk through the common-source amplifier schematic generator example, reproduced below:

```
yaml_file = pkg_resources.resource_filename(__name__, os.path.join('netlist_info', 'amp.cs.yaml'))  
  
class demo_templates_...amp.cs(Module):
```

# Get workspace repo

- Cadence gpdk45
  - <https://github.com/ucb-art>
- Workspace
  - `git clone https://github.com/ucb-art/bag_workspace_gpdk045.git`
- Get submodules
  - Go to the folder just created
  - `git submodule update --init --recursive`
- `tcsh` → change to tcsh shell (you could also use bash)
- `source .cshrc`

The screenshot shows a portion of a GitHub interface. At the top, there's a repository card for "pfb" (Polyphase Filter Bank) which is a Scala project updated on May 24, 2018. Below it is another repository card for "bag\_workspace\_gpdk045", which is a Python project using BSD-3-Clause license, updated on Apr 17, 2018. This second card is highlighted with a red border. At the bottom, there's a section for "submodule\_merge\_test" with a note about testing git submodule merge, updated on Mar 22, 2018.

# Create your own design folder

- Create design folder -- folder to put all your design in
  - `mkdir bag_tutorial`
- Create sub-folders
  - `mkdir OA scripts`
- Purpose of folders
  - OA: Cadence library
  - Scripts: test script
  - Layout: layout scripts
  - BagModules: schematic scripts

# Start virtuoso

- Go back to your workspace
  - [virtuoso&](#)
- Start bag server
  - Run following command in CIW window
  - [load "start\\_bag.il"](#)

BAG server  
successfully  
starts



```
Virtuoso® 6.1.7-64b - Log: /home/aa/users/zhongkai/CDS.log
File Tools Options Help
cadence
Loading ams.cxt
Virtuoso Framework License (111) was checked out successfully. Total checkout time was 0.04s.
STARTING CDSINIT
END OF STANDARD CONFIG SETTINGS
loading vars from ./cdsenv for tool license
loading vars from ./cdsenv for tool asimenv
loading vars from ./cdsenv for tool auCore
loading vars from ./cdsenv for tool ihdl
loading vars from ./cdsenv for tool layout
loading vars from ./cdsenv for tool ui
loading vars from ./cdsenv for tool graphic
loading vars from ./cdsenv for tool schematic
loading vars from ./cdsenv for tool viva
2019/01/29 17:49:51 WARNING This OS does not appear to be a Cadence supported Linux configuration.
2019/01/29 17:49:51 For more info, please run CheckSysConf in <cdsRoot/tools.lnx86/bin/checkSysConf <productId>
2019/01/29 17:49:53 WARNING This OS does not appear to be a Cadence supported Linux configuration.
2019/01/29 17:49:53 For more info, please run CheckSysConf in <cdsRoot/tools.lnx86/bin/checkSysConf <productId>
END OF CUSTOM CONFIG SETTINGS
Loading pe.cxt
load "start_bag.il"
*INFO* Starting BAG server process.
*INFO* BAG skill server started.

load "start_bag.il"
mouse L: M:
1 | >
```

# Create template library

- Start library manager
  - Tools -> library manager
- Create bag\_tutorial library at bag\_tutorial/OA and attach it to the technology library



# Create CS Amplifier

- File -> New -> Cell View
- Create schematic with name cs\_amp
- Create the following schematic and create symbol for it



Res:  
analogLib library  
1K

NMOS:  
Bag\_prim library  
nmos4\_fast  
nf=nf



# Create testbench (cs\_amp\_tb)

Create testbench for cs\_amp , and change instance name to “XDUT”.



# Set simulation

- Create ADEXL
- Set up simulation options
  - Dc and ac simulation
  - Three variables:
    - in\_bias, supply, nf
  - Three outputs:
    - Out\_dc, out\_ac\_amp, out\_ac\_phase
  - Save the state

Two ways to get results:

- *getData* function
- Use *calculator* to save results



# Start bag and import library

- Run `./start_bag.sh`
- In ipython console
  - `import bag`
  - `prj = bag.BagProject()`
  - `prj.import_design_library('bag_tutorial')`
- Move the generated BagModules into `bag_tutorial` folder
  - Move all the design script to your design folder
- Change `bag_tutorial` definition in `bag_lib.def`

```
[zhongkai@hpse-9 bag_workspace_gpdk045]$ ./start_bag.sh
Python 3.6.3 |Anaconda custom (64-bit)| (default, Oct 13 2017, 12:02:49)
Type 'copyright', 'credits' or 'license' for more information
IPython 6.1.0 -- An enhanced Interactive Python. Type '?' for help.

In [1]: import bag

In [2]: prj = bag.BagProject()
*WARNING*: No TechInfo class defined. Using a dummy version.

In [3]: prj.import_design_library('bag_tutorial')
```

Create BagModules/bag\_tutorial folder



The screenshot shows the GVIM editor interface with the title bar "bag\_libs.def + (~/ee240b/demo/bag\_workspace\_gpdk045) - GVIM". The menu bar includes File, Edit, Tools, Syntax, Buffers, Window, and Help. Below the menu is a toolbar with various icons. The main editor area contains the following text:

```
BAG_prim $BAG_TECH_CONFIG_DIR/DesignModules
demo_templates $BAG_WORK_DIR/BAG_XBase_demo/BagModules
demo_testbenches $BAG_WORK_DIR/BAG_XBase_demo/BagModules
bag_serdes_ec $BAG_WORK_DIR/bag_serdes_ec/BagModules
bag_testbenches_ec $BAG_WORK_DIR/bag_testbenches_ec/BagModules
bag_tutorial /home/aa/users/zhongkai/ee240b/demo/bag_workspace_gpdk045/bag_tutorial/BagModules
```

The line "bag\_tutorial /home/aa/users/zhongkai/ee240b/demo/bag\_workspace\_gpdk045/bag\_tutorial/BagModules" is highlighted with a red border.

# cs\_amp schematic design script

- Change bag\_tutorial/BagModules/bag\_tutorial/cs\_amp.py
  - Add your design script to get\_params\_info and design function

```
@classmethod
def get_params_info(cls):
    # type: () -> Dict[str, str]
    """Returns a dictionary from parameter names to descriptions.

    Returns
    ------
    param_info : Optional[Dict[str, str]]
        dictionary from parameter names to descriptions.
    """
    return dict(
        mos_l='Channel length',
        mos_w='transistor width',
        mos_nf='number of fingers',
        mos_intent='threshold option',
    )

def design(self, mos_l, mos_w, mos_nf, mos_intent, **kwargs):
    """To be overridden by subclasses to design this module.

    This method should fill in values for all parameters in
    self.parameters. To design instances of this module, you can
    call their design() method or any other ways you coded.

    To modify schematic structure, call:

    rename_pin()
    delete_instance()
    replace_instance_master()
    reconnect_instance_terminal()
    restore_instance()
    array_instance()
    """
    self.instances['X0'].design(l=mos_l, w=mos_w, nf=mos_nf, intent=mos_intent)
```

Design  
parameter  
comments

Give transistor  
parameters

# cs\_amp\_tb schematic design script

- Change bag\_tutorial/BagModules/bag\_tutorial/cs\_amp\_tb.py
  - Add your design script to get\_params\_info and design function

```
@classmethod
def get_params_info(cls):
    # type: () -> Dict[str, str]
    """Returns a dictionary from parameter names to descriptions.

    Returns
    -------
    param_info : Optional[Dict[str, str]]
        Dictionary from parameter names to descriptions.
    """
    return dict(
        impl_lib='implementation library',
        cell_name='cell name'
    )

def design(self, impl_lib, cell_name):
    """To be overridden by subclasses to design this module.

    This method should fill in values for all parameters in
    self.parameters. To design instances of this module, you can
    call their design() method or any other ways you coded.

    To modify schematic structure, call:

    rename_pin()
    delete_instance()
    replace_instance_master()
    reconnect_instance_terminal()
    restore_instance()
    array_instance()
    ...
    self.replace_instance_master('XDUT', impl_lib, cell_name, static=True)
    """

```

Design  
parameter  
comments

Replace with  
generated schematic

# Top level design script (cs\_amp\_dsn.py)

- Create bag\_tutorial/scripts/cs\_amp\_dsn.py  
(you can also copy it from bCouses)

```
if __name__ is '__main__':
    if 'bprj' not in locals():
        bprj = bag.BagProject()

    impl_lib = 'AAAF00_CS_AMP'
    temp_lib = 'bag_tutorial'
    cell_name = 'cs_amp'
    tb_name = 'cs_amp_tb'

    sch_params = dict(
        mos_l=45e-9,
        mos_w=400e-9,
        mos_nf='nf',
        mos_intent='lvt',
    )

    # generate(bprj, temp_lib, impl_lib, cell_name, sch_params)
    # simulate(bprj, temp_lib, impl_lib, tb_name, cell_name, sim_params=[1.2, 0.7, 4])
    #
    # design1(bprj, temp_lib, impl_lib, tb_name, cell_name, sch_params)
    design2(bprj, temp_lib, impl_lib, tb_name, cell_name, sch_params)
```

Libraries  
and cell  
names

Schematic  
parameters

# Generate function (in cs\_amp\_dsn.py)

- Create a dsn object
- Design with schematic parameters
- Generate schematic to implementation library

```
def generate(prj, temp_lib, impl_lib, cell_name, sch_params):  
    print("Generating schematic ...")  
    dsn = prj.create_design_module(temp_lib, cell_name)  
    dsn.design(**sch_params)  
    dsn.implement_design(impl_lib, top_cell_name=cell_name)
```

# Simulation function (in cs\_amp\_dsn.py)

```
def simulate(prj, temp_lib, impl_lib, tb_name, cell_name, sim_params, show_plot=True):

    # generate tb schematic
    print("Generating testbench ...")
    dsn = prj.create_design_module(temp_lib, tb_name)
    dsn.design(impl_lib=impl_lib, cell_name=cell_name)
    dsn.implement_design(impl_lib, top_cell_name=tb_name)

    # configure tb
    tb = prj.configure_testbench(tb_lib=impl_lib, tb_cell=tb_name)
    tb.set_parameter('supply', sim_params[0])
    tb.set_parameter('in_bias', sim_params[1])
    tb.set_parameter('nf', sim_params[2])
    tb.update_testbench()

    # run simulation
    tb.run_simulation()
    print(tb.save_dir)
    results = load_sim_results(tb.save_dir)

    out_dc = results['out_dc']
    freq = results['freq']
    out_ac_amp = results['out_ac_amp']
    out_ac_phase = results['out_ac_phase']

    if show_plot:
        print(f"Output dc is {out_dc}")
        plt.figure()
        plt.subplot(2, 1, 1)
        plt.semilogx(freq, out_ac_amp)
        plt.subplot(2, 1, 2)
        plt.semilogx(freq, out_ac_phase)
        plt.show(block=False)

    return out_dc, out_ac_amp, out_ac_phase
```

Similar with  
cs\_amp

Configure  
tesbench and  
setup parameters

Run simulation  
and get results

Plot output

# Run script

- Change cs\_amp\_dsn.py

```
generate(bprj, temp_lib, impl_lib, cell_name, sch_params)
simulate(bprj, temp_lib, impl_lib, tb_name, cell_name, sim_params=[1.2, 0.7, 4])
#design1(bprj, temp_lib, impl_lib, tb_name, cell_name, sch_params)
#design2(bprj, temp_lib, impl_lib, tb_name, cell_name, sch_params)
```

- run [-i bag\\_tutorial/scripts/cs\\_amp\\_dsn.py](#)



# Design circuit with BAG

# More fun stuff

- Object: set CS amplifier output voltage around 700mV by changing nf  
(Assume input bias is also 700mV)
  - Hand calculation?
  - Look-up table or database (design 1)
  - Iteration or Sweep. (design 2)



# Generate MOS transistor look-up table

- run –i scripts\_char/nch\_char\_90n.py
- Workspace is embedded with a mos sweep script
  - Using MeasurementManager/TestbenchManager to sweep all the bias
  - script\_char/nch\_char\_90n.py
  - parameter/spec file in specs\_mos\_char/nch\_w0d4\_45n.yaml (a template file)
  - Save result to database
- Works pretty good with other devices
  - MOMCap, Inductor, CapDAC switches ...
  - Not favorable for larger circuit
- More details in Eric's discussion slides last year!!



```
schematic_params:  
  mos_type: 'inch'  
  lch: 45.0e-9  
  w: 0.4e-6  
  fg: 20  
  intent: 'lvt'  
  stack: 1  
  dum_info: !!null  
  dut_wrappers: []  
  env_list: ['tt']  
  
measurements:  
  - meas_type: 'mos_ss'  
    meas_package: 'verification_ec.mos.sim'  
    meas_class: 'MOSCharSS'  
    out_fname: 'mos_ss.yaml'  
    is_mmos: True  
    fg: 20  
    noise_temp_kelvin: 300  
    noise_integ_fstart: 0.1e+9  
    noise_integ_fstop: 0.3e+9  
    testbenches:  
      ibias:  
        tb_package: 'verification_ec.mos.sim'  
        tb_class: 'MOSIdTB'  
        tb_lib: 'bag_testbenches_ec'  
        tb_cell: 'mos_tb_ibias'  
        sch_params: {}  
        wrapper_type: ...  
        vgs_num: 200  
        vgs_max: 1.2  
        ibias_min_fg: 1.0e-6  
        ibias_max_fg: 200.0e-6  
        vgs_resolution: 4.0e-3  
      sp:  
        tb_package: 'verification_ec.mos.sim'  
        tb_class: 'MOSSPTB'  
        tb_lib: 'bag_testbenches_ec'  
        tb_cell: 'mos_tb_sp'  
        sch_params: {}  
        wrapper_type: ...  
        vgs_num: 30  
        vds_num: 20  
        vds_min: 5.0e-3  
        vds_max: 1.2  
        vbs: [0.0, 0.15, 0.3, 0.45]  
        sp_freq: 1.0e+6  
        cfit_method: 'average'  
      noise:  
        tb_package: 'verification_ec.mos.sim'  
        tb_class: 'MOSNoiseTB'  
        tb_lib: 'bag_testbenches_ec'  
        tb_cell: 'mos_tb_noise'  
        sch_params: {}  
        wrapper_type: ...  
        vgs_num: 30  
        vds_num: 20  
        vds_min: 5.0e-3  
        vds_max: 1.2
```

# Query function

- Search for transistor results with given bias
  - scripts\_char/mos\_query.py

```
interp_method = 'spline'  
sim_env = 'tt'  
nmos_spec = 'specs_mos_char/nch_w0d5_90n.yaml'  
  
intent = 'lvt'  
  
db = get_db(nmos_spec, intent, interp_method=interp_method, sim_env=sim_env)  
  
pprint.pprint(db.query(vbs=0, vds=0.45, vgs=0.45))  
pprint.pprint(db.query(vbs=0, vds=0.5, vgs=0.5))  
pprint.pprint(db.query(vbs=0.15, vds=0.55, vgs=0.55))  
pprint.pprint(db.query(vbs=0.15, vds=0.55, vstar=0.2))
```

Read database and  
run interpolation

Query function  
with given bias

- gm/Id and V\* design methodology

# Design with look-up table

- Change cs\_amp\_dsn.py

```
#generate(bprj, temp_lib, impl_lib, cell_name, sch_params)
#simulate(bprj, temp_lib, impl_lib, tb_name, cell_name, sim_params=[1.2, 0.7, 4])
design1(bprj, temp_lib, impl_lib, tb_name, cell_name, sch_params)
#design2(bprj, temp_lib, impl_lib, tb_name, cell_name, sch_params)
```

- run -i bag\_tutorial/scripts/cs\_amp\_dsn.py

- Could get current  $I_{DC}$  from output voltage and resistor value
- From query function, get current  $I_D$  of 1 finger transistor
- $nf = I_{DC}/I_D$

```
# get mos bias information
mos_info = db.query(vbs=0, vds=0.7, vgs=0.7)
# print mos bias information
pprint.pprint(mos_info)
print("\n")

# calculate
head_room = 0.5
res = 1e3
nf = round(head_room / res / mos_info['ibias'])
sch_params['nf'] = nf
print(f"MQS finger number is {nf}")
```



# Iteration or Sweep

- Change cs\_amp\_dsn.py

```
#generate(bprj, temp_lib, impl_lib, cell_name, sch_params)
#simulate(bprj, temp_lib, impl_lib, tb_name, cell_name, sim_params=[1.2, 0.7, 4])
#design1(bprj, temp_lib, impl_lib, tb_name, cell_name, sch_params)
Design2(bprj, temp_lib, impl_lib, tb_name, cell_name, sch_params)
```

- run [-i bag\\_tutorial/scripts/cs\\_amp\\_dsn.py](#)

- Start from 1 finger and add finger number until we get the voltage we need
  - Usually use it for *larger circuit*

```
out_dc = 1.2
nf = 1
nf_list = []
out_dc_list = []
while out_dc > 0.7:
    out_dc, out_ac_amp, out_ac_phase = \
        simulate(prj, temp_lib, impl_lib, tb_name, cell_name, sim_params=[1.2, 0.7, nf], show_plot=False)

    # add to lists
    nf_list.append(nf)
    out_dc_list.append(out_dc)
    print(f"Output dc is {out_dc} @ nf is {nf}")

    # change nf
    nf = nf+1
```

Thanks!  
Have fun with BAG!

Zhongkai Wang ([zhongkai@berkeley.edu](mailto:zhongkai@berkeley.edu))  
Ayan Biswas ([ayan\\_biswas@berkeley.edu](mailto:ayan_biswas@berkeley.edu))  
Eric Chang ([pkerichang@berkeley.edu](mailto:pkerichang@berkeley.edu))

# Backup slides

- Pycharm setup

# Step1: Create pycharm project



Your BAG  
workspace from git  
clone ..



# Add python folders to project

- Open the following folders and add to project
  - BAG2\_TEMPLATE\_EC -- BAG templates
  - BAG\_framework -- BAG framework
  - gdk045 -- technology folder
  - Other design folders if you are going to use them
    - E.g. bag\_serdes\_ec
    - Include your own design folder!



# Set project dependency

- Set depends on the folder is use scripts from other folders
  - BAG2\_TEMPLATES\_EC
    - Bag\_work\_space\_gpdk045, BAG\_framework, gpdk045
  - BAG framework
    - Bag\_work\_space\_gpdk045, BAG2\_TEMPLATES\_EC, gpdk045
  - Gpdk045
    - Bag\_work\_space\_gpdk045, BAG\_framework, BAG2\_TEMPLATES\_EC
  - Bag\_tutorial\_zk (My design folder)
    - Bag\_work\_space\_gpdk045, BAG\_framework, BAG2\_TEMPLATES\_EC, gpdk045



# Project Interpreter

- Change to available python verision (Python 3.6) for each folders

