Home / Software Lab 3 (Wall Finder) / From Block Diagrams to Code

From Block Diagrams to Code

You are not logged in.

If you are a current student, please Log In for full access to this page.

Block diagrams are modular and allow for relatively easy manipulation of system functionals; by noticing certain patterns (such as the cascade, feedback addition, and feedforward addition discussed in lecture), we can convert a block diagram to a system functional. Our simulation framework will mimic block diagrams, and we will implement these patterns as our means of combination (with Gains and Delays as our primitives), so that finding a system's system functional programmatically is only a matter of recognizing these patterns (and we can leave the algebra to the computer!).

1) Example: Simulating Gain

The power of the framework this page details is in simulating large systems. The algebraic tedium can be done by a computer. However, to get a sense how how to use this framework, we will begin with a very simple example, simulating the following system:

We can create a gain block with an instance of the Gain class, whose initializer takes as its sole argument a number representing the value k of the gain block:

g = Gain(k)

Now the variable g is storing our model of the above block diagram. This is similar to manually writing out the difference equation coeficients.

To simulate this system, first we construct a simulator

g_sim = SystemSimulator(g)

Next we can run this simulator given some input.

g_in = [1] + [0]*9 # equivalent to [1,0,0,0,0,0,0,0,0]
g_out = g_sim.get_response(g_in)

g_out now stores a list of samples of the output of the system g given the input g_in This evalutes to

[k,0,0,0,0,0,0,0,0]

2) Primitives

Our primitives include gains and delays.

2.1) Gain

You can represent a gain block with an instance of the Gain class, whose initializer takes as its sole argument a number representing the value k of the gain block:

g = Gain(k)

2.2) Delay (R)

We can also represent delays with instances of the R class, whose initializer takes a single argument representing its output on the 0th timestep (note that, after the first timestep, its output at time n will be its input from time n-1):

r = R(initial_output)

3) Combinations

We can combine systems together using the following classes, representing the various patterns that were discussed in lecture:

3.1) Cascade

c = Cascade(s1, s2)

3.2) Feedforward Addition

f = FeedforwardAdd(s1, s2)

3.3) Feedback Addition

f = FeedbackAdd(s1, s2)

4) Simulating

Once the model has been constructed, you may want to simulate it to see how it responds to various inputs. You can create a simulator from one of these models by constructing a SystemSimulator from the model. The resulting SystemSimulator has a method called get_response that can compute the system's response to provided input.

5) Example

For example, consider the Fibonacci's Bunnies system from lecture 2:

We can model this system (when started from rest) with the following code:

upper_subsystem = FeedbackAdd(Gain(1), R(0))
lower_subsystem = Cascade(R(0), R(0))
bunnies_model = FeedbackAdd(upper_subsystem, lower_subsystem)

This creates a model of the system, which mimics the structure of the block diagram. To compute the unit sample response of this system, we could use the following code, which constructs a simulator to simulate our model:

bunnies_sim = SystemSimulator(bunnies_model)
bunnies_usr = bunnies_sim.get_response([1] + [0]*9)