Skip to content

Latest commit

 

History

History
345 lines (207 loc) · 13.2 KB

README.md

File metadata and controls

345 lines (207 loc) · 13.2 KB

PLCnext Technology - OpcUaMethods

License Web Community

This procedure describes the basic steps for creating an OPC UA method.

In this example, the OPC UA method will be used to call a PLCnext Engineer function block instance from an OPC UA client.

The procedure uses a custom OPC UA "Information Model", which in this case is generated using the UaModeler tool from Unified Automation. This technique can also be applied when using standard OPC UA information models for various industries and applications.

Project details

Description Value
Created 24.02.2021
Last modified 24.02.2021
Controller AXC F 1152; AXC F 2152; AXC F 3152
Firmware 2021.0 LTS

Background reading

Prerequisites for this example

  • PLCnext Control device with firmware 2021.0.0 or later, and at least one Axioline I/O module.
  • PLCnext Engineer version 2021.0.1 or later.
  • UaModeler from Unified Automation.
  • UaExpert from Unified Automation.

Example description

In this example, we will create an OPC UA method that retrieves a single item of Parameter, Diagnostics and Information (PDI) data from an Axioline I/O module connected to the PLC.

The PDI data available for each type of Axioline I/O module is described in the data sheet for that I/O module. Data sheets can be downloaded from the Phoenix Contact website.

The "address" of each item of PDI data can be uniquely identified using four integers:

  • Slot - the location of the I/O module in the Axioline bus.
  • Subslot - the location of the sub-module (if any) in the slot.
  • Index - the location of the PDI data item in the I/O module.
  • Subindex (only applies when the Index refers to a PDI "record") - the location of a single data item in a PDI record.

Each item of PDI data has a specific data type (e.g. string or integer), however for simplicity our method only returns a string.

This example has limited error checking, e.g. on input parameter values. Thorough error checking would be required for production applications, but it is omitted here for simplicity.

Procedure

Create the information model

  1. In UaModeler, create a new project.

    New Project 1

    New Project 2

    New Project 3

    New Project 4

  2. In the Information Model pane, create a new Object Type by right-clicking on the BaseObjectType.

    New Type

  3. Fill in the details of the new type.

    • Give the type a name.
    • Add the method that we will use to read PDI data, with the required input and output arguments. The names and data types of the method arguments should be as shown below.

    Type Details

    • When finished, press the OK button.
  4. Save the model.

Create the PLCnext Engineer function block

We now need to create a PLCnext Engineer function block that corresponds to the method definition in the information model.

  1. In PLCnext Engineer, create a new project based on the template for your PLC and firmware version.

  2. Change the OPC UA server settings so that "Visibility of variables" is set to either "Marked" or "All".

  3. Add your Axioline I/O module(s) to the "Axioline F" node in the Project tree.

  4. In the Axioline F "Device List" tab, note the Slot number (#) that has been assigned to each Axioline I/O module.

  5. In a Data Types worksheet, create two new data types.

    TYPE
       ByteArray100 : ARRAY [1..100] OF BYTE;
    END_TYPE
    
    TYPE
       WordArray2 : ARRAY [1..2] OF WORD;
    END_TYPE
  6. In the Components pane, create a new function block.

    Add FB

    In this example, the name of the function block is UA_PDI_READ.

  7. Add the required OPC UA variables to the function block.

    FB vars 1

    Please note the following:

    • There must be one variable for each argument that we defined in the information model method.
    • The name, type and usage of these arguments must match the arguments in the information model.
    • There must be two additional variables in our function block: UA_MethodState and UA_StatusCode. These are defined by the OPC UA standard, and are required for OPC UA method calls to work properly.
    • All these variables must me marked with the OPC attribute.
  8. Implement the function block behaviour.

    You can now implement the function block behaviour however you like.

    In this case, our function block:

    • is written using structured text,
    • uses a state machine to retrieve and process the PDI data item,
    • utilises instances of two standard function blocks - PDI_READ and BUF_TO_STRING,
    • assumes that the PDI data item is always of type string,
    • assumes that all FB calls execute successfully.

    The additional local variables used in the FB, and the code that implements the state machine, are shown below.

    FB vars 2

    // NOTE: Error handling is not implemented here,
    //       but should be for a production application.
    
    // State machine
    CASE UA_MethodState OF
    
        0 : // Waiting to be called by eUA Server
            // Writing to UA_MethodState is not allowed in this state
            
        1 : // Start (this state is set by the eUA Server)
            UA_MethodState := 10;
        
        10 : // Request PDI data
            UA_MethodState := 20;
    
        20 : // Wait for PDI response
            IF PDI_READER.NDR THEN
                UA_MethodState := 30;
            END_IF
    
        30 : // Convert the response to a String
            UA_MethodState := 40;
    
        40 : // Wait for string conversion
            IF BUF_TO_STRING1.DONE THEN
                Result := PdiString;
                UA_StatusCode := UA_StatusCodeEnum#Good;
                UA_MethodState := 0; // Finished
            END_IF
    
    END_CASE
    
    // Call FB instances
    PDI_READER(
        REQ := (UA_MethodState >= 10),
        SLOT := Slot,
        SUBSLOT := Subslot,
        INDEX := Index,
        SUBINDEX := Subindex,
        RD_1 := PdiData,
        STATUS := PdiStatus
    );
    
    BUF_TO_STRING1(
        REQ := (UA_MethodState >= 30),
        BUF_CNT := TO_UINT(PDI_READER.DATA_CNT),
        BUFFER := PdiData,
        DST := PdiString
    );

Create an instance of the function block

  1. In PLCnext Engineer, create an instance of the function block in the Main program.

    Do not connect any variables to the inputs or outputs.

    In this example, the name of the FB instance is UA_PDI_READ1.

    FB Instance

  2. Make sure that an instance of the Main program has been created in a cyclic task.

    In this example, the name of the program instance is MainInstance.

    Program Instance

  3. Write and start the PLCnext Engineer project on the PLC.

Create an object in the information model, and link it to the function block instance

So far, the OPC UA information model only contains a type definition, which corresponds to the function block definition in PLCnext Engineer. In the same way that we created an instance of our function block in PLCnext Engineer, we will now define an object (i.e. an instance of our type) in the OPC UA information model. We will then link this object to the function block instance in our PLCnext Engineer project.

  1. In the UaModeler Information Model pane, add a new instance to the Objects folder.

    Add Instance

  2. Fill in the details of the new object.

    • Give the object a name.
    • Specify the type of the new object - this must be the type that we created earlier.

    New Object 1

    • When finished, press the OK button.
  3. Select the method in the new object.

    Instance Method

  4. With the method in the new object selected, view the user-defined extensions for that method.

    Extensions Button

    This button may need to be enabled in the UaModeler Settings window:

    UA Settings

  5. Add an extension that links the method to the PLCnext Engineer function block instance.

    Extension

    • The format of the extension is:
      <MethodTarget xmlns="http://phoenixcontact.com/OpcUA/2019/NodeSetExtensions.xsd" FunctionBlock="COMPONENT/PROGRAM.FUNCTION_BLOCK"/>
    • The name of the function block instance must be fully qualified with the component name and the program instance name.
    • When finished, press the OK button.
  6. Save the model.

Load the information model into the OPC UA server

  1. In UaModeler, export the OPC UA information model as an XML file in NodeSet format by right-clicking on the model.

    Export XML

  2. Copy the exported XML file to the following directory on the PLC:

    /opt/plcnext/projects/Default/Services/OpcUA/NodeSets/
    
  3. Restart the OPC UA server on the PLC to load the new model.

    • Restart the PLCnext Runtime, or
    • Cycle power to the PLC.

    Note: Restarting the PLC from PLCnext Engineer does not restart the OPC UA server.

Execute the method from an OPC UA client

  1. In UaExpert, connect to the PLC in the usual way.

  2. Call the PdiRead function on your object.

    Method Call 1

    The list of valid PDI index and Subindex values is given in the data sheet for each Axioline I/O module.

    PDI Indexes

    When the "Call" button is pressed, the state machine in the function block instance is activated, and the value of the PDI data item is returned.

    Method Call 2

Bonus exercise

In some cases you may want to allow users to select an input argument value from a fixed list. For example, in this case we can force the user to select the Index value from a list of known valid options.

Procedure:

  1. In UaModeler, create a new Enumerated DataType.

    Add Enum

  2. Fill in the details of the new type.

    • Give the type a name.
    • Add the values that the user can select, including a string that indicates what each value means.

    Enum Details

    • When finished, press the OK button.
  3. Change the type of the Index argument to the Enumerated data type.

    Enum Argument

    • When finished, press the OK button.
  4. Save the model.

  5. Export the XML file.

  6. Copy the XML file to the PLC.

  7. In PLCnext Engineer, change the type of the Index input.

    Enumerated data types are always of type Int32, so the type of the Index input on the function block must be changed to DINT.

    Enum FB Variable

  8. In the call to the PDI_READER function block instance, convert the Index value to type WORD, using the TO_WORD function.

  9. Write and start the PLCnext Engineer project.

  10. In UaExpert, call the method.

    Enum Success

    Now, the Index value must be selected from a fixed list of options.

Next steps

Rather than building your own information model from scratch, it is worth considering a standard information model for your specific industry or application type. The PLCnext Info Center gives an example of how to use the standard PA-DIM information model in UaModeler with a PLCnext Engineer project.

Problems?

If you find a mistake in this procedure, or if you would like to suggest improvements or new features, please open an issue.

License

Copyright (c) Phoenix Contact GmbH & Co KG. All rights reserved.

Licensed under the MIT License.