# 1 TODO!!!!!!!!!!!!!!

Higher level description. Explain non-intuitive operations at the level of the algorithm below it. queue. Dequeue -¿ intuitive, described that it's a FIFO queue. CutLoops -¿ One line description of input -¿ output, not specifics of how. See Algorithm number for details. Repeat high level description for algorithm before. See section blah for a discussion of the recovery time calculation Explain WriteToFile - Just outputting

# 2 Data Structures

# 2.1 Basic Types

The following are the basic types, out of which others are built, and which will be referred to. There is generally, but not always, a direct relationship to a C++ primitive.

| Name                                 | Closest C++ Equivalent                 | Description                                                                         |
|--------------------------------------|----------------------------------------|-------------------------------------------------------------------------------------|
| Integer                              | int                                    | Whole number                                                                        |
| Float                                | float                                  | Floating point number                                                               |
| Queue                                | std::list                              | FIFO queue                                                                          |
| List(type)                           | std::list(type)                        | -                                                                                   |
| String                               | std::string                            | String object that provides operations to manipulate itself                         |
| File                                 | std::iostream                          | Abstract type to represent simple I/O operations                                    |
| $Map(KeyType \rightarrow ValueType,$ | $std::unordered\_map \langle KeyType,$ | A map to translate values of type                                                   |
| DEFAULT: DefaultValue)               | ValueType>                             | KeyType to values of type ValueType. If the key isn't present, returns DefaultValue |

The following are complex types, which are further defined below. This is merely a quick description of each.

| Name                                    | Description                                                                                                                                                                                                                                                                                                                                       |
|-----------------------------------------|---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
| Blif<br>BlifModel<br>BlifNode<br>Signal | Parent object, contains all information about a BLIF file and provides useful operations Represents a circuit within a BLIF file, and provides methods to manipulate said circuit A circuit element, or node in the DFG representing the circuit A signal within a specific circuit, or BlifModel, representing a set of edges with common source |

#### **2.2** Blif

#### 2.3 Model

Represents the circuit as a DFG. Contains a list of nodes, map of signal name  $\rightarrow$  Signal\*, and lists of primary inputs and outputs for the circuit. Each node contains the names of its input and output signals, allowing the Signal\* to be looked up, then the Signal\* contains pointers to its source and sink nodes. This allows the DFG to be traversed by going from node, to signal, to node, etc. The reason for having signals be referred to by name and looked up, rather than a node pointing directly to a signal is twofold. One, a node is an independent construct which can be added to or removed from any Model\*. For example,









when adding a node into a partition. Signals, on the other hand, are circuit specific and are created and destroyed as circumstances demand. Two, regardless of implementation, the node would need to store the signal name and the circuit would need to provide a mapping from name to Signal\*. This method just dosn't abstract it away Todo: actual reasons TODO: Image showing DFG traversal, and example of blif file and class contents

# 3 Algorithm

Types marked with an \* are defined in the previous section (Data Structures). We're given a blif file as

| <b>Algorithm 2</b> Main Algorithm | Algorithm | 2 | Main | Algor | ithm |
|-----------------------------------|-----------|---|------|-------|------|
|-----------------------------------|-----------|---|------|-------|------|

| Variable           | Type          | Description                                               |
|--------------------|---------------|-----------------------------------------------------------|
| input              | file          | Input blif file                                           |
| targetRecoveryTime | float         | Per partition recovery time (in seconds)                  |
| files              | list of files | circuit partitions, one per file                          |
| file               | file          |                                                           |
| header             | string        | string containing the first three lines of the input file |
| output             | file          | output file                                               |

```
1: procedure MAIN(input, targetRecoveryTime)
       files \leftarrow Partition(input)
2:
       for all file \in files do
3:
            file \leftarrow Triplicate(file)
4:
       end for
5:
       header \leftarrow input.lines[0 \rightarrow 3]
6:
       file \leftarrow Join(files, header)
7:
       output \leftarrow Flatten(output)
8:
9: end procedure
```

input. In line 11 we partition the input circuit into a number of sub circuits, each in a separate file, as further expanded in Algorithm [?]. Then in lines 12-13 for each partition file, we read it in as a black box, triplicate it, insert voting logic, and write it back out. Next in line 14 we extract the original header, which provides the name, inputs and outputs of the original circuit. We then, in line 15, join all the partitions together with the original name, inputs and outputs (in the same order), as the original circuit, and finally line 16 flattens the circuit, i.e. transforms the generated heirarchical netlist into a flat netlist with only one main model, or circuit, and no submodels.

| Variable           | Туре                               | Description                                                        |
|--------------------|------------------------------------|--------------------------------------------------------------------|
| $\overline{file}$  | file                               | input file                                                         |
| targetRecoveryTime | float                              | maximum per partition recovery time (in seconds)                   |
| blif               | Blif*                              | In-memory representation of input blif file                        |
| circuit            | BlifModel*                         | Main circuit from input file, represented as DFG                   |
| partition          | BlifModel*                         | Circuit, which we are adding nodes to, to make our partition       |
| queue              | Queue                              | FIFO queue of nodes to visit                                       |
| visited            | $Map(BlifNode* \rightarrow bool)$  | Map of whether a BlifNode is visited                               |
| signal             | Signal*                            |                                                                    |
| circuit.outputs    | List of Signal*                    | List of output Signal* of a circuit                                |
| signal. source     | BlifNode*                          | Node which drives this Signal*                                     |
| queue.size         | integer                            | Number of nodes in queue                                           |
| node               | BlifNode*                          |                                                                    |
| file               | file                               |                                                                    |
| files              | List of file                       |                                                                    |
| numPartitions      | int                                | Counter of number of partitions                                    |
| signal Name        | string                             | Name of a Signal*                                                  |
| node.inputs        | List of string                     | List of names of signals which are inputs to this node             |
| model.signals      | $Map(string \rightarrow Signal^*)$ | Map from signal name to Signal* representing it in that BlifModel* |

Table 1: Variables for Partition

#### Algorithm 3 Partition

```
1: procedure Partition(file)
        blif \leftarrow \text{new Blif(file)}
                                                                                                \triangleright Read in file
        circuit \leftarrow blif.main
                                                                      > The actual circuit within the blif file
 3:
 4:
        partition \leftarrow \text{new BlifModel}
                                                                                              5:
        queue \leftarrow \text{new Queue}
        visited \leftarrow \text{new Map(BlifNode} \rightarrow \text{bool, DEFAULT: false)}
 6:
 7:
        for all signal \in circuit.outputs do
 8:
            queue.Enqueue(signal.source)
 9:
        end for
10:
        while queue.size > 0 do
            node \leftarrow queue.Dequeue()
11:
12:
            if visited[node] = true then
                continue
                                                                    ▶ Handle each node once and only once
13:
            end if
14:
15:
            visited[node] \leftarrow true
            partition.AddNode(node)
16:
            if partition.RecoveryTime() > targetRecoveryTime then
17:
                partition.RemoveNode(node)
18:
                MakeIOList(partition, circuit)
19:
                file \leftarrow partition.WriteToFile()
20:
                files \leftarrow files + file
21:
22:
                numPartitions \leftarrow numPartitions + 1
23:
                partition \leftarrow \text{new BlifModel}
                                                                                              partition.AddNode(node)
24:
25:
            end if
            for all signalName \in node.inputs do
26:
                signal \leftarrow model.signals[signalName]
27:
                queue.Enqueue(siqnal)
28:
29:
            end for
        end while
30:
31:
        if partition.size > 0 then
            MakeIOList(partition, circuit)
32:
            file \leftarrow partition.WriteToFile()
33:
            files \leftarrow files + file
34:
        end if
35:
36:
        return files
37: end procedure
```

Line 2 reads a **BLIF!** (**BLIF!**) into memory, transforming from the **BLIF!** format described in **TODO**: Reference to the one represented by TODO: Reference. Lines 12-15 ensure that we visit each node only once, and thus that each node is in exactly one partition, by checking if a node has been visited before and if so, skipping it, otherwise marking it as visited and coontinuing. Lines 16/24 inserts the current node into the open partition, cutting any created cycles and updating values such as critical path length as outlined in Algorithm [?]. Line 17 tests if the current partition recovery time is greater than our specified limit, with the algorithm used to calculate the recovery time located at Algorithm [?]. If the recovery time is greater we execute lines 18-24, where we remove the just added node to bring our recovery time back under the limit, and then write the partition to a file. Line 18 calculates which signals are primary inputs or outputs for the partition, and promotes them accordingly, with more detail given in Algorithm [?]. Writing the partition to a file simply involves outputting the name, inputs, outputs, and a list of every node in the partition in **BLIF!** format. RemoveNode, on line 19, merely removes the node from the partition's list of nodes rather than fully reversing everything AddNode does. As WriteToFile simply serialises the inputs, outputs and node list this is all that's required with the caveat that some cycles may be cut unecessarily. Lines 31-34 write out the final partially full partition, if there is one. See Algorithm [?] for a description of WriteToFile.

| Algorithm 5 MakeIOList    |                                    |                                                            |  |  |
|---------------------------|------------------------------------|------------------------------------------------------------|--|--|
| Variable                  | Туре                               | Description                                                |  |  |
| partition                 | BlifModel*                         | Partition to create list of primary inputs and outputs for |  |  |
| original Circuit          | BlifModel*                         | Original model                                             |  |  |
| signal                    | Signal*                            |                                                            |  |  |
| signal. source            | BlifNode*                          | The driver for the signal                                  |  |  |
| partition.inputs          | List(BlifNode*)                    | List of primary inputs for the circuit                     |  |  |
| partition. signals        | $Map(string \rightarrow Signal^*)$ | Map from signal name to Signal*                            |  |  |
| original Circuit. signals | $Map(string \rightarrow Signal^*)$ | Map from signal name to Signal*                            |  |  |
| signal.sinks              | List(BlifNode*)                    | List of sinks for the signal                               |  |  |

```
1: procedure MAKEIOLIST(partition, originalCircuit)
       for all signal \in partition.signals do
2:
           if signal.source = NULL then
                                                                               ▶ If this signal has no driver
3:
4:
               partition.inputs.Add(signal)
5:
           end if
           other Signal \leftarrow original Circuit. signals [signal.name] \triangleright Get the corresponding signal in
6:
   the original circuit
           if other Signal.sinks - signal.sinks > 0 then \triangleright If the signal has more sinks in the original
7:
   circuit than it does in this partition
               partition.outputs.Add(signal)
8:
9:
           end if
       end for
10:
11: end procedure
```

We iterate through every signal in our partition. For each one we check if we have a source (line 4), if not it must be a primary input. Similarly, on line 7 we check if we have a sink which is not represented within our partition. If so, promote it to a primary output of the partition.

| Algorithm 7 RecoveryTime | Algorithm 7 RecoveryTime |                                                                                    |  |  |
|--------------------------|--------------------------|------------------------------------------------------------------------------------|--|--|
| Variable                 | Type                     | Description                                                                        |  |  |
| latency                  | float                    | Circuit latency (i.e. time for input to completely propagate to output) in seconds |  |  |
| clockFrequency           | Integer                  | Operating frequency of the circuit, in seconds                                     |  |  |
| critical Path            | Integer                  | Maximum number of steps between an input and an output                             |  |  |
| numFF                    | Integer                  | Number of Latches in circuit                                                       |  |  |
| numLUT                   | Integer                  | Number of look up tables in circuit                                                |  |  |
| resynchronisation Time   | Float                    | Time, in seconds, that it takes to resynchronise circuit                           |  |  |
| detection Time           | Float                    | Time, in seconds, that it takes to detect an error                                 |  |  |
| Reconfigure Time         | Float                    | Time, in seconds, that it takes to reconfigure circuit                             |  |  |
| communication Time       | Float                    | Time, in seconds, that it takes to transmit reconfiguration request                |  |  |
|                          |                          | to controller                                                                      |  |  |

# 1: **procedure** RECOVERYTIME(partition)

- 2:  $latency \leftarrow frequency \times (critical path + 1)$
- 3:  $detectionTime \leftarrow latency$
- 4:  $resynchronisationTime \leftarrow latency$
- 5:  $reconfigurationTime \leftarrow \max(numFF, numLUT)/10/15...morestuff$
- 6:  $communicationTime \leftarrow numPartitions \times latency \times morestuff$
- 7:  $recoveryTime \leftarrow detectionTime + resynchronisationTime + reconfigurationTime + communicationTime$
- 8: return recoveryTime
- 9: end procedure

The derivation of this algorithm and the values used is fully discussed in Section TODO: Reference. The critical path is a measure of the maximum number of latches on a path from input to output. The +1 is to account for the contribution of combinational logic, which may be up to one additional clock cycle of latency.

Lines 4-6 check if one of the inputs signals for this node has been cut to remove a cycle. If so, we rename the signal appropriately, to what the new primary input is called. Additionally, lines 3-11 update the appropriate signals so that this node is reachable within the **DFG!** (**DFG!**). Lines 12-19 then update the maximum path length (or latency in clock cycles) while detecting and cutting any newly created cycles.

| Algorithm 9 AddNode |                                       |                                                                                   |  |
|---------------------|---------------------------------------|-----------------------------------------------------------------------------------|--|
| Variable            | Туре                                  | Description                                                                       |  |
| partition           | BlifModel*                            | BlifModel* containing DFG representing partition to add node to                   |  |
| node                | BlifNode*                             | Node to add                                                                       |  |
| signal              | Signal*                               |                                                                                   |  |
| signal Name         | string                                |                                                                                   |  |
| partition.cut Loops | $Map[string \rightarrow string]$      | For signals which have been cut, a map of the old to the new name                 |  |
| partition. signals  | $Map[string \rightarrow Signal*]$     | Map of signal name to Signal*                                                     |  |
| signal.sinks        | list(BlifNode*)                       | List of sinks for a Signal*                                                       |  |
| signal.source       | BlifNode*                             | Source, or driver, for a Signal*                                                  |  |
| inCost              | int                                   | Maximum number of critical path steps to reach node, not counting the node itself |  |
| explored            | $Map(BlifNode^* \rightarrow boolean)$ | Whether a node has been reached yet in the current iteration                      |  |

```
1: procedure ADDNODE(partition, node)
       nodes.insert(node)
 2:
       for all signalName innode.inputs do
 3:
           if partition.cutLoops[signalName] \neq "" then
 4:
               signalName \leftarrow cutLoops[signalName]  \triangleright If this signal has been renamed already to
 5:
    avoid a cycle, rename this occurance of it.
           end if
 6:
           signal \leftarrow partition.signals[name]
 7:
           signal.sinks.Add(node)
 8:
 9:
       end for
10:
       signal \leftarrow partition.signals[node.output]
       signal.source \leftarrow node
11:
       inCost \leftarrow 0
12:
       for all signalName \in node.inputs do
13:
           source \leftarrow partition.signals[signalName].source
14:
           if partition.costs[source] > inCost then
15:
               inCost \leftarrow partition.costs[source]
16:
17:
           end if
       end for
18:
       UpdateCostsAndBreakCycles(partition, node, NULL, node, inCost, explored, costs)
19:
20: end procedure
```

We care about two things. One, the maximum cost to reach a node, and two, detecting and removing any cycles. Given an existing **DAG!** (**DAG!**) Directed Acyclic Graph which we insert a new node into, then 1. The new node is the root node of a subtree within the **DAG!**. 2. Nodes which are not within the subtree cannot have the maximum cost to reach them change (as nothing has changed in any path to them). 3. Any cycles must pass through the new root node, as there are no cycles elsewhere in the graph. 4. Without any cycles, the root node will only be reached once at the start. Using this information we develop our traversal algorithm. Line 2 demonstrates an optimisation, in that once a path has been checked we need not recheck it unless we have found a more expensive path to it as otherwise nothing will change. Lines 5-8 check if we have detected a cycle. If so, cut it through cutting the signal, detailed in algorithm [?].

| Algorithm 11 | UpdateCostsAndBreakCyc | eles |
|--------------|------------------------|------|
|--------------|------------------------|------|

| Variable                         | Туре                                  | Description                                                                     |
|----------------------------------|---------------------------------------|---------------------------------------------------------------------------------|
| partition                        | BlifModel*                            | BlifModel* containing DFG representing partition to add node to                 |
| root                             | BlifNode*                             | Newly added node                                                                |
| node                             | BlifNode*                             | Node we just came from                                                          |
| node                             | BlifNode*                             | Current node                                                                    |
| costToReach                      | int                                   | Maximum number of critical path                                                 |
|                                  |                                       | steps to reach node, not counting the node itself                               |
| explored                         | $Map(BlifNode^* \rightarrow boolean)$ | Whether a node has been reached yet in the current iteration                    |
| partition.signals                | $Map[string \rightarrow Signal^*]$    | Map of signal name to Signal*                                                   |
| partition.signals[parent.output] | BlifNode*                             | Signal driven by the prior node i.e. the graph edge we most recently travelled. |
| node.cost                        | int                                   | 1 for latches, 0 for LUTs                                                       |
| costs                            | $Map(BlifNode^* \rightarrow int)$     | Map of the cost to reach each node                                              |
| node                             | BlifNode*                             |                                                                                 |
| signal.sinks                     | list(BlifNode*)                       | List of sinks for a Signal*                                                     |
| cost                             | int                                   | Number of critical path steps to reach node, including the node itself          |

```
1: \textbf{procedure} \ \textbf{UPDATECOSTSANDBREAKCYCLES} (partition, root, parent, node, costToReach, explored) \\
```

- :: **if**  $explored[node] = true and <math>costs[node] \ge costToReach$  **then**  $\triangleright$  Already expanded this path, and it can't change
- 3: return
- 4: end if
- 5: **if**  $parent \neq NULL$  and node = root **then**  $\triangleright$  We have a cycle, as all newly created cycles must go through the new node, and the new node should only ever be reached once at the start without cycles
- 6: CutSignal(partition, partition.signals[parent.output])
- 7: return
- 8: **end if**
- 9:  $cost \leftarrow costToReach + node.cost$
- 10: **if** cost > costs[node] **then**
- 11: costs[node] = cost
- 12: **else**
- 13: cost = costs[node]
- 14: **end if**
- 15: **for all**  $node \in partition.signals[node.output].sinks$ **do**
- 16: UpdateCostsAndBreakCycles(root, node, child, cost, explored)
- 17: **end for**
- 18: explored[node] = true
- 19: end procedure

| Algorithm 13 CutSign                                                                                             | nal                                               |                                                                                                                                |  |
|------------------------------------------------------------------------------------------------------------------|---------------------------------------------------|--------------------------------------------------------------------------------------------------------------------------------|--|
| Variable                                                                                                         | Type                                              | Description                                                                                                                    |  |
| partition                                                                                                        | BlifModel*                                        | BlifModel* containing DFG representing partition to add node to                                                                |  |
| partition.cut Loops                                                                                              | $Map(string \rightarrow string)$                  | Map from old to new name of a cut signal. Note that only the input signal is renamed. The output signal retains the same name. |  |
| signal                                                                                                           | Signal*                                           | The signal we're cutting                                                                                                       |  |
| newSignal                                                                                                        | Signal*                                           | The new signal we created                                                                                                      |  |
| signal.source                                                                                                    | BlifNode*                                         | The source, or driver, of a signal                                                                                             |  |
| partition.signals                                                                                                | $Map(string \rightarrow Signal^*)$                | A map from signal name to signal                                                                                               |  |
| partition.outputs                                                                                                | list(Signal*)                                     | The primary outputs for the circuit                                                                                            |  |
| node                                                                                                             | BlifNode*                                         |                                                                                                                                |  |
| node.inputs                                                                                                      | list(string)                                      | names of input signals to a node                                                                                               |  |
|                                                                                                                  |                                                   |                                                                                                                                |  |
| 1: <b>procedure</b> CUTS                                                                                         | IGNAL(partition, signal)                          |                                                                                                                                |  |
| •                                                                                                                | $-newSignal(name) \triangleright C$               | Create a new signal with the same name, but currently no                                                                       |  |
| sources or sinks                                                                                                 |                                                   |                                                                                                                                |  |
| 9                                                                                                                | $ource \leftarrow signal.source$                  |                                                                                                                                |  |
| 4: partition.sign                                                                                                | 4: $partition.signals[name] \leftarrow newSignal$ |                                                                                                                                |  |
| 5: replace $(partition.outputs, signal, newSignal)$ $\triangleright$ Replace any occurances of the old signal in |                                                   |                                                                                                                                |  |
| the circuit outputs                                                                                              |                                                   |                                                                                                                                |  |
|                                                                                                                  | $\leftarrow$ "qqrin" + name                       |                                                                                                                                |  |
| 7: signal.source                                                                                                 |                                                   |                                                                                                                                |  |
|                                                                                                                  | $[al.name] \leftarrow signal$                     |                                                                                                                                |  |
|                                                                                                                  | $vSignal.name] \leftarrow signal$                 | .name                                                                                                                          |  |
| 10: <b>for all</b> $node \in$                                                                                    | signal.sinks <b>do</b>                            |                                                                                                                                |  |

Cutting a signal consists of splitting an existing signal into two. One with the same source but no sinks as a primary output. One with the same sinks but no source as a primary input. We also need to update existing references to point to the correct signal, and record the change to update any future references as nodes get added.

replace(node.inputs, name, "qqrin" + name)

11: 12:

12: end for13: end procedure

#### **3.1** TMR

file.write(voter)

19: end procedure

file.write(circuit)

17:

18:

Given a file containing a partition, read it in as a black box, triplicate it, add voter logic, write back out to file.

| Algorithm 15 TMR    |                                    |                                                                                                                                |
|---------------------|------------------------------------|--------------------------------------------------------------------------------------------------------------------------------|
| Variable            | Туре                               | Description                                                                                                                    |
| partition           | BlifModel*                         | BlifModel* containing DFG representing partition to add node to                                                                |
| partition.cut Loops | $Map(string \rightarrow string)$   | Map from old to new name of a cut signal. Note that only the input signal is renamed. The output signal retains the same name. |
| signal              | Signal*                            | The signal we're cutting                                                                                                       |
| newSignal           | Signal*                            | The new signal we created                                                                                                      |
| signal. source      | BlifNode*                          | The source, or driver, of a signal                                                                                             |
| partition. signals  | $Map(string \rightarrow Signal^*)$ | A map from signal name to signal                                                                                               |
| partition.outputs   | list(Signal*)                      | The primary outputs for the circuit                                                                                            |
| node                | BlifNode*                          |                                                                                                                                |
| node.inputs         | list(string)                       | names of input signals to a node                                                                                               |

```
1: procedure TMR(file)
                           circuit \leftarrow file.read()
    2:
                           header \leftarrow circuit.header
    3:
                           voter \leftarrow voter.read()
   4:
                            subcktDefinition1 \leftarrow MakeSubcktDefinition(circuit)
    5:
                            subcktDefinition2 \leftarrow MakeSubcktDefinition(circuit)
    6:
                            subcktDefinition3 \leftarrow MakeSubcktDefinition(circuit)
    7:
                           voterDefinition \leftarrow MakeSubcktDefinition(voter)
    8:
                            subcktDefinition.ConnectInputs(circuit.inputs)
    9:
                           voterDefinition. ConnectInputs (subcktDefinition1.outputs, subcktDefinition2.outputs, subcktDefinition2.outputs, subcktDefinition2.outputs, subcktDefinition3.outputs, subcktDefiniti
10:
                            voterDefinition.ConnectOutputs(circuit.outputs)
11:
                            file.write(header)
12:
                            file.write(voterDefinition)
13:
                            file.write(subcktDefinition1)
14:
                            file.write(subcktDefinition2)
15:
                            file.write(subcktDefinition3)
16:
```

This method operates on the **BLIF!** in a low level way, dealing manipulating the actual file contents, rather than operating on an abstract circuit representation, as we transform a flat circuit, into a heirarchical circuit, in which out original flat circuit remains untouched but we insert voting and similar logic around it. We read in our partition circuit and voter circuit. We now create three partition subcircuit and one voter subcircuit definitions. We match up our signal names between them appropriately, and then write out our subcircuit definitions, followed by our partition and voter subcircuits.

This transforms a file from format:

```
1 .name partition
2 .inputs ...
3 contents
```

#### Into one in format:

# 3.2 Blif.Join

Given a list of blif files, concatenates them all together, creates subcircuit definitions to connect them all together, and writes them to a file

#### 3.3 Flatten

Algorithm 17 Flatten

```
Variable
           Type
                   Description
           file
                   File to flatten
file
1: procedure FLATTEN(file)
                                            ▶ Flattening is currently performed by abc (link), called with
  parameters:
       ./abc -o output -c echo file > Due to bug in abc, clock information is stripped from latches, so we
  then call grep and sed to fix the output file
      latch \leftarrow split(grep - m 1 'latch' file)
3:
      if latch then
4:
          sed -ri 's/(\.latch.+)(2)/\1 '+latch[3] + ' ' + latch[4] + ' 2/' output
5:
      end if
6:
7: end procedure
```

BEGIN Flatten(file) END Flatten(file) ./abc is provided an input file, given the command to echo the current file, and told to output everything to output grep is called to search for latches, and return

the latch information if there is one. If there is, replace the faulty latch information with the correct information. This assumes that there is only one global clock, all latches are triggered on the same signal (e.g. rising edge, falling etc), and all latches have initial state don't care, which holds true for all provided benchmarks.