|
def current_continuity(list_of_nodes, branch_info, branch_stiff, nd_voltage, br_currents, mho_matrix, src_vector, sys_inputs): |
|
"""Following an event, to find the branch currents through nodal analysis. |
|
The objective is to determine if certain devices must change their status |
|
to maintain continuity of inductor currents.""" |
|
|
|
# Node voltages. There will be one reference node |
|
for c1 in range(len(nd_voltage)): |
|
nd_voltage[c1]=0.0 |
|
|
|
# The admittance matrix. As it is more or less |
|
# a dc analysis of a snapshot of the circuit |
|
# only resistances are considered. |
|
for c1 in range(len(mho_matrix)): |
|
for c2 in range(len(mho_matrix)): |
|
mho_matrix[c1][c2]=0.0 |
|
|
|
# The source vector. This will contain |
|
# currents sources - inductor currents, |
|
# currents through zero impedance branches |
|
# and finally also voltages across zero |
|
# impedance branches |
|
for c1 in range(len(src_vector)): |
|
src_vector[c1]=0.0 |
|
|
|
|
|
# The concept is of nodal analysis. |
|
# If a branch has only a resistance the currents |
|
# are expressed as a difference of node voltages divided |
|
# by the resistance. |
|
# If a branch has an inductance and it is non stiff, |
|
# it is treated as a current source. This is the main |
|
# feature as the inductor current must not change |
|
# instantaneously. |
|
# If it has zero impedance, it is treated as a current |
|
# source. This was done later as the currents through |
|
# these branches can't be accounted for by nodal analysis |
|
# and these currents aren't suppose to change anyway. |
|
# It is only the nonlinear devices that are essentially |
|
# variable resistances that are supposed to have changing |
|
# currents. |
|
|
|
|
|
# Start iterating through the nodes |
|
for c1 in range(len(list_of_nodes)): |
|
# Look for the node as the starting or ending node in branches |
|
for c2 in range(len(branch_info)): |
|
if ((list_of_nodes[c1]==branch_info[c2][0]) or (list_of_nodes[c1]==branch_info[c2][-2])): |
|
# Mark the position of the other node |
|
if (list_of_nodes[c1]==branch_info[c2][0]): |
|
end_node_pos=list_of_nodes.index(branch_info[c2][-2]) |
|
# The default direction of current is taken to be |
|
# away from the starting node - start to end. |
|
branch_src_dir=1.0 |
|
else: |
|
end_node_pos=list_of_nodes.index(branch_info[c2][0]) |
|
branch_src_dir=-1.0 |
|
|
|
|
|
# If resistance of the branch is non-zero |
|
if branch_info[c2][-1][0][0]: |
|
# If the branch has no inductance or is a stiff branch |
|
# in which case inductance is negligible compared |
|
# to the resistance and the branch is stiff. |
|
if (branch_info[c2][-1][0][1]==0.0): |
|
mho_matrix[c1][c1]+=1.0/branch_info[c2][-1][0][0] |
|
mho_matrix[c1][end_node_pos]-=1.0/branch_info[c2][-1][0][0] |
|
|
|
# Similarly, any voltage source that exists |
|
# will be treated as positive if it forces a |
|
# a current away the starting node. |
|
# The way branch_info contains these sources is |
|
# compatible to the definition |
|
# src_vector is on the RHS of the equation, |
|
# so thus the -ve sign |
|
for c3 in range(len(branch_info[c2][-1][1])): |
|
src_vector[c1]-=branch_src_dir*branch_info[c2][-1][1][c3]*sys_inputs.data[c3][0]/branch_info[c2][-1][0][0] |
|
|
|
|
|
|
|
# Check if inductance is non-zero |
|
if branch_info[c2][-1][0][1]: |
|
# Check if branch is not stiff |
|
if branch_stiff[c2]=="no": |
|
# Add the current to the src_vector as a current source |
|
src_vector[c1]-=branch_src_dir*br_currents[c2] |
|
|
|
|
|
# Check if is a zero impedance branch |
|
if ((branch_info[c2][-1][0][0]==0.0) and (branch_info[c2][-1][0][1]==0.0)): |
|
src_vector[c1]-=branch_src_dir*br_currents[c2] |
|
|
|
|
|
|
|
# There are some rows that are exact negatives of the other for the reason, that |
|
# they are essentially reference nodes. So these need not be included |
|
for c1 in range(len(mho_matrix)-1): |
|
for c2 in range(c1+1, len(mho_matrix)): |
|
rows_the_same="yes" |
|
for c3 in range(len(mho_matrix[c1])): |
|
if (mho_matrix[c1][c3]+mho_matrix[c2][c3]): |
|
rows_the_same="no" |
|
|
|
if rows_the_same=="yes": |
|
src_vector[c2]=0.0 |
|
for c3 in range(len(mho_matrix[c2])): |
|
mho_matrix[c2][c3]=0.0 |
|
|
|
|
|
|
|
# Finding the supernodes in the system |
|
# Supernodes are essentially those nodes that |
|
# have a zero impedance branch connected to them |
|
# These branches may or may not have a voltage source. |
|
# When there is a supernode, KCL continues to the other node |
|
# of the zero impedance branch. So essentially the two end |
|
# nodes of a zero impedance branch have their KCL equations |
|
# added and their node voltages are expressed by an equation |
|
# depending on whether they have a voltage source. |
|
|
|
# The complete list of supernodes |
|
supernode_list=[] |
|
for c1 in range(len(list_of_nodes)): |
|
# Each small segment of supernodes |
|
current_supernode=[] |
|
|
|
# Check if the node has been found as a supernode before |
|
node_found="no" |
|
for c2 in range(len(supernode_list)): |
|
if c1 in supernode_list[c2]: |
|
node_found="yes" |
|
|
|
# If not add it as a supernode |
|
# This is the first node in the current |
|
# running list. If no other nodes are found, |
|
# it will be deleted at the next iteration |
|
# as the node is not a supernode in that case. |
|
if node_found=="no": |
|
current_supernode.append(c1) |
|
|
|
# Iterate through the remaining nodes |
|
for c2 in range(len(list_of_nodes)): |
|
# Look for the nodes in both the |
|
# current list and the complete list |
|
node_found="no" |
|
if c2 in current_supernode: |
|
node_found=="yes" |
|
|
|
if node_found=="no": |
|
for c3 in range(len(supernode_list)): |
|
if c2 in supernode_list[c3]: |
|
node_found="yes" |
|
|
|
# If the node has not been found, check if it is connected |
|
# to any of the existing nodes in the current list |
|
# by branches with zero impedance. |
|
if node_found=="no": |
|
for c3 in range(len(branch_info)): |
|
# Check if a branch has zero impedance. |
|
if ((branch_info[c3][-1][0][0]==0.0) and (branch_info[c3][-1][0][1]==0.0)): |
|
# If the current node is the first node in the current |
|
# branch being examined, check if any of the nodes |
|
# in the current supernode list are the ending nodes. |
|
# If so append the current node to the current super node list |
|
if (list_of_nodes[c2]==branch_info[c3][0]): |
|
for c4 in current_supernode: |
|
if list_of_nodes[c4]==branch_info[c3][-2]: |
|
current_supernode.append(c2) |
|
|
|
# If the current node is the ending node in the current |
|
# branch being examined, check if any of the nodes |
|
# in the current supernode list are the starting nodes. |
|
# If so append the current node to the current super node list |
|
if (list_of_nodes[c2]==branch_info[c3][-2]): |
|
for c4 in current_supernode: |
|
if list_of_nodes[c4]==branch_info[c3][0]: |
|
current_supernode.append(c2) |
|
|
|
|
|
# There is another possibility |
|
# A node could be a supernode, but is connected only |
|
# to another node which appears later in the node list |
|
# So, the above calculation will not work as it looks |
|
# for a node connected to existing supernodes by zero |
|
# impedance branches. |
|
# So essentially it is a rerun to pickup any nodes |
|
# that were missed the first time. |
|
|
|
# Iterate through existing list of current supernodes. |
|
for c2 in current_supernode: |
|
# Then look through the entire list of nodes. |
|
for c3 in range(len(list_of_nodes)): |
|
# Check it is has been found as a supernode |
|
# in the current list |
|
node_found="no" |
|
if c3 in current_supernode: |
|
node_found="yes" |
|
|
|
# Check it has been found elsewhere. |
|
if node_found=="no": |
|
for c4 in range(len(supernode_list)): |
|
if c3 in supernode_list[c4]: |
|
node_found="yes" |
|
|
|
|
|
# Check if the current node is connected to any of the nodes |
|
# in the current supernode list by a zero impedance branch. |
|
if node_found=="no": |
|
for c5 in range(len(branch_info)): |
|
if ((branch_info[c5][-1][0][0]==0.0) and (branch_info[c5][-1][0][1]==0.0)): |
|
if (list_of_nodes[c3]==branch_info[c5][0]): |
|
for c6 in current_supernode: |
|
if (list_of_nodes[c6]==branch_info[c5][-2]): |
|
current_supernode.append(c3) |
|
|
|
if (list_of_nodes[c3]==branch_info[c5][-2]): |
|
for c6 in current_supernode: |
|
if (list_of_nodes[c6]==branch_info[c5][0]): |
|
current_supernode.append(c3) |
|
|
|
|
|
# Sort the list of current supernodes |
|
# The reason is so that the KCL equations can be added |
|
# and the first node will contain the sum. |
|
current_supernode.sort() |
|
|
|
# # If the current super node list has more than |
|
# # one node, there is a zero impedance branch |
|
# # and so it is super node list. |
|
if len(current_supernode)>1: |
|
supernode_list.append(current_supernode) |
|
|
|
|
|
|
|
# For every supernode list, |
|
# add the KCL equations at subsequnt supernodes to the |
|
# first node in the supernode list. Then set the other |
|
# equations to zero. Also, with the source vector. |
|
for c1 in range(len(supernode_list)): |
|
row1=supernode_list[c1][0] |
|
for c2 in range(1, len(supernode_list[c1])): |
|
row2=supernode_list[c1][c2] |
|
src_vector[row1]+=src_vector[row2] |
|
src_vector[row2]=0.0 |
|
for c3 in range(len(mho_matrix[0])): |
|
mho_matrix[row1][c3]+=mho_matrix[row2][c3] |
|
mho_matrix[row2][c3]=0.0 |
|
|
|
|
|
# Make a list of the zero rows. |
|
zero_rows=[] |
|
for c1 in range(len(mho_matrix)): |
|
is_row_zero="yes" |
|
for c2 in range(len(mho_matrix[c1])): |
|
if mho_matrix[c1][c2]: |
|
is_row_zero="no" |
|
|
|
if is_row_zero=="yes": |
|
zero_rows.append(c1) |
|
|
|
|
|
# Move the zero rows to the end of the matrix by row interchanges. |
|
for c1 in zero_rows: |
|
for c2 in range(len(mho_matrix)-1, c1, -1): |
|
if c2 not in zero_rows: |
|
src_vector[c1], src_vector[c2] = src_vector[c2], src_vector[c1] |
|
for c3 in range(len(mho_matrix[c1])): |
|
mho_matrix[c1][c3], mho_matrix[c2][c3] = mho_matrix[c2][c3], mho_matrix[c1][c3] |
|
|
|
|
|
# Now to add equations that describe the voltage difference |
|
# between two nodes connected by a zero impedance branch |
|
|
|
# The starting row if the size of the addmitance matrix |
|
# minus the zero rows. |
|
supernode_row=len(mho_matrix)-len(zero_rows) |
|
# Iterate through the supernode lists |
|
for c1 in range(len(supernode_list)): |
|
# In every supernode list, take a supernode |
|
# and express its voltage with respect to every |
|
# other supernode in that list |
|
for c2 in range(len(supernode_list[c1])-1): |
|
for c3 in range(c2+1, len(supernode_list[c1])): |
|
# Find out which supernode is the start and which is |
|
# the end node of the branch. |
|
for c4 in range(len(branch_info)): |
|
if ((branch_info[c4][0]==list_of_nodes[supernode_list[c1][c2]]) and (branch_info[c4][-2]==list_of_nodes[supernode_list[c1][c3]])): |
|
# Confirm whether the branch is a zero impedance branch |
|
if (branch_info[c4][-1][0][0]==0.0 and branch_info[c4][-1][0][1]==0.0): |
|
# This is mere V1-V2=Vbranch |
|
# The plus and minus signs are for different cases of |
|
# starting node and ending node. |
|
mho_matrix[supernode_row][supernode_list[c1][c2]]=1.0 |
|
mho_matrix[supernode_row][supernode_list[c1][c3]]=-1.0 |
|
for c5 in range(len(branch_info[c4][-1][1])): |
|
src_vector[supernode_row]+=branch_info[c4][-1][1][c5]*sys_inputs.data[c5][0] |
|
|
|
# Increment the supernode row pointer |
|
supernode_row+=1 |
|
|
|
if ((branch_info[c4][-2]==list_of_nodes[supernode_list[c1][c2]]) and (branch_info[c4][0]==list_of_nodes[supernode_list[c1][c3]])): |
|
if (branch_info[c4][-1][0][0]==0.0 and branch_info[c4][-1][0][1]==0.0): |
|
mho_matrix[supernode_row][supernode_list[c1][c2]]=-1.0 |
|
mho_matrix[supernode_row][supernode_list[c1][c3]]=1.0 |
|
for c5 in range(len(branch_info[c4][-1][1])): |
|
src_vector[supernode_row]-=branch_info[c4][-1][1][c5]*sys_inputs.data[c5][0] |
|
|
|
supernode_row+=1 |
|
|
|
|
|
|
|
# Now to solve the equation AX=B |
|
|
|
# Look for diagonal elements - check if they are zero |
|
# If so, look in that same column in subsequent rows |
|
# if there is a non zero element and exchange them |
|
# Later, make the matrix upper triangular. |
|
# Using row manipulations, make all the elements |
|
# below a diagonal row zero. |
|
for c1 in range(len(mho_matrix)): |
|
if not mho_matrix[c1][c1]: |
|
for c2 in range(c1+1, len(mho_matrix)): |
|
if mho_matrix[c2][c1]: |
|
src_vector[c1], src_vector[c2] = src_vector[c2], src_vector[c1] |
|
for c3 in range(len(mho_matrix[c1])): |
|
mho_matrix[c1][c3], mho_matrix[c2][c3] = mho_matrix[c2][c3], mho_matrix[c1][c3] |
|
|
|
if mho_matrix[c1][c1]: |
|
for c2 in range(c1+1, len(mho_matrix)): |
|
if mho_matrix[c2][c1]: |
|
src_vector[c2]-=src_vector[c1]*mho_matrix[c2][c1]/mho_matrix[c1][c1] |
|
for c3 in range(len(mho_matrix[c1])): |
|
mho_matrix[c2][c3]-=mho_matrix[c2][c1]*mho_matrix[c1][c3]/mho_matrix[c1][c1] |
|
|
|
|
|
|
|
# The last row of the manipulated admittance matrix will |
|
# be zero as it is the reference node. So taking this |
|
# voltage to be zero, calculate all the other node voltages |
|
# from second last to first. |
|
for c1 in range(len(mho_matrix)-2, -1, -1): |
|
if mho_matrix[c1][c1]: |
|
nd_voltage[c1]=src_vector[c1] |
|
for c2 in range(c1+1, len(mho_matrix[c1])): |
|
nd_voltage[c1]-=mho_matrix[c1][c2]*nd_voltage[c2] |
|
|
|
nd_voltage[c1]=nd_voltage[c1]/mho_matrix[c1][c1] |
|
|
|
|
|
# Calculate the branch currents as I=(Vnode1-Vnode2-Vsource)/Rbranch |
|
for c1 in range(len(branch_info)): |
|
if branch_stiff=="yes" or branch_info[c1][-1][0][1]==0.0: |
|
start_node=list_of_nodes.index(branch_info[c1][0]) |
|
end_node=list_of_nodes.index(branch_info[c1][-2]) |
|
|
|
if branch_info[c1][-1][0][0]: |
|
br_currents[c1]=(nd_voltage[start_node]-nd_voltage[end_node])/branch_info[c1][-1][0][0] |
|
for c2 in range(len(branch_info[c1][-1][1])): |
|
br_currents[c1]+=branch_info[c1][-1][1][c2]*sys_inputs.data[c2][0]/branch_info[c1][-1][0][0] |
|
|
|
|
|
return |