~arcachofo/simulide/1.1.0

« back to all changes in this revision

Viewing changes to src/simulator/elements/logic/e-shiftreg.cpp

  • Committer: arcachofo
  • Date: 2021-01-01 14:23:42 UTC
  • Revision ID: arcachofo@simulide.com-20210101142342-ozfljnll44g5lbl3
Initial Commit 0.5.15-RC3

Show diffs side-by-side

added added

removed removed

Lines of Context:
 
1
/***************************************************************************
 
2
 *   Copyright (C) 2012 by santiago González                               *
 
3
 *   santigoro@gmail.com                                                   *
 
4
 *                                                                         *
 
5
 *   This program is free software; you can redistribute it and/or modify  *
 
6
 *   it under the terms of the GNU General Public License as published by  *
 
7
 *   the Free Software Foundation; either version 3 of the License, or     *
 
8
 *   (at your option) any later version.                                   *
 
9
 *                                                                         *
 
10
 *   This program is distributed in the hope that it will be useful,       *
 
11
 *   but WITHOUT ANY WARRANTY; without even the implied warranty of        *
 
12
 *   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the         *
 
13
 *   GNU General Public License for more details.                          *
 
14
 *                                                                         *
 
15
 *   You should have received a copy of the GNU General Public License     *
 
16
 *   along with this program; if not, see <http://www.gnu.org/licenses/>.  *
 
17
 *                                                                         *
 
18
 ***************************************************************************/
 
19
 
 
20
#include "e-shiftreg.h"
 
21
#include "simulator.h"
 
22
 
 
23
eShiftReg::eShiftReg( QString id, int latchClk, int serOut ) 
 
24
         : eLogicDevice( id )
 
25
{
 
26
    // input0: DS    serial data input
 
27
    // input1: MR    master reset (active LOW)
 
28
    // packagePin15-QH' 
 
29
    
 
30
    m_latchClock = false;
 
31
    m_changed    = false;
 
32
    m_reset      = false;
 
33
    
 
34
    m_latchClockPin = 0l;
 
35
    m_serOutPin     = 0l;
 
36
    
 
37
    if( latchClk > 0 ) createLatchClockPin();
 
38
    if( serOut > 0 )   createSerOutPin();
 
39
 
 
40
    initialize();
 
41
}
 
42
eShiftReg::~eShiftReg()
 
43
{
 
44
    delete m_latchClockPin;
 
45
    delete m_serOutPin;
 
46
}
 
47
 
 
48
void eShiftReg::initialize()
 
49
{
 
50
    m_shiftReg.reset();
 
51
    m_latch.reset();
 
52
    m_latchClock = false;
 
53
    m_changed    = false;
 
54
    m_setSerOut  = false;
 
55
    m_resSerOut  = false;
 
56
    
 
57
    if( m_serOutPin )                              // Set Serial Out Pin
 
58
    {
 
59
        m_serOutPin->setOut( false );
 
60
        m_serOutPin->stampOutput();
 
61
    }
 
62
    eLogicDevice::initialize();
 
63
}
 
64
 
 
65
void eShiftReg::stamp()
 
66
{
 
67
    eNode* enode = m_input[1]->getEpin()->getEnode(); // m_input[1] = Reset pin
 
68
    if( enode ) enode->voltChangedCallback( this );
 
69
 
 
70
    if( m_latchClockPin )
 
71
    {
 
72
        enode = m_latchClockPin->getEpin()->getEnode();
 
73
        if( enode ) enode->voltChangedCallback( this );
 
74
    }
 
75
    eLogicDevice::stamp();
 
76
}
 
77
 
 
78
void eShiftReg::voltChanged()
 
79
{
 
80
    eLogicDevice::updateOutEnabled();
 
81
 
 
82
    bool clkRising = (eLogicDevice::getClockState() == Clock_Rising);// Get Clk to don't miss any clock changes
 
83
    double    volt = m_input[1]->getEpin()->getVolt();// Reset pin volt.
 
84
    bool     reset = eLogicDevice::getInputState( 1 );// m_input[1] = Reset
 
85
    bool  addEvent = false;
 
86
 
 
87
    if( reset != m_reset  )
 
88
    {
 
89
        m_reset = reset;
 
90
        if( reset )
 
91
        {
 
92
            m_shiftReg.reset();                  // Reset shift register  
 
93
            if( m_serOutPin ) m_resSerOut = true;// Reset Serial Out Pin
 
94
            addEvent = true;
 
95
        }
 
96
        m_changed = true;                      // Shift Register changed
 
97
    }
 
98
    else if( clkRising && !reset )                  // Clock rising edge
 
99
    {
 
100
        // Shift bits 7-1
 
101
        for( int i=7; i>0; --i )m_shiftReg[i] = m_shiftReg[i-1];
 
102
        
 
103
        if( m_serOutPin )  m_setSerOut = true;     // Set Serial Out Pin
 
104
 
 
105
        // Read data input pin & put in reg bit0
 
106
        volt = m_input[0]->getEpin()->getVolt();      // Reset pin volt.
 
107
        m_shiftReg[0] = (volt > m_inputHighV);     // input0: data input
 
108
 
 
109
        m_changed = true;                      // Shift Register changed
 
110
        addEvent = true;
 
111
    }
 
112
    
 
113
    if( m_latchClockPin )                // find Latch Clock rising edge
 
114
    {
 
115
        bool lastLcClock = m_latchClock;
 
116
 
 
117
        double volt = m_latchClockPin->getVolt();      // Clock pin volt.
 
118
 
 
119
        if     ( volt > m_inputHighV ) m_latchClock = true;
 
120
        else if( volt < m_inputLowV )  m_latchClock = false;
 
121
 
 
122
        if( m_latchClock && !lastLcClock && m_changed )
 
123
        {
 
124
            for( int i=0; i<8; ++i )   // Store Shift-reg in Output Latch
 
125
            {
 
126
                bool data = m_shiftReg[i];
 
127
                m_latch[i] = data;
 
128
            }
 
129
            m_changed = false;
 
130
            addEvent = true;
 
131
        }
 
132
    }else if( m_changed )
 
133
    {
 
134
        m_changed = false;
 
135
        addEvent = true;
 
136
    }
 
137
    if( addEvent ) Simulator::self()->addEvent( m_propDelay, this );
 
138
}
 
139
 
 
140
void eShiftReg::runEvent()
 
141
{
 
142
    if( ! m_changed )
 
143
    {
 
144
        if( m_latchClockPin )
 
145
            for( int i=0; i<8; ++i )  setOut( i, m_latch[i] );
 
146
        else
 
147
            for( int i=0; i<8; ++i )  setOut( i, m_shiftReg[i] );
 
148
    }
 
149
    if( m_resSerOut )
 
150
    {
 
151
        m_serOutPin->setOut( false );
 
152
        m_serOutPin->stampOutput();
 
153
        m_resSerOut = false;
 
154
    }
 
155
    if( m_setSerOut )
 
156
    {
 
157
        m_serOutPin->setOut( m_shiftReg[7] );
 
158
        m_serOutPin->stampOutput();
 
159
        m_setSerOut = false;
 
160
    }
 
161
}
 
162
 
 
163
void eShiftReg::createLatchClockPin()
 
164
{
 
165
    ePin* epin      = new ePin( m_elmId+"latchClockPin", 0 );
 
166
    m_latchClockPin = new eSource( m_elmId+"eSourceLatchClock", epin );
 
167
    m_latchClockPin->setImp( m_inputImp );
 
168
}
 
169
 
 
170
void eShiftReg::createSerOutPin()
 
171
{
 
172
    ePin* epin  = new ePin( m_elmId+"serOutPin", 0 );
 
173
    m_serOutPin = new eSource( m_elmId+"eSourceSerOutPin", epin );
 
174
    m_serOutPin->setVoltHigh( m_outHighV );
 
175
    m_serOutPin->setImp( m_outImp );
 
176
}
 
177
 
 
178
void eShiftReg::setResetInv( bool inv )
 
179
{
 
180
    m_resetInv = inv;
 
181
    m_input[1]->setInverted( inv );
 
182
}