Skip to content

Latest commit

 

History

History
632 lines (486 loc) · 16.9 KB

README.md

File metadata and controls

632 lines (486 loc) · 16.9 KB

RPIPLC-LIB library

by Industrial Shields

Rpiplc-lib implements some common applications on industrial environments for Raspberry based Industrial Shields PLC, based on the code structure as the Arduino IDE.

Getting started

Prerequisites

Raspberry based Industrial Shields PLC.

Installing Git

1. Start by updating the package manager:

sudo apt update

2. Run the following command to install Git:

sudo apt install git

3. Verify the installation by typing the following command which will print the Git version.

git --version

Installation

1. Go to the directory where you want to have the library.

2. Run the following command to clone the repository:

git clone -b <tagname> https://github.com/Industrial-Shields/rpiplc-lib.git

tagname = v1.X.X = RPI PLC Version 3

tagname = v2.X.X = RPI PLC Version 4

You can find the available tags here: https://github.com/Industrial-Shields/rpiplc-lib/tags

3. Access to the directory and execute the makefile:

cd rpiplc-lib

make

sudo make install

4. Now, you can open any file from the rpiplc-lib/ directory.

5. Finally, go to the /boot/config.txt file:

sudo nano /boot/config.txt

6. Comment all the following lines by adding a hash mark at the beginning of the lines like this:

#dtoverlay=mcp23017,noints,mcp23008,addr=0x20
#dtoverlay=mcp23017,noints,mcp23008,addr=0x21

#dtoverlay=i2c-pwm-pca9685a,addr=0x40
#dtoverlay=i2c-pwm-pca9685a,addr=0x41

#dtoverlay=ads1015,addr=0x48
#dtparam=cha_enable=true,cha_gain=1
#dtparam=chb_enable=true,chb_gain=1
#dtparam=chc_enable=true,chc_gain=1
#dtparam=chd_enable=true,chd_gain=
#dtoverlay=ads1015,addr=0x49
#dtparam=cha_enable=true,cha_gain=1
#dtparam=chb_enable=true,chb_gain=1
#dtparam=chc_enable=true,chc_gain=1
#dtparam=chd_enable=true,chd_gain=1
#dtoverlay=ads1015,addr=0x4a
#dtparam=cha_enable=true,cha_gain=1
#dtparam=chb_enable=true,chb_gain=1
#dtparam=chc_enable=true,chc_gain=1
#dtparam=chd_enable=true,chd_gain=1
#dtoverlay=ads1015,addr=0x4b
#dtparam=cha_enable=true,cha_gain=1
#dtparam=chb_enable=true,chb_gain=1
#dtparam=chc_enable=true,chc_gain=1
#dtparam=chd_enable=true,chd_gain=1

7 Reboot the system to apply changes:

sudo reboot now

Compilation

g++ is a GNU project C and C++ compiler. When you invoke GCC, it normally does preprocessing, compilation, assembly and linking. The gcc program accepts options and file names as operands. So, inside the rpiplc-lib directory, go to the test directory and compile the rpiplc-lib applications with the essential parameters:

g++ -o file file.cpp -L /usr/local/lib -l rpiplc -I /usr/local/include/rpiplc -D DEVICE_MODEL

-o file: Place output in file file.

-l library. Search the library named library when linking.

-I dir. Add the directory dir to the list of directories to be searched for header files. Directories named by -I are searched before the standard system include directories.

-D name=definition. The contents of definition are tokenized and processed as if they appeared during translation phase three in a #define directive.

Know more: https://linux.die.net/man/1/g++

Reference

Rpiplc-lib contains different applications:

  1. Main
  2. AnalogBlink
  3. AnalogBlinkAll
  4. AnalogRead
  5. Delay
  6. DigitalBlink
  7. DigitalBlinkAll
  8. DigitalRead
  9. Set digital output
  10. Set analog output
  11. Get digital input
  12. Get analog input
  13. Available PLC models

Main

All the applications must implement the main function that looks like the example below:

int main(int argc, char* argv[]) {
    initPins();

    setup();

    while (1) {
        loop();
    }
    
    return 0;
}

The main() function will call the setup() and loop() function so that they can be executed.

AnalogBlink

#include <rpiplc.h>
#include "common.h"

This application shows the simplest thing you can do to see physical outputs: it blinks the on-board LEDs from the analog outputs.

The pinMode function, like in Arduino, configures the specified pin to behave either as an input or an output. In the setup function, all the analog outputs are set as outputs:

void setup() {
  for (int i = 0; i < numAnalogOutputs; ++i) {
    pinMode(analogOutputs[i], OUTPUT);
  }
}

In the loop function, all the analog outputs are written with the different analog values, with a 1000 milliseconds delay in every loop.

void loop() {
  for (int i = 0; i < numValues; ++i) {
    printf("Set value %d\n", values[i]);

    for (int j = 0; j < numAnalogOutputs; ++j) {
      analogWrite(analogOutputs[j], values[i]);
    }

    delay(1000);
  }
}

So, in the ~/rpiplc-lib/test directory, execute the following command to create an executable file called analogBlink:

g++ -o analogBlink analogBlink.cpp -l rpiplc -I  /usr/local/include/rpiplc -D RPIPLC_58 (or any other Raspberry PLC model)

Execute the created file named analogBlink, to run the application:

./analogBlink

and start getting an output every 1000 milliseconds:

Set value 0
Set value 511
Set value 1023
Set value 2047
Set value 4095
Set value 2047
Set value 1023
Set value 511
Set value 0

AnalogBlinkAll

#include <rpiplc.h>
#include "common.h"

This application, like analog blink, shows how to blink the on-board LEDs from all the analog outputs. All the values are written at the same time by minimizing the i2c writes, in order to reduce the speed of the analog outputs.

The pinMode function, like in Arduino, configures the specified pin to behave either as an input or an output. In the setup function, all the analog outputs are set as outputs:

void setup() {
  for (int i = 0; i < numAnalogOutputs; ++i) {
    pinMode(analogOutputs[i], OUTPUT);
  }
}

In the loop function, all the analog outputs are written with the different analog values, with a 1000 milliseconds delay in every loop.

void loop() {
    for (int i = 0; i < numValues; ++i) {
        printf("Set value %d\n", values[i]);

        for (int j = 0; j < PCA9685_NUM_OUTPUTS; ++j) {
            analogValues[j] = values[i];
        }
        for (int j = 0; j < rpiplc_num_pca9685; ++j) {
            analogWriteAll(rpiplc_pca9685[j], analogValues);
        }
        delay(1000);
        }
}

So, in the ~/rpiplc-lib/test directory, execute the following command to create an executable file called analogBlinkAll:

g++ -o analogBlinkAll analogBlinkAll.cpp -l rpiplc -I  /usr/local/include/rpiplc -D RPIPLC_58 (or any other Raspberry PLC model)

Execute the created file named analogBlinkAll, to run the application:

./analogBlinkAll

and start getting an output every 1000 milliseconds:

Set value 0
Set value 511
Set value 1023
Set value 2047
Set value 4095
Set value 2047
Set value 1023
Set value 511
Set value 0
...

AnalogRead

The analogRead application reads the value from the analog input pins. In this setup function, all the inputs are set as analog inputs:

void setup() {
  for (int i = 0; i < numAnalogInputs; ++i) {
    pinMode(analogInputs[i], INPUT);
  }
}

In the loop function below, it prints out all the analog input values, and there is a 1000 milliseconds delay after that.

void loop() {
  for (int i = 0; i < numAnalogInputs; ++i) {
    uint16_t value = analogRead(analogInputs[i]);
    printf("Pin %08x value: %u\n", analogInputs[i], value);
  }
  delay(1000);
}

Compile the application executing the following command:

g++ -o analogRead analogRead.cpp -l rpiplc -I /usr/local/include/rpiplc -DRPIPLC_58 (or any other Raspberry PLC model)

Run it executing the following:

./analogRead

The output on the terminal will be shown like this:

Pin 00004a00 value: 1
Pin 00004a01 value: 1
Pin 00004b00 value: 1
Pin 00004802 value: 1
Pin 00004800 value: 2
Pin 00004801 value: 2
Pin 00004900 value: 1
Pin 00004a03 value: 1
Pin 00004b02 value: 2
Pin 00004b03 value: 2
Pin 00004a02 value: 2
Pin 00004901 value: 1
Pin 00004903 value: 2
Pin 00004902 value: 1
Pin 00004803 value: 2
Pin 00004b01 value: 1

Delay

The delay application pauses the program for the amount of time (in milliseconds) specified as parameter.

Syntax:

delay(ms) - ms: the number of milliseconds to pause.

int counter = 0;
void setup() {
}

void loop() {
  printf("%d\n", counter++);
  delay(1000);
}

Compile the application executing the following command:

g++ -o delay delay.cpp -l rpiplc -I /usr/local/include/rpiplc -DRPIPLC_58 (or any other Raspberry PLC model)

Run it executing the following:

./delay

The example file in the rpiplc-lib/test directory, delays the program and it prints a counter every 1000 milliseconds. The output is shown like this:

0
1
2
3
4
5

DigitalBlink

This application shows the on-board LEDs blinking from the digital outputs. The values change every 1000 milliseconds.

The setup function configures all the possible outputs as digital outputs.

void setup() {
  printf("Num digital outputs: %d\n", numDigitalOutputs);
  for (int i = 0; i < numDigitalOutputs; ++i) {
    pinMode(digitalOutputs[i], OUTPUT);
  }
}

The loop function makes all the digital output LEDs blink every 1000 milliseconds.

void loop() {
  value = value == 0 ? 1 : 0;

  printf("Set value %d\n", value);
  for (int i = 0; i < numDigitalOutputs; ++i) {
    digitalWrite(digitalOutputs[i], value);
  }

  delay(1000);
}

Compile the application executing the following command:

g++ -o digitalBlink digitalBlink.cpp -l rpiplc -I /usr/local/include/rpiplc -DRPIPLC_58 (or any other Raspberry PLC model)

Run it executing the following:

./digitalBlink

The output on the terminal will be shown like this:

Num digital outputs: 24

Set value 1
Set value 0
Set value 1
Set value 0
Set value 1

DigitalBlinkAll

This application shows the on-board LEDs blinking from all the digital outputs, by reducing the i2c writes in order to decrease the speed of the outputs. The values change every 1000 milliseconds.

The setup function configures all the possible outputs as digital outputs.

void setup() {
  printf("Num digital outputs: %d\n", numDigitalOutputs);
  for (int i = 0; i < numDigitalOutputs; ++i) {
    pinMode(digitalOutputs[i], OUTPUT);
  }
}

The loop function makes all the digital output LEDs blink every 1000 milliseconds.

void loop() {
    value = value == 0 ? 0xffffffff : 0;

    for (int i = 0; i < rpiplc_num_mcp23008; ++i) {
        digitalWriteAll(rpiplc_mcp23008[i], value);
    }
    for (int i = 0; i < rpiplc_num_pca9685; ++i) {
        digitalWriteAll(rpiplc_pca9685[i], value);
    }
    delay(1000);
}

Compile the application executing the following command:

g++ -o digitalBlinkAll digitalBlinkAll.cpp -l rpiplc -I /usr/local/include/rpiplc -DRPIPLC_58 (or any other Raspberry PLC model)

Run it executing the following:

./digitalBlinkAll

The output on the terminal will be shown like this:

Num digital outputs: 24

Set value 1
Set value 0
Set value 1
Set value 0
Set value 1

DigitalRead

The digitalRead application reads the value from the digital input pins. In this setup function, all the inputs are set as digital inputs:

void setup() {
  printf("Num digital inputs: %d\n", numDigitalInputs);
  for (int i = 0; i < numDigitalInputs; ++i) {
    pinMode(digitalInputs[i], INPUT);
  }
}

In the loop function below, it prints out all the digital input values, and there is a 1000 milliseconds delay after that.

void loop() {
  for (int i = 0; i < numDigitalInputs; ++i) {
    int value = digitalRead(digitalInputs[i]);
    printf("Pin %08x value: %u\n", digitalInputs[$
  }
  delay(1000);
}

To run the application, execute the following command:

g++ -o digitalRead digitalRead.cpp -lrpiplc -I/usr/local/include/rpiplc -DRPIPLC_38AR (or any other Raspberry PLC model)

Run it executing the following:

./digitalRead

The output on the terminal will be shown like this:

Pin 00002105 value: 0
Pin 00002103 value: 0
Pin 00002102 value: 0
Pin 00002101 value: 0
Pin 00002100 value: 0
Pin 0000000d value: 0
Pin 0000000c value: 0
Pin 00002002 value: 0
Pin 00002001 value: 0
Pin 00002000 value: 0
Pin 00002107 value: 0
Pin 00002106 value: 0
Pin 0000001b value: 0
Pin 00000004 value: 0
Pin 00002006 value: 0
Pin 00002005 value: 0
Pin 00002007 value: 0
Pin 00002004 value: 0
Pin 00002003 value: 0
Pin 00000011 value: 0
Pin 00000010 value: 0

Set digital output

This application sets a digital output to the specified value.

The main function initializes the microcontrollers with the initPins() function. Then, like in Arduino programming, it sets the output to the output mode, and it writes to the pin the specified value in the parameter: either 1 or 0

So, in the ~/rpiplc-lib/test directory, execute the following command to create an executable file called set-digital-output:

g++ -o set-digital-output set-digital-output.cpp -l rpiplc -I /usr/local/include/rpiplc -DRPIPLC_42 (or any other Raspberry PLC model)

Execute the compiled file named set-digital-output with two parameters: 1: the output to control 2: the value to set

./set-digital-output Q0.0 1

And see how the Q0.0 output has been activated.

Set analog output

This application sets an analog output to the specified value.

The main function initializes the microcontrollers with the initPins() function. Then, like in Arduino programming, it sets the output to the output mode, and it writes to the pin the specified value in the parameter.

So, in the ~/rpiplc-lib/test directory, execute the following command to create an executable file called set-analog-output:

g++ -o set-analog-output set-analog-output.cpp -l rpiplc -I /usr/local/include/rpiplc -DRPIPLC_19R (or any other Raspberry PLC model)

Execute the compiled file named set-analog-output with two parameters: 1: the output to control 2: the value to set

./set-analog-output A0.5 4095

And see how the A0.5 output has been given the specified resolution.

PWM

This application sets a PWM output to the specified value using the set-analog-output function with a digital output. All the digital outputs of the PLC can be used for the PWM function.

The main function initializes the microcontrollers with the initPins() function. Then, like in Arduino programming, it sets the output to the output mode, and it writes to the pin the specified value in the parameter.

So, in the ~/rpiplc-lib/test directory, execute the following command to create an executable file called set-analog-output:

g++ -o set-analog-output set-analog-output.cpp -l rpiplc -I /usr/local/include/rpiplc -DRPIPLC_19R (or any other Raspberry PLC model)

Execute the compiled file named set-analog-output with two parameters: 1: the digital output to control 2: the PWM (analog) value to set (from 0 to 4095)

./set-analog-output Q0.0 4095

And see how the Q0.0 output has been given the specified resolution.

Get digital input

This application gets the value for a digital input.

The main function initializes the microcontrollers with the initPins() function. Then, like in Arduino programming, it sets the input to the input mode, and it reads the value of the pin specified in the parameter

So, in the ~/rpiplc-lib/test directory, execute the following command to create an executable file called get-digital-input:

g++ -o get-digital-input get-digital-input.cpp -l rpiplc -I /usr/local/include/rpiplc -DRPIPLC_21 (or any other Raspberry PLC model)

Execute the compiled file named get-digital-input with the input as parameter.

./get-digital-input I0.0

And get the value for that input.

1

Get analog input

This application gets the value for a analog input.

The main function initializes the microcontrollers with the initPins() function. Then, like in Arduino programming, it sets the input to the input mode, and it reads the value of the pin specified in the parameter

So, in the ~/rpiplc-lib/test directory, execute the following command to create an executable file called get-analog-input:

g++ -o get-analog-input get-digital-input.cpp -l rpiplc -I /usr/local/include/rpiplc -DRPIPLC_21 (or any other Raspberry PLC model)

Execute the compiled file named get-analog-input with the input as parameter.

./get-analog-input I0.7

And get the value for that input.

2000

Available PLC models

RPIPLC (for Raspberry PLC CPU)
RPIPLC_19R
RPIPLC_21
RPIPLC_38AR
RPIPLC_38R
RPIPLC_42
RPIPLC_50RRA
RPIPLC_53ARR
RPIPLC_54ARA
RPIPLC_57AAR
RPIPLC_57R
RPIPLC_58