amc2020:group_n:sn74hc595n
Differences
This shows you the differences between two versions of the page.
Both sides previous revisionPrevious revisionNext revision | Previous revision | ||
amc2020:group_n:sn74hc595n [2020/07/18 19:05] – jonas001 | amc2020:group_n:sn74hc595n [2023/01/05 14:38] (current) – external edit 127.0.0.1 | ||
---|---|---|---|
Line 1: | Line 1: | ||
+ | < | ||
+ | < | ||
+ | <a href=" | ||
+ | <span style=" | ||
+ | ↩ Back to the main page | ||
+ | </ | ||
+ | </a> | ||
+ | </ | ||
+ | </ | ||
+ | |||
====== SN74HC595(N) Shift Register ====== | ====== SN74HC595(N) Shift Register ====== | ||
Line 12: | Line 22: | ||
| **//Figure 1//** SN74HC595(N) Shift Register and pinout. | | **//Figure 1//** SN74HC595(N) Shift Register and pinout. | ||
- | ===== Working Principle ===== | + | ===== 2. Working Principle ===== |
The MCU issues a clock signal (SRCLK) to the shift register and sends the data (SER) bitwise in 8-bit groups to the shift register (figure 2). If afterwards another byte is sent, the shift register outputs the first byte through the serial output pin (QH’), which may be connected to another shift register receiving the data. So, the shift register always stores the latest byte and if it receives a new byte, it shifts the old one to the next shift register. | The MCU issues a clock signal (SRCLK) to the shift register and sends the data (SER) bitwise in 8-bit groups to the shift register (figure 2). If afterwards another byte is sent, the shift register outputs the first byte through the serial output pin (QH’), which may be connected to another shift register receiving the data. So, the shift register always stores the latest byte and if it receives a new byte, it shifts the old one to the next shift register. | ||
Line 23: | Line 33: | ||
| **//Figure 2//** SN74HC595(N) pinout diagram (left) and logic diagram (right), there are different names and abbreviations for the pins depending on the source; the labels in the logic diagram are the official ones from the datasheet (also see table 2). | | | **//Figure 2//** SN74HC595(N) pinout diagram (left) and logic diagram (right), there are different names and abbreviations for the pins depending on the source; the labels in the logic diagram are the official ones from the datasheet (also see table 2). | | ||
+ | ===== 3. Technical Specifications and Setup of the Module ===== | ||
+ | |||
+ | The ESP32 can supply 3.3 V, which is in the recommended operating range of the SN74HC595 (table 1). The shift register consumes a maximum of 80 μA which occurs at a supply voltage of 6 V while the shift register is actively used; that means during data transmission or transfer from shift register to storage register. Normally the module consumes much less current. The DHT-22 alone consumes 50 μA during idle state continuously which is much higher than the current in the idle shift register. The more sensors are connected to the shift register, the higher the efficiency of the sensor/ | ||
+ | |||
+ | It is important to note that the continuous output current should be limited to a maximum of 70 mA. As the voltage drops when more current is drawn, the current should be limited to about 5 mA per pin. As both the DHT-22 and the DS18B20 only use about 1.5 mA during measuring, this is not an issue. | ||
+ | |||
+ | An overview over the pins and what they do can be found in table 2. When setting up the module the notch helps orienting the shift register in the right direction. The pins were connected the following way: | ||
+ | |||
+ | * VDD to 3.3V | ||
+ | * GND to GND | ||
+ | * OE to GND | ||
+ | * SRCLR to 3.3V | ||
+ | * SRCLK to GPIO 4 | ||
+ | * RCLK to GPIO 2 | ||
+ | * SER to GPIO 0 | ||
+ | |||
+ | Furthermore, | ||
+ | |||
+ | The parallel outputs can be connected to any device operated at 3.3 V which draws less than 5 mA. To test the working principle of the shift register, the outputs can be connected to red LEDs with a sufficient resistor in series: | ||
+ | |||
+ | $$R=\frac{V_{DD}-U_F}{I_{LED}}=\mathrm{\frac{3.3V-1.7V}{0.005A}=320\Omega}$$ | ||
+ | |||
+ | The next bigger resistor (usually 330Ω) should be used. After testing, in the real application, | ||
+ | |||
+ | ^ **//Table 1 //** SN74HC595(N) specifications ^^ | ||
+ | ^ Module | ||
+ | | Supply Voltage VDD | 2 – 6 V | | ||
+ | | Operating Temperature | ||
+ | | Power consumption | ||
+ | |||
+ | Further information can be found in the {{https:// | ||
+ | |||
+ | ^ **//Table 2 //** SN74HC595(N) pin overview and description | ||
+ | ^ Pin ^ Label ^ Alternative Labels | ||
+ | | 1 | QB | Q2 | Parallel Output 2 | | ||
+ | | 2 | QC | Q3 | Parallel Output 3 | | ||
+ | | 3 | QD | Q4 | Parallel Output 4 | | ||
+ | | 4 | QE | Q5 | Parallel Output 5 | | ||
+ | | 5 | QF | Q6 | Parallel Output 6 | | ||
+ | | 6 | QG | Q7 | Parallel Output 7 | | ||
+ | | 7 | QH | Q8 | Parallel Output 8 | | ||
+ | | 8 | GND | - | Ground (0 V) | | ||
+ | | 9 | QH’ | Q8’ | Serial Data Output | ||
+ | | 10 | SRCLR | MR | Shift Register Clear / Master Reset | | ||
+ | | 11 | SRCLK | Clock / SHCP | Shift Register Clock | | ||
+ | | 12 | RCLK | Latch / STCP | Storage Register Clock | | ||
+ | | 13 | OE | - | Output Enable | ||
+ | | 14 | SER | Data / DS | Serial Data Input | | ||
+ | | 15 | QA | Q1 | Parallel Output 1 | | ||
+ | | 16 | VDD | - | Power Supply (3.3V) | ||
+ | |||
+ | ===== 4. The Code ===== | ||
+ | |||
+ | The following code is to test the function of the shift register with LEDs to see if it is working properly. | ||
+ | To see the sketch working, the shift register needs to be connected as described above and red LEDs have to be connected with a 330Ω resistor in series to the pins QA, QB and QC. | ||
+ | |||
+ | ==== 4.1 Code for the Shift Register ===== | ||
+ | |||
+ | <file c++ ShiftRegisterFunction.ino> | ||
+ | //ESP32 74HC595 Shift Register Test with LEDs | ||
+ | |||
+ | //ESP32 PINS | ||
+ | const int LATCHPIN = 2; //1 | ||
+ | const int CLOCKPIN = 4; | ||
+ | const int DATAPIN = 0; | ||
+ | |||
+ | byte sensor_numbers[9] = { //2 | ||
+ | B00000000, | ||
+ | B00000001, | ||
+ | B00000010, | ||
+ | B00000100, | ||
+ | B00001000, | ||
+ | B00010000, | ||
+ | B00100000, | ||
+ | B01000000, | ||
+ | B10000000, | ||
+ | }; | ||
+ | |||
+ | void setup() { | ||
+ | pinMode(LATCHPIN , | ||
+ | pinMode(CLOCKPIN ,OUTPUT); | ||
+ | pinMode(DATAPIN ,OUTPUT); | ||
+ | } | ||
+ | |||
+ | void loop() { | ||
+ | powerSwitch(0); | ||
+ | delay(1000); | ||
+ | powerSwitch(1); | ||
+ | delay(1000); | ||
+ | powerSwitch(2); | ||
+ | delay(1000); | ||
+ | powerSwitch(3); | ||
+ | delay(1000); | ||
+ | } | ||
+ | |||
+ | //Power Supply Function | ||
+ | void powerSwitch(byte pin){ //5 | ||
+ | digitalWrite(LATCHPIN, | ||
+ | shiftOut(DATAPIN, | ||
+ | digitalWrite(LATCHPIN, | ||
+ | } | ||
+ | </ | ||
+ | |||
+ | ===== 4.2. Explanation of the Code ===== | ||
+ | |||
+ | - The GPIO for latch, clock and data input were chosen as described before. | ||
+ | - The array sensor_numbers is used to configure different configurations for which pins should be high. the pattern of 0s and 1s in the binary numbers directly shows which pins are powered and which are not. A 1 means that it is switched on, a 0 means it is switched off. The most significant bit refers to pin QH/Q8 and the least significant bit refers to QA/Q1. Any combination of pins to be powered can be chosen by adjusting the binary numbers in the array accordingly. The elements in an array are zero indexed, so the first element has the index 0, the second has index 1 and so on. Here the elements were chosen such that the index 0 put all outputs low and the index 1 to 8 put only the corresponding pin high. | ||
+ | - The shift register receives latch, data and clock as input, so the GPIOs need to be configured as outputs. | ||
+ | - Here the function powerSwitch(), | ||
+ | - The function powerSwitch() expects a byte type number as argument which represents the parallel output pin to be switched on. | ||
+ | - Putting the latch pin at a low voltage level results in the storage register not being actualized while the new byte is shifted into the shift register; this prevents a disturbance of the output pins. | ||
+ | - The function shiftOut() is already included in the Arduino library and is used for shift registers. As argument, it expects the data pin, the clock pin, the bit order and a value. The pins are given in the int variables in (1). The bit order defines in in which order the bits are shifted out. It can be either MSBFIRST, so the most significant bit is shifted out first, or LSBFIRST, so the least significant bit is shifted out first. When MSBFIRST is chosen, the MSB is pin QH/Q8 and the LSB is pin QA/Q8. Using LSBFIRST switches the order around. The value needs to below or equal to 255 because the function can only shift out data byte-wise. Here the element with the index given in the powerSwitch() argument from the array sensor_number initialized in (2) is used as value. | ||
+ | - Giving a high signal on the latch pin transmits the fully shifted out byte to the storage register which adjusts the three-state outputs accordingly causing the LEDs/ | ||
+ | |||
+ | ===== 4.3 Combining the Sketches ===== | ||
+ | |||
+ | This code can be combined with the code from the DHT-22 and the DS18B20s such that the sensors are only powered when they are supposed to measure and are switched off again afterwards. | ||
+ | |||
+ | The sensors and the shift register need to be connected like explained in their pages, respectively. Additionally, | ||
+ | |||
+ | ^{{: | ||
+ | |**//Figure 3//** ESP32 connected with DHT-22, DS18B20 and SN74HC595.| | ||
+ | |||
+ | ==== 4.3.1 The Code combined ==== | ||
+ | |||
+ | The individual sections of the code are explained in detail in the pages of the [[amc2020: | ||
+ | |||
+ | <file c++ ESP32_Sensors_Combined.ino> | ||
+ | //ESP32 + DHT-22 + DS18B20 + 74HC595(n) Test | ||
+ | |||
+ | //The code in this sketch is explained in detail in the DHT-22, the DS18B20 and the SN74HC595N pages | ||
+ | //and just combines the different codes. | ||
+ | |||
+ | / | ||
+ | * DHT-22 | ||
+ | | ||
+ | */ | ||
+ | // | ||
+ | #define DHTTYPE DHT22 // | ||
+ | const int DHTPIN = 15; //define the Data Pin (GPIO 2) | ||
+ | |||
+ | //Libraries & Objects | ||
+ | #include < | ||
+ | |||
+ | DHT dht(DHTPIN, DHTTYPE); | ||
+ | |||
+ | //Variables | ||
+ | float dht22AirTem = 0; //takes up new readings + final average | ||
+ | float dht22AirTemSum = 0; // | ||
+ | String dht22Temperature = ""; | ||
+ | |||
+ | float dht22RelHum = 0; //takes up new readings + final average | ||
+ | float dht22RelHumSum = 0; // | ||
+ | String dht22Humidity = ""; | ||
+ | |||
+ | const uint8_t AveragingNumberDHT22 = 5; //Number of measurements to be averaged | ||
+ | |||
+ | / | ||
+ | * DS18B20 | ||
+ | | ||
+ | */ | ||
+ | // | ||
+ | const int ONE_WIRE_BUS = 14; //=GPIO 14 of the ESP32 as 1-Wire Bus | ||
+ | |||
+ | //Libraries and Objects | ||
+ | #include < | ||
+ | #include < | ||
+ | |||
+ | DeviceAddress bottomSensorAddress = {0x28, | ||
+ | DeviceAddress surfaceSensorAddress = {0x28, | ||
+ | OneWire oneWire(ONE_WIRE_BUS); | ||
+ | DallasTemperature DS18B20(& | ||
+ | |||
+ | //Variables | ||
+ | float bottomTem = 0.00; //for Temperature readings | ||
+ | String bottomTemperature = ""; | ||
+ | |||
+ | float surfaceTem = 0.00; | ||
+ | String surfaceTemperature = ""; | ||
+ | |||
+ | const uint8_t AveragingNumberDS18B20 = 5; //Number of measurements to be averaged | ||
+ | |||
+ | int RESOLUTION = 12; // | ||
+ | int TCONV = 750; // | ||
+ | int delayTime = (TCONV/ | ||
+ | |||
+ | / | ||
+ | * Shift Register | ||
+ | | ||
+ | */ | ||
+ | //Shift Register Pins | ||
+ | const int LATCHPIN = 2; | ||
+ | const int CLOCKPIN = 4; | ||
+ | const int DATAPIN = 0; | ||
+ | |||
+ | //Shift Register Pin Array | ||
+ | byte sensor_numbers[9] = { | ||
+ | B00000000, | ||
+ | B00000001, | ||
+ | B00000010, | ||
+ | B00000100, | ||
+ | B00001000, | ||
+ | B00010000, | ||
+ | B00100000, | ||
+ | B01000000, | ||
+ | B10000000, | ||
+ | }; | ||
+ | |||
+ | / | ||
+ | * Setup | ||
+ | | ||
+ | */ | ||
+ | |||
+ | void setup() { | ||
+ | pinMode(LATCHPIN ,OUTPUT); | ||
+ | pinMode(CLOCKPIN ,OUTPUT); | ||
+ | pinMode(DATAPIN ,OUTPUT); | ||
+ | | ||
+ | Serial.begin(115200); | ||
+ | delay(5000); | ||
+ | Serial.println(" | ||
+ | Serial.println(" | ||
+ | Serial.println(delayTime); | ||
+ | } | ||
+ | |||
+ | / | ||
+ | * Loop | ||
+ | | ||
+ | */ | ||
+ | |||
+ | void loop() { | ||
+ | Serial.println(" | ||
+ | powerSwitch(2); | ||
+ | measureDHTTemHum(AveragingNumberDHT22); | ||
+ | powerSwitch(1); | ||
+ | delay(200); | ||
+ | powerSwitch(3); | ||
+ | measureDS18B20Tem(AveragingNumberDS18B20); | ||
+ | powerSwitch(1); | ||
+ | delay(200); | ||
+ | Serial.println(" | ||
+ | Serial.println(" | ||
+ | Serial.println(" | ||
+ | Serial.println(" | ||
+ | Serial.println(" | ||
+ | Serial.println(" | ||
+ | Serial.println(" | ||
+ | } | ||
+ | |||
+ | / | ||
+ | * Sensor and Shift Register Functions | ||
+ | | ||
+ | */ | ||
+ | |||
+ | //DS18B20 Measurement function | ||
+ | void measureDS18B20Tem (const uint8_t AveragingNumber){ | ||
+ | DS18B20.begin(); | ||
+ | bottomTem = 0; | ||
+ | surfaceTem = 0; | ||
+ | bottomTemperature = ""; | ||
+ | surfaceTemperature = ""; | ||
+ | |||
+ | for(byte i = 0; i < AveragingNumber; | ||
+ | { | ||
+ | DS18B20.requestTemperatures(); | ||
+ | delay(delayTime); | ||
+ | bottomTem += DS18B20.getTempC(bottomSensorAddress); | ||
+ | surfaceTem += DS18B20.getTempC(surfaceSensorAddress); | ||
+ | } | ||
+ | bottomTem /= AveragingNumber; | ||
+ | surfaceTem /= AveragingNumber; | ||
+ | |||
+ | if(bottomTem< | ||
+ | bottomTemperature = " | ||
+ | bottomTemperature += bottomTem; | ||
+ | |||
+ | if(surfaceTem< | ||
+ | surfaceTemperature = " | ||
+ | surfaceTemperature += surfaceTem; | ||
+ | } | ||
+ | |||
+ | //DHT22 Measurement function | ||
+ | void measureDHTTemHum (const uint8_t AveragingNumber) | ||
+ | { | ||
+ | dht.begin(); | ||
+ | delay(1000); | ||
+ | dht22AirTemSum = 0; | ||
+ | dht22RelHumSum = 0; | ||
+ | dht22Temperature = ""; | ||
+ | dht22Humidity = ""; | ||
+ | for (byte i = 0; i < AveragingNumber; | ||
+ | { | ||
+ | do { //execute this at least once | ||
+ | dht22AirTem = dht.readTemperature(); | ||
+ | dht22RelHum = dht.readHumidity(); | ||
+ | if (!isnan(dht22AirTem) && !isnan(dht22RelHum)) | ||
+ | { | ||
+ | dht22AirTemSum += dht22AirTem; | ||
+ | dht22RelHumSum += dht22RelHum; | ||
+ | } | ||
+ | else | ||
+ | delay(2000); | ||
+ | } while (isnan(dht22AirTem) || isnan(dht22RelHum)); | ||
+ | if (i < (AveragingNumber - 1)) //after the last measurement, | ||
+ | delay(2000); | ||
+ | } | ||
+ | dht22AirTem = dht22AirTemSum / AveragingNumber; | ||
+ | if(dht22AirTem< | ||
+ | dht22Temperature = ' | ||
+ | dht22Temperature = dht22Temperature + dht22AirTem; | ||
+ | | ||
+ | dht22RelHum = dht22RelHumSum / AveragingNumber; | ||
+ | if(dht22RelHum< | ||
+ | dht22Humidity = ' | ||
+ | dht22Humidity = dht22Humidity + dht22RelHum; | ||
+ | } | ||
+ | |||
+ | //Shift Register Power Switch | ||
+ | void powerSwitch(byte pin){ | ||
+ | digitalWrite(LATCHPIN, | ||
+ | shiftOut(DATAPIN, | ||
+ | digitalWrite(LATCHPIN, | ||
+ | } | ||
+ | </ | ||
+ | |||
+ | ==== 4.3.2 Results ==== | ||
+ | |||
+ | The results of the sensor measurements are printed to the serial monitor: | ||
+ | |||
+ | < | ||
+ | 18: | ||
+ | 18: | ||
+ | 18: | ||
+ | 18: | ||
+ | 18: | ||
+ | 18: | ||
+ | 18: | ||
+ | 18: | ||
+ | 18: | ||
+ | 18: | ||
+ | 18: | ||
+ | 18: | ||
+ | 18: | ||
+ | 18: | ||
+ | 18: | ||
+ | 18: | ||
+ | 18: | ||
+ | 18: | ||
+ | |||
+ | </ | ||
+ | |||
+ | During this test, all sensors were in equilibrium, | ||
+ | |||
+ | < | ||
+ | < | ||
+ | < | ||
+ | <a href=" | ||
+ | <span style=" | ||
+ | Back to the top ⤴ | ||
+ | </ | ||
+ | </a> | ||
+ | </ | ||
+ | </ | ||
+ | </ |
amc2020/group_n/sn74hc595n.1595091907.txt.gz · Last modified: 2023/01/05 14:38 (external edit)