## Redesigning Power Systems on a Single Chip Micro Mote with Berkeley Analog Generator Low Dropout Series Regulator Generation



Jackson Paddock

#### Electrical Engineering and Computer Sciences University of California, Berkeley

Technical Report No. UCB/EECS-2021-124 http://www2.eecs.berkeley.edu/Pubs/TechRpts/2021/EECS-2021-124.html

May 14, 2021

Copyright © 2021, by the author(s). All rights reserved.

Permission to make digital or hard copies of all or part of this work for personal or classroom use is granted without fee provided that copies are not made or distributed for profit or commercial advantage and that copies bear this notice and the full citation on the first page. To copy otherwise, to republish, to post on servers or to redistribute to lists, requires prior specific permission.

#### Redesigning Power Systems on SCµM with BAG LDO Generation

by Jackson Paddock

#### **Research Project**

Submitted to the Department of Electrical Engineering and Computer Sciences, University of California at Berkeley, in partial satisfaction of the requirements for the degree of **Master of Science**, **Plan II**.

Approval for the Report and Comprehensive Examination:

**Committee:** 

Professor Kristofer S.J. Pister Research Advisor

(Date)

\*

Professor Ali Niknejad Second Reader

(Date)

#### Abstract

Redesigning Power Systems on a Single Chip Micro Mote with Berkeley Analog Generator Low Dropout Series Regulator Generation

by

Jackson Paddock

Master of Science in Electrical Engineering and Computer Science

University of California, Berkeley

Professor Kristofer S.J. Pister, Chair

The Single Chip Micro Mote (SC $\mu$ M) is a crystal-free radio chip with an on-board CPU developed at UC Berkeley in the Swarm Lab. This chip was designed to function as the brain of an untethered microrobot with no external components other than a power source required. SC $\mu$ M also features an optical programmer so that no cables are even needed to program it. With its size and functionality, SC $\mu$ M has the potential to allow a swarm of microrobots to communicate and perform complex tasks in tandem.

Like every other circuit, SCµM needs power to operate, and one of the most common power regulation circuits is an LDO (low dropout series regulator), which unlike many other DC-DC converter designs, does not require any switching or inductors, which take up a lot of area. This circuit is relatively simple, requiring only a voltage amplifier and one additional transistor, but the challenges in design come from balancing stability, accuracy, and area. The goal of this project was to write a process-independent script that will automatically design such a circuit quickly and accurately in any technology while requiring minimal manual adjustment on the part of the designer. Due to the prevalence of LDOs in integrated circuit chips, the script discussed in this report has many real-world applications. To my parents and my brother, and my friends from BareStage.

Thank you for supporting me in everything I do and making my life that much brighter.

# Contents

| Co            | ontents                                                                                                                                                    | ii                             |
|---------------|------------------------------------------------------------------------------------------------------------------------------------------------------------|--------------------------------|
| Li            | st of Figures                                                                                                                                              | iii                            |
| $\mathbf{Li}$ | st of Tables                                                                                                                                               | iv                             |
| 1             | Background on SCµM Power Systems         1.1       Power Domains         1.2       Power Consumption                                                       | <b>1</b><br>1<br>1             |
| 2             | LDO Design with BAG2.1Berkeley Analog Generator (BAG)2.2LDO Design2.3Script Design Approach2.4Comparison to Hand-Designed LDOs2.5Approximations and Errors | <b>4</b><br>4<br>8<br>14<br>25 |
| 3             | Conclusion         3.1       Future Work                                                                                                                   | <b>31</b><br>31                |
| Bi            | ibliography                                                                                                                                                | 33                             |
| $\mathbf{A}$  | BAG Setup                                                                                                                                                  | 34                             |
| в             | Example Characterization Input YAML File                                                                                                                   | 39                             |
| С             | Example Design Script Input YAML File                                                                                                                      | 43                             |
| D             | LDO Design Script                                                                                                                                          | 45                             |
| $\mathbf{E}$  | VDDD Tap (SCµM)                                                                                                                                            | 60                             |

# List of Figures

| 1.1  | LDO with PMOS series device.                                                           | 2  |
|------|----------------------------------------------------------------------------------------|----|
| 2.1  | Block diagram of LDO feedback loop for a PMOS series device                            | 6  |
| 2.2  | Block diagram of LDO power supply gain loop for a PMOS series device                   | 6  |
| 2.3  | Small signal model for LDO load regulation calculation at low frequency with a         |    |
|      | PMOS series device.                                                                    | 7  |
| 2.4  | LDO with 5T differential amplifier topology and PMOS series device                     | 8  |
| 2.5  | SCµM 3C auxiliary and digital LDO transient response at 5MHz.                          | 15 |
| 2.6  | SCµM generated LDO transient for amplifier channel length of 500nm at 5MHz.            | 19 |
| 2.7  | SCµM generated LDO transient for amplifier channel length of 1µm at 5MHz.              | 21 |
| 2.8  | $SC\mu M$ generated LDO transient for amplifier channel length of $5\mu m$ at $5MHz$ . | 23 |
| 2.9  | EE290C hand-designed LDO transient response at 5MHz.                                   | 24 |
| 2.10 | EE290C generated LDO transient for amplifier channel length of 400nm at 5MHz.          | 27 |
| 2.11 | EE290C generated LDO transient for amplifier channel length of 1 $\mu$ m at 5MHz.      | 29 |
| E.1  | VDDD tap device proposed layout area.                                                  | 61 |
|      | SCµM Digital LDO schematic with NMOS tap device.                                       | 62 |

# List of Tables

| 1.1  | SCµM 3C current draw in different operating states [3]                              | 1  |
|------|-------------------------------------------------------------------------------------|----|
| 1.2  | $SC\mu M$ power domains [2]                                                         | 2  |
| 2.1  | LDO design script input parameters.                                                 | 9  |
| 2.2  | LDO design script output dictionary keys.                                           | 9  |
| 2.3  | SCµM 3C auxiliary and digital LDO performance.                                      | 14 |
| 2.4  | $SC\mu M$ generated LDO input parameters for amplifier channel length of 500nm.     | 18 |
| 2.5  | $SC\mu M$ generated LDO performance for amplifier channel length of 500nm           | 18 |
| 2.6  | $SC\mu M$ generated LDO input parameters for amplifier channel length of $1\mu m$ . | 20 |
| 2.7  | SCµM generated LDO performance for amplifier channel length of 1µm                  | 20 |
| 2.8  | $SC\mu M$ generated LDO input parameters for amplifier channel length of 5 $\mu m$  | 22 |
| 2.9  | SCµM generated LDO performance for amplifier channel length of 5µm                  | 22 |
| 2.10 | EE290C LDO performance with 10mA load current                                       | 25 |
| 2.11 | EE290C generated LDO input parameters for amplifier channel length of 400nm.        | 26 |
| 2.12 | EE290C generated LDO performance for amplifier channel length of 400nm              | 26 |
| 2.13 | EE290C generated LDO input parameters for amplifier channel length of $1\mu m$ .    | 28 |
| 2.14 | EE290C generated LDO performance for amplifier channel length of 1µm                | 28 |

#### Acknowledgments

I would like to offer many thanks to my advisor, Professor Kristofer Pister for guiding me through this program and always having more confidence in me than I at times have have in myself. Without him, I would not be where I am today and for that I would like to offer my utmost gratitude.

Thank you to Lydia Lee, who has been my mentor on this project and has always given me the guidance I needed when I was unsure of how to proceed with my work. Thank you for your endless support and patience.

Thank you to Alex Moreno, Austin Patel, David Burnett, Fil Maksimovic, and the rest of the SCµM team for all the advice and knowledge you have given me in the short time I've worked with you.

## Chapter 1

# Background on $SC\mu M$ Power Systems

## 1.1 Power Domains

SCµM has eight power domains as listed in Table 1.2, each of which is supplied by an LDO connected to the off-chip battery voltage. There are also four different bandgap references, inside the blocks for the digital LDO, optical/alwayson, radio divider, and radio LO. Each of these LDOs is connected to the battery voltage for power, which is nominally at 1.5V, but can be brought down to 1.2V. The chip also uses an external VDDIO to supply the level shifters in the pad ring and GPIO to connect to chip-level inputs and outputs.

## 1.2 Power Consumption

The power consumption of the LDOs themselves are listed in Table 1.2. Each LDO reference voltage is set by a constant bandgap reference current running through a tunable resistor. The resistor for each LDO is the same and can be tuned to achieve discrete voltages between 0.8V and 1.2V. With a reference current of 500nA or 1µA, the resistor ranges between 950k $\Omega$  and 1.7M $\Omega$ . The LDOs that consume the most power in the amplifier, the Always On and Sensor ADC LDOs, also have the largest biasing networks, which are partially for the enable/disable bits that control whether or not the LDOs supply power to their loads.

| State         | Clock Rate | Average Current Consumption |  |  |  |  |
|---------------|------------|-----------------------------|--|--|--|--|
| Normal        | 5MHz       | 350µA                       |  |  |  |  |
| Radio on 5MHz |            | 1.6mA                       |  |  |  |  |
| Low power     | 78kHz      | 200µA                       |  |  |  |  |

Table 1.1: SCµM 3C current draw in different operating states [3].

<sup>&</sup>lt;sup>1</sup>Better performance was seen in practice with a slightly higher reference voltage.

| Domain         | Operating Voltage |            | Amplifier |        |                 | Sub-Blocks |                             |
|----------------|-------------------|------------|-----------|--------|-----------------|------------|-----------------------------|
|                |                   |            |           | Qui    | iescent Current |            |                             |
|                | Min               | Nom        | Max       | Min    | Nom             | Max        |                             |
| Digital (VDDD) |                   | 1V         |           |        | 4.763µA         |            | Cortex, RAM                 |
| Auxiliary      |                   | 1V         |           |        | 4.763µA         |            | Auxiliary Digital Circuits, |
|                |                   |            |           |        |                 |            | GPIO Levelshifters          |
| Always On      | 0.8V              |            | 1.2V      | 7.12µA |                 | 7.24µA     | Osc. DAC,                   |
|                |                   |            |           |        |                 |            | Optical Receiver            |
| Sensor ADC     | 0.8V              |            | 1.2V      | 7.16µA |                 | 7.16µA     | ADC                         |
| Radio IF       | 0.8V              | $0.8V^{1}$ | 1.2V      | 4.85µA | 4.85µA          | 5.32µA     | Radio IF                    |
| Radio LO       |                   | 0.8V       |           |        | 5.98µA          |            | Radio LO                    |
| Radio PA       |                   | 0.8V       |           |        | 4.84µA          |            | Radio TX PA                 |
| Radio Divider  |                   | 0.8V       |           |        | 760nA           |            | Radio Divider               |

Table 1.2: SCµM power domains [2].



Figure 1.1: LDO with PMOS series device.

Combined, the regulators consume  $40.5\mu$ A, which is more than 10% and 20% of the total operating current in the normal and low power states respectively (Table 1.1). Even in the radio on state, this current is not negligible. Disabling some LDOs may reduce this current consumption, but the LDO quiescent current is still high compared to the results seen in Section 2.4.

The dominant pole for the radio's IF, LO, and PA LDOs was placed at the LDO output to reduce the high frequency noise at the output and accommodate the sensitivity of the attached circuits. The rest of the LDOs have the dominant pole at the output of the amplifier. A dominant amplifier pole can lead to lower current consumption as discussed in Sections 2.2 and 2.3. The radio divider LDO does clearly have lower current than the radio LDOs with dominant output poles, but there is a lot of room for improvement with the other LDOs in terms of power efficiency. The current consumption of the digital, auxiliary, always on, and ADC LDOs should be much closer to that of the radio divider with the right design, if not even lower. With the varying requirements for nominal output voltages and currents on each LDO, and the sensitivity of the radio LDOs excepting the divider LDO, the Berkeley Analog Generator (BAG), discussed in the next chapter, becomes a useful tool for efficiently reducing power consumption and catering to the specifications of each LDO.

For SC $\mu$ M, reducing leakage current is a very important goal. For any application where the power supply is coming from a battery and not a lab bench, any reduction in power will allow SC $\mu$ M to run for a longer time, increasing the scope of potential applications.

## Chapter 2

## LDO Design with BAG

### 2.1 Berkeley Analog Generator (BAG)

The Berkeley Analog Generator (BAG) [1] is in infrastructure that supports a collection of process-independent design scripts which, when given a set of input parameters, will automatically design a circuit to meet those parameters down to the transistor level. These scripts can make the design process of a circuit block significantly faster and have the potential to design more efficient circuit blocks than would practically be created by hand. This chapter discusses a BAG script in BAG2 created to design an LDO optimized for low power consumption in any process, and potential applications for these designs.

This chapter describes the process the design script in Appendix D uses to generate LDOs and provides examples of its results. The LDO design script has been validated in both the TSMC 65nm LP and TSMC 28nm processes. The BAG framework and scripts used are process-independent, but the process information for the technologies used is protected under nondisclosure agreements. All process-specific information is located on private servers hosted by the Berkeley Wireless Research Center (BWRC). The process information for TSMC 65nm LP and TSMC 28nm is included in the library paths of the respective BAG workspace repositories described in Appendix A. For non-BWRC readers, a template for establishing the correct hooks can be found at:

https://github.com/ucb-art/BAG2\_cds\_ff\_mpt.

### 2.2 LDO Design

The performance parameters this report uses to design an LDO are the static output error, load regulation, noise rejection, and stability. Physical parameters used to determine transistor sizings and bias points are the load voltage, load current draw, load capacitance, power supply, and amplifier reference current. For this report and the associated design script, a 5T amplifier topology with an NMOS input pair was used. All diagrams in this section and LDOs designed with BAG use a PMOS series device. Although an NMOS series device is possible, it may be difficult to keep in saturation unless the output common mode of the amplifier is near the supply voltage. This issue is discussed in more detail in Section 2.3.

In this circuit, the two main poles are at the output of the amplifier and at the output of the LDO. Although there are other capacitors that contribute to the frequency response of the LDO, for the purposes of this section the amplifier and series device are assumed to only have a single pole each. There is also a zero created by the capacitor (a combination of explicit and parasitic) between the amplifier output of the amplifier and the LDO output with the effective resistance it sees at  $1/g_{m,ser}$  considered in this section. The zero may affect the phase margin at higher values of  $C_{amp}$  or higher gain from the series device, but the phase margin, derived from the loop gain in Figure 2.1 and given by Equation 2.2, is primarily determined by how far away the amplier and LDO output poles are. The unity gain frequency  $\omega_0$  of the loop gain in Equation 2.1 is affected by the DC gain as well as the poles and zeros.

$$LoopGain = \frac{A_{amp}A_{ser}(1+\frac{j\omega}{\omega_z})}{(1+\frac{j\omega}{\omega_{amp}})(1+\frac{j\omega}{\omega_{out}})}$$
(2.1)

$$PM = 180^{\circ} + \arctan\left(\frac{\omega_0}{\omega_z}\right) - \arctan\left(\frac{\omega_0}{\omega_{amp}}\right) - \arctan\left(\frac{\omega_0}{\omega_{out}}\right)$$
(2.2)

An unknown load capacitance can pose challenges for ensuring stability, for example if the load capacitance is higher than expected and the amplifier pole is dominant, the poles in Equation 2.1 move closer together, leading to instability if they become too close. This problem and possible solutions are discussed in more detail in Section 2.3.

After choosing a bias point for the gate of the series device (any voltage that keeps the transistor in saturation), the bias points for a 5T differential amplifier topology as shown in Figure 2.4 are entirely determined assuming a constant V<sup>\*</sup> for the devices in the amplifier. The BAG LDO design script makes this assumption to simplify the choices it makes, but setting the amplifier input pair's V<sup>\*</sup> higher than the amplifier load pair's V<sup>\*</sup> improves noise and matching.

Modeling the circuit as a block diagram like in Figure 2.1, the relationship between the reference and the output of the LDO with a PMOS series is given by Equation 2.3, and the corresponding static error, expressed as a unitless fraction between the difference between the reference and output voltages and the reference voltage throughout this report, is given by Equation 2.4 where  $A_{ser} = g_{m,ser}r_{o,ser} > 0$ .

$$A_{amp}(-A_{ser})(V_{out} - V_{ref}) = V_{out}$$
$$\frac{V_{out}}{V_{ref}} = \frac{A_{amp}A_{ser}}{A_{amp}A_{ser} + 1}$$
(2.3)

$$\mathcal{E}_s = 1 - \frac{V_{out}}{V_{ref}} = \frac{1}{A_{amp}A_{ser} + 1} \tag{2.4}$$



Figure 2.1: Block diagram of LDO feedback loop for a PMOS series device.



Figure 2.2: Block diagram of LDO power supply gain loop for a PMOS series device.

For high amplifier and series device gains, the static error in Equation 2.4 will be very small and the actual static error will be dominated by the device offsets of the amplifier devices. The LDO design script in Appendix D does not account for this in the approximations it makes.

Modifying Figure 2.1 slightly to find the gain from the power supply to the output through both the amplifier and the series device produces Figure 2.2, which in combination with Equation 2.3 can be used to find the PSRR as in Equation 2.5. For a 5T differential amplifier,  $A_{pow}$  approaches 1 as the tail resistance increases.

$$-A_{ser}(A_{amp}V_{out} + A_{pow}V_{pow}) + A_{ser}V_{pow} = V_{out}$$

$$PSRR = \frac{V_{pow}}{V_{out}} \cdot \frac{V_{out}}{V_{ref}} = \frac{A_{amp}A_{ser} + 1}{A_{ser}(1 - A_{pow})} \cdot \frac{A_{amp}A_{ser}}{A_{amp}A_{ser} + 1} = \frac{A_{amp}}{1 - A_{pow}}$$
(2.5)

Considering the amplifier output pole as a part of  $A_{amp}$  and the LDO output pole as a part of  $A_{ser}$  and that  $A_{pow}$  is relatively independent of frequency near the amplifier and



Figure 2.3: Small signal model for LDO load regulation calculation at low frequency with a PMOS series device.

output poles, the dominant pole and bandwidth of the PSRR equation is the amplifier output pole.

To calculate load regulation for a peak load current variation of 10% from the nominal value, which is expressed as a unitless fraction between the resulting peak to peak regulator output voltage and the average regulator output voltage, the small signal model with an abstracted amplifier in Figure 2.3 provides Equation 2.6.

$$v_{g,ser} = A_{amp}v_{out}$$

$$\frac{v_{out}}{r_{o,ser}} = i_{load} - g_m v_{g,ser} = i_{load} - g_m A_{amp}v_{out}$$

$$v_{out} = \frac{r_{o,ser}}{1 + A_{amp}A_{ser}}i_{load}$$

$$LoadReg = \frac{\Delta v_{out}}{V_{out}} = \frac{\Delta i_{load}r_{o,ser}}{V_{out}(1 + A_{amp}A_{ser})} = \frac{0.2 I_{load}r_{o,ser}}{V_{out}(1 + A_{amp}A_{ser})}$$
(2.6)

From these equations, a higher amplifier and series device gain improves static error, PSRR, and load regulation. A lower series device output resistance also improves the load regulation, but shorter channel devices also have a lower gain. For  $A_{amp}A_{ser} >> 1$ , the load regulation equation becomes dependent on  $1/g_{m,ser}$ . A shorter channel length also decreases the capacitance at the load of the amplifier, bringing the PSRR bandwidth higher. Because the amplifier pole sees the output resistance of the amplifier, larger devices and smaller effective widths will decrease the amplifier pole frequency and PSRR bandwidth, and decrease the power consumed in the amplifier. The PSRR will also be increased by the higher gain from longer channel devices in the amplifier pole frequency and PSRR bandwidth, and larger effective widths will increase the amplifier pole frequency and PSRR bandwidth, and increase the power consumed in the amplifier pole frequency and PSRR bandwidth, and increase the power consumed in the amplifier pole frequency and PSRR bandwidth, and increase the power consumed in the amplifier pole frequency and PSRR bandwidth, and increase the power consumed in the amplifier pole frequency and PSRR bandwidth, and increase the power consumed in the amplifier pole frequency and PSRR bandwidth, and increase the power consumed in the amplifier pole frequency and PSRR bandwidth, and increase the power consumed in the amplifier. The PSRR will also decrease with shorter channel amplifier devices.



Figure 2.4: LDO with 5T differential amplifier topology and PMOS series device.

### 2.3 Script Design Approach

The file regulator\_ldo\_series.py defines bag2\_analog\_\_regulator\_ldo\_series\_dsn as a subclass of the DesignModule class, which given a set of input parameters described in Table 2.1 will design an LDO with either an n-type or p-type series device and a 5T differential amplifier with an n-type input pair for the amplifier as shown in Figure 2.4. See Appendix C for an example input YAML file, and Appendix D for the script itself. Although this script has the option of using an NMOS series device, using a PMOS is recommended. Unless the supply voltage is high or the device thresholds are very low, using an NMOS series device may be difficult. The minimum series device gate voltage to keep the device out of the subthreshold region is the device's threshold voltage plus the regulator output voltage, which may need to exceed the supply voltage to provide the nominal load current. A different amplifier topology with a greater output swing would be more suited to using an NMOS series device.

The inputs to this script are listed with descriptions in Table 2.1. The output is a single dictionary, containing more dictionaries listed in Table 2.2 of the component sizes of the particular design.

The first part of the LDO that the code designs is the series device. Because the current is known, the drain and source voltages are known (the supply voltage and the output voltage, depending on what type of device it is), and the channel length is known, the only unknowns are the device's effective width and gate voltage. By setting the gate voltage at some value between the bounds that keep all devices in saturation, the effective width is easily determined by a ratio of the total load current to the device current at the base width. Since the transistor data is simulated with a fixed finger width, this code assumes that the

| Parameter     | Description                                                              |
|---------------|--------------------------------------------------------------------------|
| specfile_dict | Transistor database spec file names for each device                      |
| ser_type      | n or p for type of series device                                         |
| vdd           | Supply voltage in volts                                                  |
| vout          | Reference voltage to regulate the output to                              |
| iload         | Bias current of series device, in amperes                                |
| iref          | Reference current for amplifier biasing, in amperes                      |
| iampmax       | Maximum amplifier current, in amperes                                    |
| cload         | Load capacitance from the output of the LDO to ground                    |
| cdecap        | Maximum additional capacitance added to circuit                          |
| rsource       | Resistance from the power supply, in ohms                                |
| err           | Maximum percent static error at output (as decimal)                      |
| psrr          | Minimum power supply rejection ratio (dB, 20*log10(dVdd/dVout))          |
| psrr_fbw      | Minimum bandwidth for power supply rejection roll-off                    |
| pm            | Minimum phase margin for the large feedback loop, in degrees             |
| loadreg       | Maximum absolute change in output voltage given change in output current |
| load_pole     | True to ensure dominant pole is at the<br>regulator output               |
| v_res         | Resolution of voltage bias point sweeps, in volts                        |
| sim_env       | Simulation environment                                                   |
| l_dict        | Transistor channel length dictionary                                     |
| th_dict       | Transistor flavor dictionary                                             |

Table 2.1: LDO design script input parameters.

| Parameter | Description                                                       |
|-----------|-------------------------------------------------------------------|
| w_dict    | Dictionary of transistor channel width per finger for each device |
| l_dict    | Dictionary of transistor channel length for each device           |
| nf_dict   | Dictionary of transistor number of fingers for each device        |
| th_dict   | Dictionary of transistor flavor for each device                   |
| type_dict | Dictionary of transistor type, n or p, for each device            |
| cap_dict  | Dictionary of values of $C_{amp}$ and $C_{load}$                  |

Table 2.2: LDO design script output dictionary keys.

device currents, capacitances, and transconductances scale linearly with increased effective width. Additionally, without any information on minimum or maximum finger widths of the devices used, this code assumes that the base finger width is the minimum value, and twice that width is the maximum finger width. This assumption also ensures that any effective width greater than the minimum finger width is possible, with accuracy of device matching only limited to the how well the model matches the devices and what scale of resolution is possible for fabrication. The class function resize\_op(op,wm) takes in a dictionary of transistor parameters and scales them up by the factor wm that the effective width increases by. In pseudocode, this first layer of the design is:

best LDO design  $\leftarrow$  None,  $I_{amp,best} = \infty$ for  $V_{g,ser}$  where  $M_{ser}$  in saturation do size  $M_{ser}$  for  $I_{load}$ design amplifier(specs) if  $I_{amp,new} < I_{amp,best}$  then best LDO design  $\leftarrow$  new LDO design,  $I_{amp,new}$ end if end for

return best LDO design

To simplify the calculations, all amplifier stage devices ( $M_{load,p}$ ,  $M_{load,n}$ ,  $M_{in,p}$ ,  $M_{in,n}$ , and  $M_{tail}$ ) are assumed to have the same V<sup>\*</sup>. With the series device gate voltage set, the output common mode voltage of the amplifier is also set. Because the gate and drain of the amplifier load pair are tied together on  $M_{load,p}$ , all bias voltages for the amplifier devices are set by V<sup>\*</sup>. The only parameter that is not fixed for these devices at this point is the effective width, which will be determined later.

The drains of  $M_{in,p}$  and  $M_{in,n}$  are tied to the series device gate voltage as previously set in the sweep of the series device gate, and the gates are tied to  $V_{ref}$ . V\* is already determined for the devices in this amplifier stage, so a sweep of the input pair source voltage will determine what value will best match their V\* with the load devices.

The only remaining unknown bias point is the tail device gate, which can be determined through a sweep to match V<sup>\*</sup> as with the input devices. Anticipating that an LDO produced by this script will be fabricated and therefore require layout, this script also sizes  $M_{mir}$  to ensure that they have the same finger width to promote better matching between devices. Also with layout in mind, each device is later sized to have an even number of fingers so a common centroid pattern is possible. In the case where the minimum number of fingers and  $M_{tail}$  is at its minimum, but the fingers will not fit in a single row given the area prescribed for the LDO on a chip, the script can easily be modified to set a new guaranteed factor of the number of fingers for a device. For the  $M_{tail}$ , line 156 in the code in Appendix D, which guarantees an even number of fingers, can be changed from:

```
156 nf_tail = int(2*max(((2*op_load['ibias'])//Id_tail)+1,((2*op_in['ibias'
])//Id_tail)+1))
```

to the following to guarantee that the number of fingers is a multiple of four:

```
156 nf_tail = int(4*max(((2*op_load['ibias'])//Id_tail)+1,((2*op_in['ibias'
])//Id_tail)+1))
```

For  $M_{mir}$ , lines 147 through 149 can be changed from:

```
147 m_mir = iref/(2*op_mir['ibias'])
148 wm_mir = (m_mir%1 + 1)
149 nf_mir = 2*int(m_mir)
```

to the following to guarantee that the number of fingers is a multiple of four:

```
147 m_mir = iref/(4*op_mir['ibias'])
148 wm_mir = (m_mir%1 + 1)
149 nf_mir = 4*int(m_mir)
```

For  $M_{\text{load},p}$ ,  $M_{\text{load},n}$ ,  $M_{\text{in},p}$ , and  $M_{\text{in},n}$ , the guaranteed factor can be modified similarly to line 149 on lines 163 and 167 for the load devices and input devices respectively, additionally changing the multipliers within the max() function in line 153 from '2' to the desired factors. While these modifications will always produce a potential sizing if the load, input, or tail device factors are changed, increasing the guaranteed factor for the number of mirror device fingers may increase the total current at a given bias point past the value of *iref*, and that bias point will be ignored.

With the bias current *iref* as an input, the effective width of  $M_{mir}$  is uniquely determined (as long as *iref* is larger than twice the base finger width current at the bias point). All of the amplifier devices excluding the mirror are then set at minimum effective width based on which device has the largest base current, accounting for the fact that the tail device sinks the current of both input and load devices. Up to this point, the amplifier design in pseudocode (with ranges on the for loops to keep devices in saturation) is:

```
V^*_{amp} \leftarrow V^*_{load}
for V \in [0, V_{ref} - V_{th,in}] do
   if V_{in}^* = V_{amp}^* then
       V_{s,in} \leftarrow V
   end if
end for
for V \in [V_{th,tail}, V_{s,in} + V_{th,tail}] do
   if V_{in}^* = V_{amp}^* then
       V_{g,tail} \leftarrow V
   end if
end for
size M_{mir} for I_{ref}
W_{finger,tail} \leftarrow W_{finger,mir}
I_{tail} \leftarrow 2 * max(I_{tail, finger}, 2 * I_{in, finger}, 2 * I_{load, finger})
size M_{tail} for I_{tail}
size M_{in}, M_{load} for I_{tail}/2
```

The final part of the amplifier design checks that specs are met and adds capacitors to make the feedback loop stable. The specs that are most significantly affected by adding capacitors and sizing up the amplifier devices are the phase margin and the power supply rejection ratio bandwidth. There are two places to put the dominant pole in this circuit: at the output of the amplifier and at the output of the LDO. The primary advantages of placing the dominant pole at the output of the amplifier are low power and a lower value of explicit capacitance added to the circuit, which can take up a lot of space on a chip. Power consumption is generally lower with a dominant amplifier pole because the amplifier output resistance is highest with minimally sized devices, and a higher resistance means a lower pole, even with less capacitance than may be necessary for a dominant pole at the output of the LDO. The disadvantages of a dominant amplifier pole are that the PSRR bandwidth follows this pole closely, so a lower pole that is necessary for stability kills higher frequency noise rejection, and if the estimate of the LDO load capacitance is too low, the poles will be closer than expected and may lead to instability. In case the LDO load capacitance is unknown, one of the inputs to the design script load\_pole ensures that the dominant pole is placed at the LDO output and the attempt to set it at the amplifier output is skipped.

If a load pole at the amplifier output is allowed, the script begins by checking if the LDO meets the specs without any additional capacitance, then increases the capacitor tied to the amplifier output until either all specs are met, the PSRR bandwidth drops below the minimum spec, or the maximum allowable capacitance is added. In the case where the spec is not met or the dominant pole is placed at the output of the LDO, the script adds the maximum allowable capacitance to the LDO output to bring that pole as low as possible. This has little effect on the PSRR bandwidth, which is still tied to the amplifier output pole. If the phase margin and PSRR bandwidth are still too low, the script begins increasing the effective width of the amplifier devices to decrease the output resistance and drive the amplifier pole higher. This excludes the tail mirror device because the reference current it takes is fixed and does not affect the pole.

With the tail finger width fixed so it matches the reference current mirror, the only way to increase its effective width is by adding fingers. This script does so two fingers at a time to promote symmetry in layout as mentioned before. If the desired factor for the tail device is a number other than two, changing the number of fingers added to the number of tail fingers nf\_tail on line 237 to that factor will ensure the proper number of fingers. At this point in the loop, the tail device is resized but the amplifier load and input devices also have to be resized relative to the tail. Simply adding fingers to these devices will not be enough to match their currents because they don't necessarily have the same finger widths as the tail or each other, so they are resized relative to the new tail bias current. This process continues until either a solution is found, or the maximum amplifier current is exceeded. The pseudocode for upsizing transistors and ensuring stability is:

if specs met then return amplifier design end if if not load\_pole then  $C_{load} \leftarrow 0$ for  $C_{amp} < \max \operatorname{decap} \operatorname{do}$ if specs met then return amplifier design end if end for end if  $C_{load} \leftarrow C_{max}$  $C_{amp} \leftarrow 0$ while  $I_{tail} < I_{max}$  do if specs met then return amplifier design end if  $nf_tail \leftarrow nf_tail + 2$ size  $M_{in}$ ,  $M_{load}$  for  $I_{tail}/2$ end while

return no design

Once the amplifier at a bias point is designed, the function op\_compare(op1,op2) is called in meet\_spec(\*\*params) to determine whether to keep or ignore the current iteration. This function is set to compare only the amplifier bias current but can be changed to optimize for any other performance parameter specified in the design input file. Line 335 also sets the maximum amplifier bias current to the bias current of the best amplifier design found up to that point. This line may be slightly redundant considering that the previous line calling op\_compare(op1,op2) also compares the amplifier bias currents and will always choose the new design because of the assignment in line 335, but calling op\_compare(op1,op2) allows for the optimized parameter to be swapped out more easily, and the assignment of the maximum bias current may shorten the run time of the while loop on line 213 in the amplifier design if the bias current is already greater than will be considered for the final design.

This script utilizes the LTICircuit class as defined in BAG\_framework/data/lti.py, which can use the transistor parameters obtained for the design and solve the small signal equations of the circuit model when the nodes each component connects to are defined. The resulting circuit model can be used to calculate the transfer function between any two nodes with voltage or current inputs, giving the gain and bandwidth of a particular transfer function. Using this class to calculate the circuit performance parameters is advantageous because it allows for more rigorous calculations of small signal circuits (i.e. including all parasitic capacitors) and can easily be used to model more complicated amplifier topologies not require a license to use as other circuit modeling software like Cadence Virtuoso does.

| Reference Voltage | 1V              |
|-------------------|-----------------|
| Load Current      | 1mA             |
| Static Error      | -2.756m         |
| Amp Current       | 4.763µA         |
| Phase Margin      | 85.62°          |
| PSRR              | 43.87dB         |
| PSRR Bandwidth    | 461.9Hz         |
| Load Regulation   | 96.50µ          |
| Amp Capacitor     | 3.60pF          |
| Load Capacitor    | 452pF           |
| Total Gate Area   | $756.6 \mu m^2$ |

Table 2.3: SCµM 3C auxiliary and digital LDO performance.

### 2.4 Comparison to Hand-Designed LDOs

#### SCµM Digital and Auxiliary LDOs

SCµM is generally operated with an auxiliary and digital LDO reference and output voltage of 1V and a battery voltage of 1.5V. Both of these LDOs, which use the same schematic, have an PMOS series device. Table 1.1 lists the three main operating points of SCµM and their total operating current draws. With these numbers, a reasonable nominal current to design for seems to be about 1mA based on the increase in current when the radio is on.

The load capacitance of both the auxiliary and digital LDOs on SC $\mu$ M contributed by the layout and other circuits is unknown. For this application, PSRR bandwidth is not as important because the regulator loads are entirely digital, so the dominant pole will be placed at the output of the amplifier as it is on SC $\mu$ M 3C. With this in mind, what is necessary to generate a viable LDO for SC $\mu$ M is an upper bound on that unknown load capacitance. For this thesis, the assumed load capacitance value will be 100pF. If a better estimate is determined through extraction or measurement, the BAG design script can be rerun with a higher load capacitance by changing a single line in the input YAML file.

Tables 2.4 through 2.9 and Figures 2.6 through 2.8 show the inputs (in the format the design script would read them) and respective output performances of the LDO design script for various series device and amplifier channel lengths. There is a slight variation between the performance calculated by the script and the simulated performance, but generally the two are fairly close. The biggest issue with these designs is the bias points of the amplifier. The script uses the function estimate\_vth(db,is\_nch,lch,vgs,vbs) to estimate the threshold voltage of a device given a bias point, but is not very accurate. This function uses the the device's V\* and  $V_{gs}$  to estimate the threshold, assuming a quadratic model for vgs/lch <  $1V/\mu m$  and a linear model otherwise, which varies dramatically as  $V_{gs}$  changes. Due to the



Figure 2.5: SCµM 3C auxiliary and digital LDO transient response at 5MHz.

inaccuracies in the function it is difficult to ensure that all devices are close enough to the threshold to achieve high gain and low current consumption but are still in saturation. Even using the estimated threshold of the amplifier load (which is higher than the series device for the designs in the tables because of the channel length), the PMOS devices generally have a lower V\* than the NMOS, so the load devices can be in saturation with the input and tail devices pushed subthreshold. Of the LDOs listed in Tables 2.4 through 2.9, the designs with all devices in saturation are the ones with an amplifier channel length of 5µm. The remaining designs with amplifier channel lengths of 500nm and 1µm have the input and tail devices subthreshold (as determined by simulation), as well as the load devices for the designs with 280nm channel length series devices. On average, the subthreshold devices only had  $V_{gs}$  lower than the threshold by about 71mV, with the greatest difference being about 150mV.

The 5µm amplifier channel length designs are each better than the auxiliary and digital LDOs currently on SCµM for each parameter, with the exception of the phase margin, PSRR bandwidth, and peak to peak of the transient plots at 5MHz. Although the phase margin is slightly lower for these designs, they are all still well within the bounds for stability. The PSRR bandwidths are low, especially with larger series devices, but the PSRR bandwidth of SCµM is also only 461.9Hz, less than twice the PSRR bandwidth for the design with a 280nm series device channel length. Given that this script was designed to optimize for power, this design was very successful. Replacing the auxiliary and digital LDOs on SCµM with the 5µm amplifier channel length and 280nm series device channel length design would decrease the amplifier current consumption by 94.8% from 4.763µA to 248.1nA at a load current draw of 1mA. Without even considering the large resistors and capacitors needed for the LDOs presently on SCµM, this design only takes up 10.3% of the device gate area those LDOs need. On top of this, the design script runs much faster than it takes to design an LDO by hand, with a total runtime of about 80 seconds.

If having the amplifier devices slightly below the threshold is not an issue for matching in the particular process all nine designs listed in Tables 2.4 through 2.9 are viable options with significantly lower amplifier current consumption, the highest being  $1.432\mu$ A with an amplifier channel length of 500nm and device channel length of 320nm. Across every single option, the only specs that are worse than the parameters in Table 2.3 are again the phase margin and PSRR bandwidth, but all options are stable and some even have a higher PSRR bandwidth. The only point of concern is the slightly lower phase margin of the design with an amplifier channel length of 500nm and a series device channel length of 300nm in combination with the unknown parasitic load capacitance on SC $\mu$ M, but if that value is discovered to be more than 100pF, the PSRR bandwidth is high enough that adding a very small capacitor at the amplifier output will likely resolve the issue of stability without bringing the bandwidth down too far compared to the LDOs on SC $\mu$ M.

Each transient simulation in Figures 2.6 through 2.8 has a load current that switches between 100µA and 1.1mA at 5MHz, which is an overestimate of how quickly the current will change between two extreme values. At this frequency and magnitude, the output voltage only changes by about 300mVpp for amplifier channel lengths on 500nm and 1µm, and 200mVpp for an amplifier channel length of 5µm. The digital and auxiliary LDOs on SCµM only have a voltage variation of about 100mVpp, which is due to the larger explicit capacitor at the load.

While this data is only for the auxiliary and digital LDOs, with the right set of parameters the design script can also generate a new schematic for each of the other LDOs on  $SC\mu M$ . There has been discussion of moving  $SC\mu M$  to a different technology process. Given reasonable specs, this script can generate the LDO designs for that project with ease, leaving only the verification simulations and layout to be done manually.

#### EE290C: 28nm SoC for IoT

For the Spring 2021 iteration of EE290C at UC Berkeley, the students collectively designed and taped out a system-on-a-chip (SoC) for internet-of-things (IoT) applications, with an on-chip CPU, radio, and Bluetooth Low Energy (BLE) and Advanced Encryption Standard (AES) compatibility. This chip design also includes two LDOs for the auxiliary and digital domains both of which use the same amplifier designs and device sizings. These LDOs were designed without the use of BAG. The simulated specs across temperature and process corners are shown in Table 2.10.

These LDOs were designed with 400nm channel length devices for the amplifier devices and a channel length of 150nm for the series device. Due to the nature of the course, these LDOs were designed at the same time as the rest of the chip, so almost no information about the load was known until about halfway through the design process. Even after the current draw from the analog and digital halves of the chip were estimated at 10mA or lower, the load capacitance was still unknown. Because of this, and a desire for a higher frequency power supply noise rejection bandwidth, the dominant pole was placed at the output of the LDO to keep the amplifier output pole and PSRR bandwidth higher and ensure the stability of the feedback loop. Although the phase margin for this design is very low, the simulated performance does not include the additional capacitance from the chip's power grid and circuit blocks attached to it. Because the LDO output is the dominant pole, that additional parasitic capacitance will only increase the phase margin. Through simulation, it also appears that the phase margin increases if the load does not draw the full 10mA allotted to each regulator.

With the same channel lengths for each device as the hand-designed LDO, the results of running the generator script with the parameters listed in Table 2.11 are shown in Table 2.12 (for a series device channel length of 150nm). For this set of inputs, the PSRR and phase margin improved significantly, with a slight decrease in required gate area. The static error has notably gotten worse, but the other parameters have stayed relatively similar. Each generated design of this LDO in Tables 2.11 through 2.14 has all devices in saturation, unlike the designs for SCµM in the 65nm process. Of the options, targeting low power and higher PSRR bandwidth, the 1µm amplifier channel length and 150nm series device channel length

<sup>&</sup>lt;sup>1</sup>Text in YAML file is the corner name instead of "nominal."

| Spec     | Value   |
|----------|---------|
| ser_type | р       |
| vdd      | 1.5     |
| vout     | 1       |
| iload    | 1e-3    |
| iref     | 500e-9  |
| iamp_max | 100e-6  |
| cload    | 100e-12 |
| cdecap   | 100e-12 |
| rsource  | 0       |

| Spec        | Value                |
|-------------|----------------------|
| err         | 1e-3                 |
| psrr        | 50                   |
| psrr_fbw    | 1e3                  |
| pm          | 60                   |
| loadreg     | 1e-3                 |
| load_pole   | False                |
| v_res       | 10e-3                |
| $sim_{env}$ | nominal <sup>1</sup> |

| Device   | Base Finger Width | Channel Type | Device Type |  |
|----------|-------------------|--------------|-------------|--|
| amp_in   | 500nm             | NMOS         | Input       |  |
| amp_load | 500nm             | PMOS         | Input       |  |
| amp_tail | 500nm             | NMOS         | Input       |  |
| amp_mir  | 500nm             | NMOS         | Input       |  |
| ser      | 500nm             | PMOS         | Input       |  |

Table 2.4: SCµM generated LDO input parameters for amplifier channel length of 500nm.

| $L_{ser}$       | 280nm           | 280nm               |                     | 300nm               |                     | 320nm               |                |
|-----------------|-----------------|---------------------|---------------------|---------------------|---------------------|---------------------|----------------|
| Data Source     | SCµM            | Script              | Sim.                | Script              | Sim.                | Script              | Sim.           |
| Static Error    | -2.756m         | 672.8µ              | 123.9µ              | 528.9µ              | $-225.5\mu$         | 462.4µ              | $-266.3\mu$    |
| Amp Current     | 4.763µA         | 465.3nA             | 297.8nA             | 878.9nA             | 727.4nA             | 1.462µA             | 1.432µA        |
| Phase Margin    | 85.62°          | 61.88°              | 69.07°              | 62.94°              | 60.53°              | 82.33°              | 80.06°         |
| PSRR            | 43.87dB         | $63.37 \mathrm{dB}$ | $60.77 \mathrm{dB}$ | $66.08 \mathrm{dB}$ | $68.31 \mathrm{dB}$ | $68.97 \mathrm{dB}$ | 71.48dB        |
| PSRR Bandwidth  | 461.9Hz         | 2.073kHz            | 3.040kHz            | 1.602kHz            | 2.116kHz            | 1.708kHz            | 717.1Hz        |
| Load Regulation | 96.50µ          | 170.4µ              | 132.8µ              | 162.9µ              | 10.95µ              | 10.50µ              | 11.87µ         |
| Amp Capacitor   | 3.60pF          | 0F                  | 0F                  | 0F                  | 0F                  | $625.7 \mathrm{fF}$ | 625.7fF        |
| Load Capacitor  | 452pF           | 0F                  | 0F                  | 0F                  | 0F                  | 0F                  | 0F             |
| Total Gate Area | $756.6 \mu m^2$ | $112.3 \mu m^2$     | $112.3 \mu m^2$     | $153.8 \mu m^2$     | $153.8 \mu m^2$     | $137.2 \mu m^2$     | $137.2\mu m^2$ |

Table 2.5: SCµM generated LDO performance for amplifier channel length of 500nm.



Figure 2.6: SCµM generated LDO transient for amplifier channel length of 500nm at 5MHz.

| Spec     | Value   |
|----------|---------|
| ser_type | р       |
| vdd      | 1.5     |
| vout     | 1       |
| iload    | 1e-3    |
| iref     | 500e-9  |
| iamp_max | 100e-6  |
| cload    | 100e-12 |
| cdecap   | 100e-12 |
| rsource  | 0       |

| Value                |
|----------------------|
| 1e-3                 |
| 50                   |
| 100                  |
| 60                   |
| 1e-3                 |
| False                |
| 10e-3                |
| nominal <sup>1</sup> |
|                      |

| Device   | Base Finger Width | Channel Type | Device Type |
|----------|-------------------|--------------|-------------|
| amp_in   | 500nm             | NMOS         | Input       |
| amp_load | 500nm             | PMOS         | Input       |
| amp_tail | 500nm             | NMOS         | Input       |
| amp_mir  | 500nm             | NMOS         | Input       |
| ser      | 500nm             | PMOS         | Input       |

Table 2.6: SCµM generated LDO input parameters for amplifier channel length of 1µm.

| $L_{ser}$       | 280nm           | 280                 | nm                  | 300nm               |                 | 320nm           |                |
|-----------------|-----------------|---------------------|---------------------|---------------------|-----------------|-----------------|----------------|
| Data Source     | SCµM            | Script              | Sim.                | Script              | Sim.            | Script          | Sim.           |
| Static Error    | -2.756m         | 513.2µ              | 116.8µ              | 294.8µ              | $-206.1\mu$     | 233.8µ          | $-274.5\mu$    |
| Amp Current     | 4.763µA         | 310.2nA             | 198.4nA             | 338.0nA             | 327.0nA         | 499.1nA         | 488.9nA        |
| Phase Margin    | 85.62°          | 70.80°              | 74.33°              | 77.64°              | 74.74°          | 79.56°          | 75.7°          |
| PSRR            | 43.87dB         | $62.27 \mathrm{dB}$ | $65.90 \mathrm{dB}$ | $66.86 \mathrm{dB}$ | 82.82dB         | 68.37dB         | 87.03dB        |
| PSRR Bandwidth  | 461.9Hz         | 1.252kHz            | 1.031kHz            | 529.2Hz             | 163.7Hz         | 404.1Hz         | 159.7Hz        |
| Load Regulation | 96.50µ          | 130.0µ              | 5.328µ              | 74.04µ              | 5.170µ          | 60.21µ          | 4.829µ         |
| Amp Capacitor   | 3.60pF          | 0F                  | 0F                  | 0F                  | 0F              | 0F              | 0F             |
| Load Capacitor  | 452pF           | 0F                  | 0F                  | 0F                  | 0F              | 0F              | 0F             |
| Total Gate Area | $756.6 \mu m^2$ | $122.9 \mu m^2$     | $122.9 \mu m^2$     | $157.1 \mu m^2$     | $157.1 \mu m^2$ | $233.0 \mu m^2$ | $233.0\mu m^2$ |

Table 2.7: SCµM generated LDO performance for amplifier channel length of 1µm.



Figure 2.7: SCµM generated LDO transient for amplifier channel length of 1µm at 5MHz.

| Spec     | Value   |
|----------|---------|
| ser_type | р       |
| vdd      | 1.5     |
| vout     | 1       |
| iload    | 1e-3    |
| iref     | 500e-9  |
| iamp_max | 100e-6  |
| cload    | 100e-12 |
| cdecap   | 100e-12 |
| rsource  | 0       |

| Spec        | Value                |
|-------------|----------------------|
| err         | 1e-3                 |
| psrr        | 70                   |
| psrr_fbw    | 100                  |
| pm          | 60                   |
| loadreg     | 1e-3                 |
| load_pole   | False                |
| v_res       | 10e-3                |
| $sim_{env}$ | nominal <sup>1</sup> |

| Device   | Base Finger Width | Channel Type | Device Type |
|----------|-------------------|--------------|-------------|
| amp_in   | 500nm             | NMOS         | Input       |
| amp_load | 500nm             | PMOS         | Input       |
| amp_tail | 500nm             | NMOS         | Input       |
| amp_mir  | 500nm             | NMOS         | Input       |
| ser      | 500nm             | PMOS         | Input       |

Table 2.8: SCµM generated LDO input parameters for amplifier channel length of 5µm.

| $L_{ser}$       | 280nm           | 280                 | nm                  | m 300nm         |                     | 320nm           |                |
|-----------------|-----------------|---------------------|---------------------|-----------------|---------------------|-----------------|----------------|
| Data Source     | SCµM            | Script              | Sim.                | Script          | Sim.                | Script          | Sim.           |
| Static Error    | -2.756m         | 126.7µ              | 38.46µ              | 121.3µ          | 11.21µ              | 110.6µ          | $-40.47\mu$    |
| Amp Current     | 4.763µA         | 314.1nA             | 248.1nA             | 638.1nA         | 495.8nA             | 990.4nA         | 991.1nA        |
| Phase Margin    | 85.62°          | 70.83°              | $71.38^{\circ}$     | 70.36°          | 70.35°              | 69.31°          | 66.81°         |
| PSRR            | 43.87dB         | $78.83 \mathrm{dB}$ | $75.08 \mathrm{dB}$ | 79.36dB         | $78.92 \mathrm{dB}$ | 81.42dB         | 93.80dB        |
| PSRR Bandwidth  | 461.9Hz         | 1.589kHz            | 244.6Hz             | 4.432kHz        | 199.2Hz             | 1.474kHz        | 52.13Hz        |
| Load Regulation | 96.50µ          | 47.49µ              | $2.325\mu$          | 44.67µ          | 2.117µ              | 44.67µ          | 2.242µ         |
| Amp Capacitor   | 3.60pF          | 168.9fF             | $168.9 \mathrm{fF}$ | 240.7fF         | 240.7fF             | 279.8fF         | 279.8fF        |
| Load Capacitor  | 452pF           | 0F                  | 0F                  | 0F              | 0F                  | 0F              | 0F             |
| Total Gate Area | $756.6 \mu m^2$ | $130.1 \mu m^2$     | $130.1 \mu m^2$     | $154.2 \mu m^2$ | $154.2 \mu m^2$     | $148.3 \mu m^2$ | $148.3\mu m^2$ |

Table 2.9: SCµM generated LDO performance for amplifier channel length of 5µm.



Figure 2.8: SCµM generated LDO transient for amplifier channel length of 5µm at 5MHz.



Figure 2.9: EE290C hand-designed LDO transient response at 5MHz.

|                 | Temperature (nominal <sup>1</sup> ) |                     | Corners $(27^{\circ}C)$ |                 | nominal <sup>1</sup> , $27^{\circ}C$ |                 |
|-----------------|-------------------------------------|---------------------|-------------------------|-----------------|--------------------------------------|-----------------|
| Parameter       | $0^{\circ}C$                        | $27^{\circ}C$       | $85^{\circ}C$           | Minimum         | Maximum                              | $R_{sup} = 10$  |
| Static Error    | 61.93µ                              | $-54.9\mu$          | -184.4µ                 | -1.076m         | 923.5µ                               | 260.8µ          |
| Amp Current     | 192.5µA                             | 196.4µA             | 203.8µA                 | 176.6µA         | 217.7µA                              | 192.1µA         |
| Phase Margin    | $49.57^{\circ}$                     | $50.54^{\circ}$     | 52.84°                  | 47.22°          | $55.95^{\circ}$                      | $50.35^{\circ}$ |
| PSRR            | 52.03dB                             | $52.47 \mathrm{dB}$ | 62.99dB                 | 50.17dB         | $59.25 \mathrm{dB}$                  | 49.89dB         |
| PSRR Bandwidth  | 733.8kHz                            | 663.0kHz            | 180.9kHz                | 316kHz          | 891.5kHz                             | 877.8kHz        |
| Load Regulation | 380.5µ                              | 408.1µ              | 471.5µ                  | 329.0µ          | 483.1µ                               | 488.6µ          |
| Amp Capacitor   | 0F                                  | 0F                  | 0F                      | 0F              | 0F                                   | 0F              |
| Load Capacitor  | 100nF                               | 100nF               | 100nF                   | 100nF           | 100nF                                | 100nF           |
| Total Gate Area | $208.7\mu m^2$                      | $208.7\mu m^2$      | $208.7\mu m^2$          | $208.7 \mu m^2$ | $208.7 \mu m^2$                      | $208.7 \mu m^2$ |

Table 2.10: EE290C LDO performance with 10mA load current.

design seems to be best. It has comparable power consumption, phase margin, and load regulation to the hand-designed LDO, with significant increases to both PSRR and PSRR bandwidth. The static error is a lot larger, but only amounts to about a 1mV offset from the nominal 900mV. The area is also slightly larger, but would still easily fit within the area bounds of the power block on the chip for EE290C. The transient responses of each generated LDO and the hand-designed LDO are shown in Figures 2.10 and 2.11. The chip for EE290C is designed to run at 100MHz, but the response at this frequency and the plot of the load current were too fast to show at the same scale as the startup transient due to the large output capacitor. The plots at 100MHz and 5MHz are virtually indistinguishable from each other, even between designs with different channel lengths. The main difference is the settling time of the startup transient due to the frequency of the secondary pole at the amplifier output. Although the difference in performance is not as dramatic as it is for the redesigned SCµM LDOs, the speed at which this script is able to generate this designs is still significantly faster than designing without it.

## 2.5 Approximations and Errors

The main sources of error in this script are likely from the device characterization and simplification of the model the LTICircuit functions use for calculation. The LTICircuit class assumes the canonical small signal model [4, Fig. 2.38]. Inaccuracies with this assumption produce the error seen in the design script outputs.

This script uses is the function estimate\_vth(db,is\_nch,lch,vgs,vbs) to approximate the threshold voltage for the output device, because this information is not included in the device characterization BAG uses due to the variation from linear or quadratic models in real devices. This estimation is very rough and usually inaccurate if the bias points the threshold is estimated at put the device out of saturation. To avoid this, the script estimates

| Spec     | Value  |
|----------|--------|
| ser_type | р      |
| vdd      | 1.5    |
| vout     | 0.9    |
| iload    | 10e-3  |
| iref     | 10e-6  |
| iamp_max | 500e-6 |
| cload    | 0      |
| cdecap   | 100e-9 |
| rsource  | 0      |

| Spec         | Value                        |
|--------------|------------------------------|
| err          | 0.01                         |
| psrr         | 40                           |
| psrr_fbw     | 500e3                        |
| pm           | 60                           |
| loadreg      | 10e-3                        |
| load_pole    | True                         |
| v_res        | 10e-3                        |
| $sim_{-}env$ | $\operatorname{nominal}^{1}$ |

| Device   | Base Finger Width | Channel Type | Device Type |
|----------|-------------------|--------------|-------------|
| amp_in   | 500nm             | NMOS         | Input       |
| amp_load | 500nm             | PMOS         | Input       |
| amp_tail | 270nm             | NMOS         | Input       |
| amp_mir  | 270nm             | NMOS         | Input       |
| ser      | 500nm             | PMOS         | Input       |

Table 2.11: EE290C generated LDO input parameters for amplifier channel length of 400nm.

| $L_{ser}$       | 150nm          | 150nm                  |                 | 180nm                |                 |
|-----------------|----------------|------------------------|-----------------|----------------------|-----------------|
| Data Source     | Original       | Script                 | Sim.            | Script               | Sim.            |
| Static Error    | $-54.9\mu$     | 2.523m                 | -1.848m         | 1.862m               | -2.123m         |
| Amp Current     | 196.4µA        | 251.0µA                | 258.3µA         | 393.2µA              | 405.4µA         |
| Phase Margin    | 50.54°         | 65.70°                 | $63.07^{\circ}$ | $64.48^{\circ}$      | $62.49^{\circ}$ |
| PSRR            | 52.47dB        | $59.56 \mathrm{dB}$    | 63.81dB         | $60.85 \mathrm{dB}$  | 84.72dB         |
| PSRR Bandwidth  | 663.0kHz       | 525.8kHz               | 436.7kHz        | $508.8 \mathrm{kHz}$ | 81.15kHz        |
| Load Regulation | 408.1µ         | 774.9µ                 | 432.5µ          | 694.1µ               | 400.3µ          |
| Amp Capacitor   | 0F             | 0F                     | 0F              | 0F                   | 0F              |
| Load Capacitor  | 100nF          | 100nF                  | 100nF           | 100nF                | 100nF           |
| Total Gate Area | $208.7\mu m^2$ | $185.4\mu\mathrm{m}^2$ | $185.4\mu m^2$  | $295.9 \mu m^2$      | $295.9\mu m^2$  |

Table 2.12: EE290C generated LDO performance for amplifier channel length of 400nm.



Figure 2.10: EE290C generated LDO transient for amplifier channel length of 400nm at 5MHz.

| Spec     | Value  |
|----------|--------|
| ser_type | р      |
| vdd      | 1.5    |
| vout     | 0.9    |
| iload    | 10e-3  |
| iref     | 10e-6  |
| iamp_max | 500e-6 |
| cload    | 0      |
| cdecap   | 100e-9 |
| rsource  | 0      |

| Spec         | Value       |  |
|--------------|-------------|--|
| err          | 0.01        |  |
| psrr         | 60          |  |
| psrr_fbw     | 1e6         |  |
| pm           | 60          |  |
| loadreg      | 10e-3       |  |
| load_pole    | True        |  |
| v_res        | 10e-3       |  |
| $sim_{-}env$ | $nominal^1$ |  |

| Device   | Base Finger Width | Channel Type | Device Type |  |
|----------|-------------------|--------------|-------------|--|
| amp_in   | 500nm             | NMOS         | Input       |  |
| amp_load | 500nm             | PMOS         | Input       |  |
| amp_tail | 270nm             | NMOS         | Input       |  |
| amp_mir  | 270nm             | NMOS         | Input       |  |
| ser      | ser 500nm PMOS    |              | Input       |  |

Table 2.13: EE290C generated LDO input parameters for amplifier channel length of 1µm.

| $L_{ser}$       | 150nm          | $150 \mathrm{nm}$      |                        | 180nm           |                   |
|-----------------|----------------|------------------------|------------------------|-----------------|-------------------|
| Data Source     | Original       | Script                 | Sim.                   | Script          | Sim.              |
| Static Error    | $-54.9\mu$     | 2.225m                 | 1.262m                 | 1.539m          | 489.0µ            |
| Amp Current     | 196.4µA        | 192.9µA                | 205.9µA                | 281.9µA         | 294.5 <i>u</i> µA |
| Phase Margin    | 50.54°         | $66.36^{\circ}$        | $54.90^{\circ}$        | $64.67^{\circ}$ | 54.24°            |
| PSRR            | 52.47dB        | $62.50\mathrm{dB}$     | 51.72dB                | 64.46dB         | 55.61dB           |
| PSRR Bandwidth  | 663.0kHz       | 1.039MHz               | $1.877 \mathrm{MHz}$   | 1.080MHz        | 1.434MHz          |
| Load Regulation | 408.1µ         | 963.9µ                 | 408.8µ                 | $958.4\mu$      | 393.6µ            |
| Amp Capacitor   | 0F             | $0\mathrm{F}$          | 0F                     | 0F              | 0F                |
| Load Capacitor  | 100nF          | 100nF                  | 100nF                  | 100nF           | 100nF             |
| Total Gate Area | $208.7\mu m^2$ | $235.0\mu\mathrm{m}^2$ | $235.0\mu\mathrm{m}^2$ | $302.7\mu m^2$  | $302.7\mu m^2$    |

Table 2.14: EE290C generated LDO performance for amplifier channel length of 1µm.



Figure 2.11: EE290C generated LDO transient for amplifier channel length of 1µm at 5MHz.

the threshold of the series device with its defined drain-source voltage (between the supply voltage and the output voltage) with the gate source voltage likely overestimated at half the supply voltage.

The static error is currently measured as the error caused by the non-infinite loop gain as shown in Equation 2.4. As mentioned briefly in Section 2.2, the static error calculated by this design script will likely be dominated by device offsets in the amplifier, which is not taken into account.

The PSRR is also approximated as the inverse of the gain from the power supply to the output rather than the gain from the input to the output divided by the power supply gain as described in Equation 2.5. The PSRR bandwidth would require the convolution of two lists of coefficients for both the numerator and denominator. For the purposes of this script, because the gain from the reference to the output is approximately 1 for a high loop gain, it was omitted from this calculation.

The YAML file that the specs of a completed generation are dumped into appears to have trouble with numbers such as the capacitances or device finger widths that are NumPy data types, but the data is stored properly even though the formatting is not as readable as a basic Python integer or float type is. The results are directly printed to that terminal, and if lost, can easily be generated again in very little time.

## Chapter 3 Conclusion

This report details the approach of a BAG design script to generate LDOs with real applications on chips like SC $\mu$ M. Although there are improvements to be made on this script, as it stands it is still an incredibly efficient and practical tool for LDO design. Using only the most basic LDO topology, design can be done quickly and is easily transferable between processes, allowing for fast estimates of performance and comparisons between designs, optimization goals, and technologies. With known specs for each LDO on SC $\mu$ M, it is possible to redesign lower power LDOs for each of the eight domains listed in Table 1.2 in under a day, potentially increasing their accuracy and noise reduction as well. The ability to design so many circuits in so little time is invaluable to so many projects, especially in an academic setting, allowing designers to focus more of their energy on the circuits powered by LDOs. With the necessity for power regulation on any chip, this script will undoubtedly see use in the future.

#### 3.1 Future Work

This design script does not check for the amplifier device thresholds, assuming that if the series device is in saturation the amplifier load will be as well, and then the V\* of the other amplifier devices will be set to match the load pair keeping them in saturation if they have similar V\* characteristics in the different regions of operation. If the channel length for the amplifier devices is significantly larger than the series device length, the amplifier load threshold will exceed the estimate for the series device and put all amplifier devices gate voltages subthreshold. If the script did include threshold estimates for the amplifier devices as well using the function estimate\_vth(db,is\_nch,lch,vgs,vbs), many of the tests run in Section 2.5 may not produce a solution due to the inaccuracy of the estimation function. A near-threshold biasing point improves the gain and power consumption of the amplifier significantly, and the overestimation of the threshold voltage may push devices too far from that point to meet more demanding specs. Each device that was operating subthreshold in Section 2.5 was within 100mV of the actual device threshold. Implementing a more accurate

threshold estimation or even extraction during characterization for BAG would significantly improve the performance of this design script and likely others as well.

Other features this script does not include but could be beneficial for any updated versions of this script are the frequency response of the load regulation, static error caused by device offsets, and the option of increasing the width of the amplifier devices for a pole at the amplifier output if the phase margin spec is met but the PSRR bandwidth spec is not. For the load regulation at higher frequencies, there may be peaking if the load capacitance is small, which depending on the operating frequency of the load circuits, could mean a significantly worse load regulation. This issue occurs most often if the dominant pole of the LDO is at the amplifier output. Adding estimates of the device offset contribution to static error will produce more accurate results in comparison to simulation, particularly at higher loops gains where Equation 2.4 is small. The modification to the amplifier device width for pole placement is only useful if the phase margin of a design is already higher than the spec without additional decap added, and if the PSRR spec is not met. Increasing the amplifier width increases the amplifier pole frequency, bringing the phase margin down and PSRR bandwidth up at the cost of power. It is possible that moving the pole to the output of the LDO will produce a design that meets both specs anyway and uses less power at the same time, but this may be a worthwhile modification if area for capacitors is limited and a higher PSRR bandwidth is necessary.

The final area in which this script could be expanded upon is automating layout. Approximating the area as the device gate area is easy to do by multiplying device widths and lengths. Some scripts do have the infrastructure to generate a layout from the produced design [5], which would be helpful in situations like with EE290C where the area allowed for the LDOs changed multiple times during the design process. Due to lack of setup in the processes used in this report and time, this script does not contain layout functionality, but adding it to this script will unquestionably speed up the design process further.

Appendix D describes additional issues with the power systems on SCµM that have not been resolved to date and possible routes to finding a solution.

### Bibliography

- Eric Chang et al. "BAG2: A process-portable framework for generator-based AMS circuit design". In: 2018 IEEE Custom Integrated Circuits Conference (CICC). 2018, pp. 1–8. DOI: 10.1109/CICC.2018.8357061.
- Filip Maksimovic. "Monolithic Wireless Transceiver Design". PhD thesis. EECS Department, University of California, Berkeley, May 2020, pp. 64-65. URL: http://www2.eecs.berkeley.edu/Pubs/TechRpts/2020/EECS-2020-33.html.
- [3] Alex Moreno et al. "Single-Chip micro-Mote for Microrobotic Platforms". In: Government Microcircuit Applications & Critical Technology Conference. GOMACTech, 2020.
- Behzad Razavi. Design of Analog CMOS Integrated Circuits. first. McGraw-Hill, 2001, p. 36.
- [5] Nicholas Werblun. "Closing the Analog Design Loop with the Berkeley Analog Generator". MS. EECS Department, University of California, Berkeley, May 2019. URL: http://www2.eecs.berkeley.edu/Pubs/TechRpts/2019/EECS-2019-23.html.

# Appendix A BAG Setup

#### Workspace Repository

To get the files needed to run BAG in the TSMC 65nm LP process, clone the repository at https://bwrcrepo.eecs.berkeley.edu/tsmc65/tsmc65lp/bag2\_tsmc65lp\_workspace by running in a terminal:

```
$ git clone git@bwrcrepo.eecs.berkeley.edu:tsmc65/tsmc65lp/
bag2_tsmc65lp_workspace
```

To get the files needed to run BAG in the TSMC 28nm process, clone the repository at https://bwrcrepo.eecs.berkeley.edu/tstech28/bag2\_tsmc28\_workspace by running in a terminal:

\$ git clone git@bwrcrepo.eecs.berkeley.edu:tstech28/ bag2\_tsmc28\_workspace \$ git checkout ee194\_290c

Switching to the ee194\_290c branch is necessary for TSMC 28nm to avoid an error with submodule paths missing in setup. After cloning the repository, add submodules in the BAG workspace directory with the following commands:

```
$ git submodule add https://github.com/PisterLab/span_ion
    span_ion_proj
$ git submodule add https://github.com/PisterLab/bag2_analog
$ git submodule add https://github.com/PisterLab/bag2_digital
$ git submodule add https://github.com/PisterLab/
    bag2_wrappers
$ git submodule init
$ git submodule update --recursive
```

The cds.lib for both technologies will need to be modified so it contains:

```
INCLUDE cds.lib.bag
INCLUDE cds.lib.core
DEFINE BAG_prim $BAG_WORK_DIR/BAG_prim
```

These will provide Cadence Virtuoso a path to the respective technology libraries and BAG frameworks. The final line redefines a library path in cds.lib.bag to point to the work directory instead of the technology configuration directory.

For TSMC 28nm, it will also be necessary to copy the directory scripts\_char from the TSMC 65nm LP workspace, because it does not exist in the 28nm repository. Additionally, in bag\_libs.def, remove the line:

```
serdes_digitalbase_templates $BAG_WORK_DIR/serdes_digitalbase
    /BagModules
```

It is also necessary to copy bag\_testbenches from the TSMC 65nm LP workspace into the TSMC 28nm workspace so BAG sees that the correct directory name exists (i.e. without the extension \_ec). In cds.lib.bag add the line:

```
DEFINE bag_testbenches $BAG_WORK_DIR/bag_testbenches/
bag_testbenches
```

In bag\_libs.def add the lines:

bag\_testbenches \$BAG\_WORK\_DIR/bag\_testbenches/BagModules bag2\_analog \$BAG\_WORK\_DIR/bag2\_analog/BagModules span\_ion \$BAG\_WORK\_DIR/span\_ion\_proj/BagModules

Finally, in bag\_startup.py, change the line:

```
sys.path.append(os.path.join(os.environ['BAG_WORK_DIR'], '
    bag_testbenches_ec'))
```

to the following:

```
sys.path.append(os.path.join(os.environ['BAG_WORK_DIR'], '
    bag_testbenches'))
```

To get the LDO design script, example input spec file, and another file dsn\_cell.py necessary to run the design script, also clone https://github.com/PisterLab/bag2\_analog. Copy the directory scripts\_dsn from this new into the main BAG workspace folder, and move dsn\_cell.py into BAG\_framework/run\_scripts inside the workspace. The final hierarchy of these added files should be:

bag2\_tsmc<process>\_workspace

```
BAG_framework
    run_scripts
    dsn_cell.py
    scripts_dsn
    ldo_params.yaml
    regulator_ldo_series.py
```

Create a symbolic link to scripts\_dsn in the main workspace directory with the command:

```
$ ln -s BAG_framework/run_scripts/dsn_cell.py ./
```

BAG provides a framework for storing and accessing process characterization data for use in design scripts. To run characterization on a transistor, a YAML file such as the example in Appendix B will be necessary. Process-sensitive information has been removed from this example, so in addition to changing any transistor dimensions appropriately, the contents of the env\_list list will have to be changed from 'nominal' to the appropriate corner names, and any time 'th\_type' appears it should be replaced with the name of the intent as described later in the Primitive Setup subsection. To characterize a PMOS device, change every instance of 'nch' to 'pch' and set is\_nmos to 'False.' This file is read by BAG to set the testbench parameters for characterization. Once the YAML file and BAG workspace are setup, the process of characterization in the next subsection can begin.

All setup before this point only needs to be run once. The next subsections about characterization and primitive setup only have to be run once for each transistor setting or device type, but must be rerun if any settings change. When changing settings, it is also helpful to set a different filepath for the characterization data to be written to so previously characterized transistor data is still accessible without rerunning characterization.

#### **Process Characterization**

To run characterization once the BAG workspace is set up and a YAML file is configured for the appropriate transistor (examples in specs\_mos\_char/ in the 65nm workspace), start Virtuoso from a C shell by running:

```
$ source .cshrc
$ virtuoso &
```

If using Bash instead of a C shell, replace source .cshrc with:

```
$ source .bashrc
```

Once Virtuoso opens, run the following command from the Virtuoso Log window (not the terminal):

load("start\_bag.il")

To start the BAG session, run the following command from the terminal in the BAG workspace directory Virtuoso was launched in:

```
$ ./start_bag.sh
```

This will begin a terminal-based IPython session. From here, to run characterization:

```
$ run -i scripts_char/nmos_char.py
$ run -i scripts_char/pmos_char.py
```

Each command will run the characterization in the respective YAML files pointed to in nmos\_char.py and pmos\_char.py. The config\_file path in these scripts can be modified to point to the proper characterization input YAML file.

If a testbench for a particular type of transistor already exists in the testbench libraries BAG creates called AAAFOO\_MOSCHAR\_NCH or AAAFOO\_MOSCHAR\_PCH, the characterization script may raise an error that it "cannot load adexl database" if Cadence has locked the ADE XL. To fix this error and run the characterization, delete the existing testbench library, close Virtuoso, and exit the BAG session in the terminal. Restart Virtuoso and BAG using the process described above and run the characterization as before. The characterization should run to completion.

#### **Primitive Setup**

If a particular transistor does not have an existing cell in the library BAG\_Prim, a new cell can be made with drain, source, gate, and body contacts connected to input/output pins labelled D, S, G, and B respectively, and transistor length, width, and number of fingers set at pPar("1"), pPar("w"), and pPar("nf") respectively. The name of this cell should be nmos4\_[type] or pmos4\_[type] where '[type]' is the string in the characterization parameters YAML file used to direct the 'intent' to the correct cell, such as 'lvt' or 'hvt.'

Another error that may appear during characterization is "KeyError: 'vgs,'" which occurs if the device width or length is out of the bounds required by the process.

#### Running a Design Script

To run a design script, from the BAG session run:

```
$ run dsn_cell.py spec_dir/spec.yaml -dump spec_dir/result.
yaml
```

where spec\_dir/spec.yaml is the filepath to the YAML file with the design script to use and parameters to run it with such as the example in Appendix C (which is scrubbed of process-sensitive information), and spec\_dir/result.yaml is the filepath to the YAML file to write the output of the script to.

If changes have been made to any file in the BAG workspace, it may be necessary to exit the BAG session and restart it as described in the Process Characterization subsection above. In some cases, it may be necessary to also close Virtuoso, then rerun all startup commands, also as described in the Process Characterization subsection above.

### Appendix B

### Example Characterization Input YAML File

```
1 dut_lib: 'bag_testbenches'
2 dut_cell: 'mos_analogbase'
3 layout_package: 'abs_templates.mos_char'
4 layout_class: 'Transistor'
5
6 impl_lib: 'AAAFOO_MOSCHAR_NCH'
7 dsn_basename: 'NCH'
8
9 rcx_params:
    capacitance:
10
      ground_net: b
11
12 view_name: 'schematic'
13
14 root_dir: 'data/nch_scrubbed'
15 summary_fname: 'summary.yaml'
16
17 routing_grid:
    layers: [1, 2, 3, 4, 5, 6, 7]
18
    widths: [0.080, 0.080, 0.080, 0.080, 0.080, 0.080, 0.080]
19
    spaces: [0.080, 0.080, 0.080, 0.080, 0.080, 0.080, 0.080]
20
    bot_dir: 'y'
21
22
23 sweep_params:
    intent: ['th_type']
24
25
26 layout_params:
    mos_type: 'nch'
27
    lch: 1.0e-6
28
29
    w: 0.5e-6
    fg: 20
30
    intent: 'th_type'
31
    fg_dum: 4
32
    stack: 1
33
    ptap_w: 0.5e-6
34
    ntap_w: 0.5e-6
35
    tr_w_dict:
36
      g: 1
37
      d: 2
38
      s: 2
39
    tr_sp_dict:
40
     gs: 1
41
      gd: 1
42
      sb: 1
43
44
      db: 1
```

```
45
46 # Used only if we're doing schematic-only simulation
47 schematic_params:
    mos_type: 'nch'
48
    lch: 1.0e-6
49
    w: 0.5e-6
50
51
    fg: 20
    intent: 'th_type'
52
    stack: 1
53
    dum_info: !!null
54
55
56 dut_wrappers: []
57
58 env_list: ['nominal']
59
60 measurements:
    - meas_type: 'mos_ss'
61
      meas_package: 'verification.mos.sim'
62
      meas_class: 'MOSCharSS'
63
      out_fname: 'mos_ss.yaml'
64
      is_nmos: True
65
      fg: 20
66
      testbenches:
67
         ibias:
68
           tb_package: 'verification.mos.sim'
69
           tb_class: 'MOSIdTB'
70
           tb_lib: 'bag_testbenches'
71
           tb_cell: 'mos_tb_ibias'
72
73
           sch_params: {}
           wrapper_type: ''
74
           vgs_num: 200
75
           vgs_max: 1.0
76
           ibias_min_fg: 1.0e-9
77
           ibias_max_fg: 200.0e-6
78
           vgs_resolution: 2.0e-3
79
80
         sp:
           tb_package: 'verification.mos.sim'
81
           tb_class: 'MOSSPTB'
82
           tb_lib: 'bag_testbenches'
83
           tb_cell: 'mos_tb_sp'
84
           sch_params: {}
85
           wrapper_type: ''
86
           vgs_num: 30
87
88
           vds_num: 20
```

- 89 vds\_min: 5.0e-3
- 90 vds\_max: 1.0
- 91 vbs: [0.0, 0.15, 0.3, 0.45]
- 92 sp\_freq: 1.0e+6
- 93 cfit\_method: 'average'

### Appendix C

### Example Design Script Input YAML File

```
1 dsn_mod: scripts_dsn.regulator_ldo_series
2 dsn_cls: bag2_analog__regulator_ldo_series_dsn
3
4 params:
5
    ser_type: p
    vdd: !!float 1.5
6
    vout: !!float 1
7
    iload: !!float 1e-3
8
    iref: !!float 1e-6
9
    iamp_max: !!float 100e-6
10
    cload: !!float 1e-9
11
    cdecap: !!float 1e-9
12
    rsource: !!float 0
13
    err: !!float 1e-3
14
    psrr: !!float 40
15
    psrr_fbw: !!float 1e3
16
    pm: !!float 60
17
    loadreg: !!float 1e-3
18
    load_pole: False
19
    v_res: !!float 10e-3
20
21
    specfile_dict:
22
      amp_in: specs_mos_char/nch.yaml
23
      amp_load: specs_mos_char/pch.yaml
24
      amp_tail: specs_mos_char/nch.yaml
25
      amp_mir: specs_mos_char/nch.yaml
26
27
      ser: specs_mos_char/pch.yaml
    th_dict:
28
29
      amp_in: th_type
      amp_load: th_type
30
      amp_tail: th_type
31
      amp_mir: th_type
32
33
      ser: th_type
    l_dict:
34
      amp_in: !!float 500e-9
35
      amp_load: !!float 500e-9
36
      amp_tail: !!float 500e-9
37
      amp_mir: !!float 500e-9
38
      ser: !!float 500e-9
39
    sim_env: nominal
40
```

Appendix D LDO Design Script

```
1 # -*- coding: utf-8 -*-
2
3 from typing import Mapping, Tuple, Any, List
4
5 import os
6 import pkg_resources
7 import numpy as np
8 import warnings
9 from pprint import pprint
10
11 from bag.design.module import Module
12 from bag.core import BagProject
13 from span_ion_proj.scripts_dsn import DesignModule, get_mos_db, estimate_vth,
      parallel, verify_ratio, num_den_add, enable_print, disable_print
14 from bag.data.lti import LTICircuit, get_w_3db, get_stability_margins
15
16 # noinspection PyPep8Naming
17 class bag2_analog_regulator_ldo_series_dsn(DesignModule):
      """Module for library bag2_analog cell regulator_ldo_series
18
      Fill in high level description here.
19
      .....
20
21
      @classmethod
22
      def get_op_info(cls) -> Mapping[str,str]:
23
          # type: () -> Dict[str, str]
24
          """Returns a dictionary from parameter names to descriptions.
25
          Returns
26
27
            ____
          param_info : Optional[Dict[str, str]]
28
               dictionary from parameter names to descriptions.
29
           .....
30
          ans = super().get_op_info()
31
32
          ans.update(dict(
               specfile_dict = 'Transistor database spec file names for each device'
33
               ser_type = 'n or p for type of series device',
34
               th_dict = 'Transistor flavor dictionary.',
35
               l_dict = 'Transistor channel length dictionary',
36
               sim_env = 'Simulation environment',
37
               vdd = 'Supply voltage in volts.',
38
               vout = 'Reference voltage to regulate the output to',
39
               loadreg = 'Maximum absolute change in output voltage given change in
40
      output current',
               iload = 'Bias current of series device, in amperes.',
41
```

```
iref = 'Reference current for amplifier biasing, in amperes',
42
43
               iamp_max = 'Maximum amplifier current, in amperes',
               cload = 'Load capacitance from the output of the LDO to ground',
44
               cdecap = 'Maximum additional capacitance added to circuit',
45
46
               rsource = 'Resistance from the power supply, in ohms',
               err = 'Maximum percent static error at output (as decimal)',
47
               psrr = 'Minimum power supply rejection ratio (dB, 20*log10(dVdd/dVout
48
      ))',
               psrr_fbw = 'Minimum bandwidth for power supply rejection roll-off',
49
               pm = 'Minimum phase margin for the large feedback loop, in degrees',
50
               load_pole = 'True to ensure dominant pole is at theregulator output',
51
               v_res = 'Resolution of voltage bias point sweeps, in volts'
52
           ))
53
          return ans
54
      def resize_op(self, op, wm):
56
           op_new = dict()
           for key,value in op.items():
58
               if key[0] != 'v':
59
60
                   op_new[key] = wm*value
               else:
61
                   op_new[key] = value
62
           return op_new
63
64
      def dsn_fet(self, **params):
65
           specfile_dict = params['specfile_dict']
66
           ser_type = params['ser_type']
67
           th_dict = params['th_dict']
68
           sim_env = params['sim_env']
69
70
           db_dict = {k:get_mos_db(spec_file=specfile_dict[k],
71
                                    intent=th_dict[k],
72
73
                                    sim_env=sim_env) for k in specfile_dict.keys()}
74
           vdd = params['vdd']
75
           vout = params['vout']
76
           vg = params['vg']
77
           iload = params['iload']
78
79
          vs = vout if ser_type == 'n' else vdd
80
           vd = vdd if ser_type == 'n' else vout
81
           vb = 0 if ser_type == 'n' else vdd
82
83
           ser_op = db_dict['ser'].query(vgs=vg-vs, vds=vd-vs, vbs=vb-vs)
84
```

```
nf = int(round(iload/ser_op['ibias']))
85
           m = iload/(2*ser_op['ibias'])
86
           wm = (m\%1 + 1)
87
           nf = 2*int(m)
88
89
            ser_op = self.resize_op(ser_op,wm)
           return m > 1, dict(nf=nf, wm=wm, op=ser_op)
90
91
       def dsn_amp(self, **params):
92
           vdd = params['vdd']
93
           vincm = params['vout']
94
           voutcm = params['voutcm']
95
            iload = params['iload']
96
           iref = params['iref']
97
            iamp_max = params['iamp_max']
98
            cload = params['cload']
99
            cdecap_max = params['cdecap']
100
           rsource = params['rsource']
            err_max = params['err']
102
           psrr_min = params['psrr']
103
           psrr_fbw_min = params['psrr_fbw']
104
           pm_min = params['pm']
105
           loadreg_max = params['loadreg']
106
           load_pole = params['load_pole']
           v_res = params['v_res']
108
           ser_type = params['ser_type']
109
            ser_info = params['ser_info']
110
           db_dict = params['db_dict']
111
           amp_in = 'n'
112
113
           # Get amplifier load pair parameters
114
           op_load = db_dict['amp_load'].query(vgs=-(vdd-voutcm), vds=-(vdd-voutcm),
115
        vbs=0)
           Vstar_load = op_load['vstar']
116
117
            amp_dsn_info = dict()
118
119
           # Choose amp bias voltages
120
           Vstar_in_err = float('inf')
121
           vtail = 0
122
           op_in = dict()
123
           for vtail_i in np.arange(0,min(voutcm,vincm),v_res):
124
                op_in_i = db_dict['amp_in'].query(vgs=vincm-vtail_i, vds=voutcm-
125
       vtail_i, vbs=-vtail_i)
                Vstar_in_i = op_in_i['vstar']
126
```

```
if Vstar_in_err > abs(Vstar_load-Vstar_in_i) and op_in_i['ibias'] >
127
       0:
                    Vstar_in_err = abs(Vstar_load-Vstar_in_i)
128
                    vtail = vtail_i
129
                    op_in = op_in_i
130
                    Vstar_in = Vstar_in_i
131
132
           Vstar_tail_errsq = float('inf')
133
           vgtail = 0
134
            op_tail = dict()
135
           for vgtail_i in np.arange(0,vdd,v_res):
136
                op_tail_i = db_dict['amp_tail'].query(vgs=vgtail_i, vds=vtail, vbs=0)
137
                Vstar_tail_i = op_tail_i['vstar']
138
                if Vstar_tail_errsq > abs(Vstar_load-Vstar_tail_i)**2+abs(Vstar_in-
139
       Vstar_tail_i)**2 and op_tail_i['ibias'] > 0:
                    Vstar_tail_errsq = abs(Vstar_load-Vstar_tail_i)**2+abs(Vstar_in-
140
       Vstar_tail_i)**2
                    vgtail = vgtail_i
141
                    op_tail = op_tail_i
142
143
                    Vstar_tail = Vstar_tail_i
144
           # Size reference current mirror
145
            op_mir = db_dict['amp_mir'].query(vgs=vgtail, vds=vgtail, vbs=0)
146
           m_mir = iref/(2*op_mir['ibias'])
147
           wm_mir = (m_mir%1 + 1)
148
           nf_mir = 2*int(m_mir)
149
           if nf_mir == 0:
150
                return False, amp_dsn_info
151
152
           # Size amplifier devices and base current
           Id_tail = wm_mir*op_tail['ibias']
154
           wm_tail = wm_mir
155
           nf_tail = int(2*max(((2*op_load['ibias'])//Id_tail)+1,((2*op_in['ibias'])
156
       //Id_tail)+1))
           if Id_tail*nf_tail > iamp_max:
157
                return False, amp_dsn_info
158
159
           Id_load = Id_tail*nf_tail/2
160
           m_load = Id_load/op_load['ibias']
161
           wm_load = (m_load/2)%1 + 1
162
           nf_load = 2*int(m_load/2)
163
164
           m_in = Id_load/op_in['ibias']
165
           wm_{in} = (m_{in}/2)\%1 + 1
166
```

```
nf_{in} = 2*int(m_{in}/2)
167
168
           # Resize op and format parameters
169
           op_dict = {'amp_in' : self.resize_op(op_in, wm_in),
170
                       'amp_tail' : self.resize_op(op_tail, wm_tail),
171
                       'amp_load' : self.resize_op(op_load, wm_tail),
172
                       'amp_mir' : self.resize_op(op_mir, wm_mir),
173
                       'ser' : ser_info['op']}
174
           nf_dict = {'amp_in' : nf_in,
175
                       'amp_tail' : nf_tail,
176
                       'amp_load' : nf_load,
177
                       'amp_mir' : nf_mir,
178
                       'ser' : ser_info['nf']}
179
           wm_dict = { 'amp_in' : wm_in,
180
                       'amp_tail' : wm_tail,
181
                       'amp_load' : wm_load,
182
                       'amp_mir' : wm_mir,
183
                       'ser' : ser_info['wm']}
184
185
           A = abs(self._get_loopgain_lti(op_dict, nf_dict, ser_type, amp_in,
186
       rsource))
           dc_{err} = 1/(A+1)
187
           loadreg = self._get_loadreg_lti(op_dict, nf_dict, ser_type, amp_in, cload
188
       , 0, rsource, vincm, iload)
           psrr, psrr_fbw = self._get_psrr_lti(op_dict, nf_dict, ser_type, amp_in,
189
       cload, 0, rsource)
           pm = self._get_stb_lti(op_dict, nf_dict, ser_type, amp_in, cload, 0,
190
       rsource)
           if pm > pm_min and psrr > psrr_min and psrr_fbw > psrr_fbw_min and
191
       loadreg < loadreg_max and dc_err < err_max and Id_tail*nf_tail < iamp_max and
       not load_pole:
                amp_dsn_info.update(dict(op_dict=op_dict,nf_dict=nf_dict,wm_dict=
192
       wm_dict))
                amp_dsn_info.update(cap_dict=dict(cdecap_amp=0, cdecap_load=0))
193
                amp_dsn_info.update(dict(loadreg=loadreg, psrr=psrr, psrr_fbw=
194
       psrr_fbw, pm=pm, err=dc_err, ibias=Id_tail*nf_tail))
                return True, amp_dsn_info
195
           if psrr_fbw > psrr_fbw_min and Id_tail*nf_tail < iamp_max and not
196
       load_pole:
                # Find minimum decap necessary with dominant amplifier pole
197
                cdecap_min = ser_info['op']['cgg']*ser_info['nf']
198
                for cdecap_amp in np.logspace(np.log10(cdecap_min),np.log10(
199
       cdecap_max),100):
                    loadreg = self._get_loadreg_lti(op_dict, nf_dict, ser_type,
200
```

```
amp_in, cload, cdecap_amp, rsource, vincm, iload)
                    psrr, psrr_fbw = self._get_psrr_lti(op_dict, nf_dict, ser_type,
201
       amp_in, cload, 0, rsource)
                    pm = self._get_stb_lti(op_dict, nf_dict, ser_type, amp_in, cload,
202
        cdecap_amp, rsource)
                    if psrr_fbw < psrr_fbw_min:</pre>
203
204
                        break
                    if pm > pm_min and psrr > psrr_min and psrr_fbw > psrr_fbw_min
205
       and loadreg < loadreg_max and dc_err < err_max:
                        amp_dsn_info.update(dict(op_dict=op_dict,nf_dict=nf_dict,
206
       wm_dict=wm_dict))
                        amp_dsn_info.update(cap_dict=dict(cdecap_amp=cdecap_amp,
207
       cdecap_load=0))
                        amp_dsn_info.update(dict(loadreg=loadreg, psrr=psrr, psrr_fbw
208
       =psrr_fbw, pm=pm, err=dc_err, ibias=Id_tail*nf_tail))
                        return True, amp_dsn_info
209
            if psrr > psrr_min and loadreg < loadreg_max:</pre>
210
                pm = 0
211
               psrr_fbw = 0
212
                while (pm < pm_min or psrr_fbw < psrr_fbw_min) and Id_tail*nf_tail <
213
       iamp_max:
                    # Resize amp parameters
214
                    Id_load = Id_tail*nf_tail/2
215
216
                    m_load = Id_load/op_load['ibias']
217
                    wm_load = (m_load/2)%1 + 1
218
                    nf_load = 2*int(m_load/2)
219
220
                    m_in = Id_load/op_in['ibias']
221
                    wm_{in} = (m_{in}/2)%1 + 1
222
                    nf_{in} = 2*int(m_{in}/2)
223
224
                    op_dict.update({'amp_in' : self.resize_op(op_in, wm_in),
225
                                     'amp_tail' : self.resize_op(op_tail, wm_tail),
226
                                     'amp_load' : self.resize_op(op_load, wm_tail)})
227
                    nf_dict.update({'amp_in' : nf_in,
228
                                     'amp_tail' : nf_tail,
229
                                     'amp_load' : nf_load})
230
                    wm_dict.update({'amp_in' : wm_in,
231
                                     'amp_tail' : wm_tail,
232
                                     'amp_load' : wm_load})
233
                    loadreg = self._get_loadreg_lti(op_dict, nf_dict, ser_type,
234
       amp_in, cload+cdecap_max, 0, rsource, vincm, iload)
                    psrr, psrr_fbw = self._get_psrr_lti(op_dict, nf_dict, ser_type,
235
```

```
amp_in, cload+cdecap_max, 0, rsource)
                    pm = self._get_stb_lti(op_dict, nf_dict, ser_type, amp_in, cload+
236
       cdecap_max, 0, rsource)
                    nf_tail += 2
237
                amp_dsn_info.update(dict(op_dict=op_dict,nf_dict=nf_dict,wm_dict=
238
       wm_dict))
                amp_dsn_info.update(cap_dict=dict(cdecap_amp=0, cdecap_load=
239
       cdecap_max))
                amp_dsn_info.update(dict(loadreg=loadreg, psrr=psrr, psrr_fbw=
240
       psrr_fbw, pm=pm, err=dc_err, ibias=Id_tail*nf_dict['amp_tail']))
                spec_met = pm > pm_min and psrr > psrr_min and psrr_fbw >
241
       psrr_fbw_min and loadreg < loadreg_max and dc_err < err_max and Id_tail*
       nf_dict['amp_tail'] < iamp_max</pre>
                return spec_met, amp_dsn_info
242
            else:
243
                return False, amp_dsn_info
244
245
246
       def meet_spec(self, **params) -> List[Mapping[str,Any]]:
247
            specfile_dict = params['specfile_dict']
248
           th_dict = params['th_dict']
249
            l_dict = params['l_dict']
250
            sim_env = params['sim_env']
251
252
           db_dict = {k:get_mos_db(spec_file=specfile_dict[k],
253
                                     intent=th_dict[k],
254
                                     sim_env=sim_env) for k in specfile_dict.keys()}
255
           params.update(dict(db_dict=db_dict))
256
257
            ser_type = params['ser_type']
258
           vdd = params['vdd']
259
           vout = params['vout']
260
            iload = params['iload']
261
            iref = params['iref']
262
            iamp_max = params['iamp_max']
263
            cload = params['cload']
264
            cdecap_max = params['cdecap']
265
           rsource = params['rsource']
266
           err_max = params['err']
267
           psrr_min = params['psrr']
268
           psrr_fbw_min = params['psrr_fbw']
269
           pm_min = params['pm']
270
           loadreg_max = params['loadreg']
271
           load_pole = params['load_pole']
272
```

```
v_res = params['v_res']
273
274
            vth_ser = estimate_vth(db=db_dict['ser'],
275
                                     is_nch=ser_type=='n',
276
                                     lch=l_dict['ser'],
277
                                     vgs=vdd-vout if ser_type=='n' else -vdd/2,
278
                                     vbs=0-vout if ser_type=='n' else 0)
279
280
            # Keep track of best option
281
            best_op = dict(ibias=float('inf'),
282
                            err=float('inf'),
283
                            psrr=0,
284
                            psrr_fbw=0,
285
                            pm=0,
286
                            loadreg=float('inf'))
287
288
            type_dict = {'ser' : ser_type,
289
290
                          'amp_load' : 'p',
                          'amp_in' : 'n',
291
                          'amp_tail' : 'n',
292
                          'amp_mir' : 'n'}
293
294
            w_dict = {k:db.width_list[0] for k,db in db_dict.items()}
295
296
            self.other_params = dict(l_dict=l_dict,
297
                                       w_dict=w_dict,
298
                                       th_dict=th_dict,
299
                                       cload=cload.
300
301
                                       ser_type=ser_type)
302
            # Sweep gate bias voltage of the series device
303
            vg_min = vout+vth_ser
304
            vg_max = min(vdd+vth_ser, vdd)
305
            vg_vec = np.arange(vg_min, vg_max, v_res)
306
307
            for vg in vg_vec:
308
                print('Designing the series device...')
309
                # Size the series device
310
                match_ser, ser_info = self.dsn_fet(vg=vg, **params)
311
                if not match_ser:
312
                    continue
313
                print('Done')
314
315
316
                # Design amplifier s.t. output bias = gate voltage
```

```
# This is to maintain accuracy in the computational design proces
317
                print('Designing the amplifier...')
318
319
                params.update(dict(voutcm=vg,
320
321
                                     iamp_max=iamp_max,
                                     ser_info=ser_info))
322
                spec_met, amp_dsn_info = self.dsn_amp(**params)
323
                print('Done')
324
325
                if not spec_met:
326
                    print('Amp specs not met.')
327
                    continue
328
                else:
329
                    print('AMP SPECS MET.')
330
331
                amp_dsn_info.update(dict(w_dict=w_dict, l_dict=l_dict, th_dict=
332
       th_dict, type_dict=type_dict))
333
                best_op.update(self.op_compare(best_op,amp_dsn_info))
334
                iamp_max = best_op['ibias']
335
            return [best_op]
336
337
       def _get_loopgain_lti(self, op_dict, nf_dict, ser_type, amp_in, rsource) ->
338
       float:
            1.1.1
339
           Returns:
340
                A: DC loop gain
341
            1.1.1
342
            ckt = LTICircuit()
343
344
           n_ser = ser_type == 'n'
345
            n_amp = amp_in == 'n'
346
            vdd = 'vdd' if rsource != 0 else 'gnd'
347
348
            # Series device
349
            ser_d = vdd if n_ser else 'reg'
350
            ser_s = 'reg' if n_ser else vdd
351
            ckt.add_transistor(op_dict['ser'], ser_d, 'out', ser_s, fg=nf_dict['ser'
352
       ], neg_cap=True)
353
            # Passives
354
            if rsource != 0:
355
                ckt.add_res(rsource, 'gnd', 'vdd')
356
357
```

```
# Amplifier
358
           tail_rail = 'gnd' if n_amp else vdd
359
           load_rail = vdd if n_amp else 'gnd'
360
            inp_conn = 'gnd' if n_ser else 'amp_in'
361
            inn_conn = 'gnd' if not n_ser else 'amp_in'
362
            ckt.add_transistor(op_dict['amp_in'], 'outx', inp_conn, 'tail', fg=
363
       nf_dict['amp_in'], neg_cap=False)
            ckt.add_transistor(op_dict['amp_in'], 'out', inn_conn, 'tail', fg=nf_dict
364
       ['amp_in'], neg_cap=False)
            ckt.add_transistor(op_dict['amp_tail'], 'tail', 'gnd', tail_rail, fg=
365
       nf_dict['amp_tail'], neg_cap=False)
            ckt.add_transistor(op_dict['amp_load'], 'outx', 'outx', load_rail, fg=
366
       nf_dict['amp_load'], neg_cap=False)
            ckt.add_transistor(op_dict['amp_load'], 'out', 'outx', load_rail, fg=
367
       nf_dict['amp_load'], neg_cap=False)
368
           # Calculating stability margins
369
370
           num, den = ckt.get_num_den(in_name='amp_in', out_name='reg', in_type='v')
           A = num[-1]/den[-1]
371
372
373
           return A
374
       def _get_psrr_lti(self, op_dict, nf_dict, ser_type, amp_in, cload, cdecap_amp
375
       , rsource) -> float:
            1.1.1
376
           Returns:
377
                psrr: PSRR (dB)
378
                fbw: Power supply -> output 3dB bandwidth (Hz)
379
            1.1.1
380
           n_ser = ser_type == 'n'
381
           n_amp = amp_in == 'n'
382
383
384
           # Supply -> output gain
            ckt_sup = LTICircuit()
385
            ser_d = 'vdd' if n_ser else 'reg'
386
            ser_s = 'reg' if n_ser else 'vdd'
387
            inp_conn = 'gnd' if n_ser else 'reg'
388
            inn_conn = 'reg' if n_ser else 'gnd'
389
           tail_rail = 'gnd' if n_amp else 'vdd'
390
           load_rail = 'vdd' if n_amp else 'gnd'
391
392
           # Passives
393
394
            if rsource != 0:
                ckt_sup.add_res(rsource, 'vbat', 'vdd')
395
```

```
ckt_sup.add_cap(cload, 'reg', 'gnd')
396
            ckt_sup.add_cap(cdecap_amp, 'out', 'reg')
397
398
           # Series device
399
            ckt_sup.add_transistor(op_dict['ser'], ser_d, 'out', ser_s, fg=nf_dict['
400
       ser'], neg_cap=False)
401
           # Amplifier
402
            ckt_sup.add_transistor(op_dict['amp_in'], 'outx', inp_conn, 'tail', fg=
403
       nf_dict['amp_in'], neg_cap=False)
            ckt_sup.add_transistor(op_dict['amp_in'], 'out', inn_conn, 'tail', fg=
404
       nf_dict['amp_in'], neg_cap=False)
            ckt_sup.add_transistor(op_dict['amp_tail'], 'tail', 'gnd', tail_rail, fg=
405
       nf_dict['amp_tail'], neg_cap=False)
            ckt_sup.add_transistor(op_dict['amp_load'], 'outx', 'outx', load_rail, fg
406
       =nf_dict['amp_load'], neg_cap=False)
            ckt_sup.add_transistor(op_dict['amp_load'], 'out', 'outx', load_rail, fg=
407
       nf_dict['amp_load'], neg_cap=False)
408
409
            if rsource == 0:
                num_sup, den_sup = ckt_sup.get_num_den(in_name='vdd', out_name='reg',
410
        in_type='v')
            else:
411
               num_sup, den_sup = ckt_sup.get_num_den(in_name='vbat', out_name='reg'
412
       , in_type='v')
           gain_sup = num_sup[-1]/den_sup[-1]
413
           wbw_sup = get_w_3db(den_sup, num_sup)
414
415
           if gain_sup == 0:
416
                return float('inf')
417
           if wbw_sup == None:
418
                wbw_sup = 0
419
           fbw_sup = wbw_sup / (2*np.pi)
420
421
           psrr = 10*np.log10((1/gain_sup)**2)
422
423
           return psrr, fbw_sup
424
425
       def _get_stb_lti(self, op_dict, nf_dict, ser_type, amp_in, cload, cdecap_amp,
426
        rsource) -> float:
            1.1.1
427
           Returns:
428
429
                pm: Phase margin (degrees)
            . . .
430
```

```
ckt = LTICircuit()
431
432
           n_ser = ser_type == 'n'
433
           n_amp = amp_in == 'n'
434
           vdd = 'vdd' if rsource != 0 else 'gnd'
435
436
           # Series device
437
            ser_d = vdd if n_ser else 'reg'
438
            ser_s = 'reg' if n_ser else vdd
439
            ckt.add_transistor(op_dict['ser'], ser_d, 'out', ser_s, fg=nf_dict['ser'
440
       ], neg_cap=False)
441
           # Passives
442
            ckt.add_cap(cload, 'reg', 'gnd')
443
            ckt.add_cap(cdecap_amp, 'out', 'reg')
444
            if rsource != 0:
445
                ckt.add_res(rsource, 'gnd', 'vdd')
446
447
           # Amplifier
448
449
           tail_rail = 'gnd' if n_amp else vdd
           load_rail = vdd if n_amp else 'gnd'
450
            inp_conn = 'gnd' if n_ser else 'amp_in'
451
            inn_conn = 'gnd' if not n_ser else 'amp_in'
452
            ckt.add_transistor(op_dict['amp_in'], 'outx', inp_conn, 'tail', fg=
453
       nf_dict['amp_in'], neg_cap=False)
            ckt.add_transistor(op_dict['amp_in'], 'out', inn_conn, 'tail', fg=nf_dict
454
       ['amp_in'], neg_cap=False)
            ckt.add_transistor(op_dict['amp_tail'], 'tail', 'gnd', tail_rail, fg=
455
       nf_dict['amp_tail'], neg_cap=False)
            ckt.add_transistor(op_dict['amp_load'], 'outx', 'outx', load_rail, fg=
456
       nf_dict['amp_load'], neg_cap=False)
            ckt.add_transistor(op_dict['amp_load'], 'out', 'outx', load_rail, fg=
457
       nf_dict['amp_load'], neg_cap=False)
458
           # Calculating stability margins
459
           num, den = ckt.get_num_den(in_name='amp_in', out_name='reg', in_type='v')
460
           pm, _ = get_stability_margins(np.convolve(num, [-1]), den)
461
462
           return pm
463
464
       def _get_loadreg_lti(self, op_dict, nf_dict, ser_type, amp_in, cload,
465
       cdecap_amp, rsource, vout, iout) -> float:
            1.1.1
466
467
           Returns:
```

```
loadreg: Load regulation for peak-to-peak load current variation of
468
       20\% (V/V)
            1.1.1
469
           n_ser = ser_type == 'n'
470
           n_amp = amp_in == 'n'
471
           vdd = 'vdd' if rsource != 0 else 'gnd'
472
473
           # Supply -> output gain
474
           ckt = LTICircuit()
475
            ser_d = vdd if n_ser else 'reg'
476
            ser_s = 'reg' if n_ser else vdd
477
            inp_conn = 'gnd' if n_ser else 'reg'
478
           inn_conn = 'reg' if n_ser else 'gnd'
479
           tail_rail = 'gnd' if n_amp else vdd
480
           load_rail = vdd if n_amp else 'gnd'
481
482
           # Passives
483
           if rsource != 0:
484
                ckt.add_res(rsource, 'gnd', 'vdd')
485
            ckt.add_cap(cload, 'reg', 'gnd')
486
            ckt.add_cap(cdecap_amp, 'out', 'reg')
487
488
           # Series device
489
            ckt.add_transistor(op_dict['ser'], ser_d, 'out', ser_s, fg=nf_dict['ser'
490
       ], neg_cap=False)
491
            # Amplifier
492
            ckt.add_transistor(op_dict['amp_in'], 'outx', inp_conn, 'tail', fg=
493
       nf_dict['amp_in'], neg_cap=False)
            ckt.add_transistor(op_dict['amp_in'], 'out', inn_conn, 'tail', fg=nf_dict
494
       ['amp_in'], neg_cap=False)
            ckt.add_transistor(op_dict['amp_tail'], 'tail', 'gnd', tail_rail, fg=
495
       nf_dict['amp_tail'], neg_cap=False)
            ckt.add_transistor(op_dict['amp_load'], 'outx', 'outx', load_rail, fg=
496
       nf_dict['amp_load'], neg_cap=False)
            ckt.add_transistor(op_dict['amp_load'], 'out', 'outx', load_rail, fg=
497
       nf_dict['amp_load'], neg_cap=False)
498
499
           num, den = ckt.get_num_den(in_name='reg', out_name='reg', in_type='i')
500
           transimpedance = num[-1]/den[-1]
501
502
503
           loadreg = transimpedance*0.2*iout/vout
504
```

```
return loadreg
505
506
       def op_compare(self, op1:Mapping[str,Any], op2:Mapping[str,Any]):
507
           return op1 if op1['ibias'] < op2['ibias'] else op2</pre>
508
509
       def get_sch_params(self, op):
510
511
           try:
                w_dict_new = dict()
512
                for key in op['w_dict'].keys():
513
                    w_dict_new[key]=op['wm_dict'][key]*op['w_dict'][key]
514
                dsn_op = dict(w_dict=w_dict_new,
515
                               l_dict=op['l_dict'],
516
                               nf_dict=op['nf_dict'],
517
                               th_dict=op['th_dict'],
518
                               type_dict=op['type_dict'],
519
                               cap_dict=op['cap_dict'])
520
           except KeyError:
521
                dsn_op = 'No solution found within specs'
522
           return dsn_op
523
```

Appendix E VDDD Tap (SCµM)



Figure E.1: VDDD tap device proposed layout area.

The biggest unresolved issue with the power systems on SCuM is the "VDDD tap" problem. A large proportion of the fabricated chips will stop executing instructions in the boot sequence unless the output of the digital LDO is briefly shorted to some voltage between the nominal LDO output (1V) and the battery voltage. The amount of time the LDO output must remain shorted is currently unknown. The cause of this problem is also still unknown, but the issue seems to occur after the first SRAM instruction, when the second instruction does not run. One proposed solution is to add a small transistor either in parallel with the series device (PMOS) or between the amplifier output and ground (NMOS) with an inverter at the gate. The proposed area for these added devices is shown in the 45µm by 25µm red box in Figure E.1, and the schematic with an added NMOS is shown in Figure E.2. The gate or inverter could then be connected to a repurposed GPO pin that would default low when power is connected and the boot sequence begins, shorting the series device gate to ground. Once the SRAM instructions continue to run and SCµM continues to boot, the GPO would switch high and the LDO would be allowed to operate normally. If the issue stems from the design of the digital LDO on SCuM, redesigning it may solve this issue on its own, but the auxiliary LDO uses an identical schematic and does not experience this problem. The SRAM on chip is provided by TSMC and was not designed specifically for  $SC\mu M$ , so the interaction between this block and the digital LDO could be another reason for the problem.



