I come from a world (line-of-business applications written in higher programming languages (e.g. C#)) where this knowledge comes natural to most of the people I worked with. I've recently dipped my toes into PLC programming and I was amazed how some of the software best-practices seem to only now being slowly picked up in this PLC world. Thanks for spreading this knowledge!
Nice video Jakob, quite interesting topic and the concept you've mentioned is clear for the most. I started programming PLC's from the late '80s. At those days the situation was totally different. The PLC's programming tools rely on IL/AWL , Ladder and few on FBD (Siemens). The CPU's elaboration power was a lot less and the use of FBs and FCs slowed down the process capabilities, needless to say the low memory available. Moreover some industries like automotive industries made the use of the Ladder the only one accepted (Allen Bradley for the most in the US and Siemens in Europe). In the last decade the improvement of the CPUs performances and the IT technoligies introduced for the Industry 4.0 paradigm changed everything. The Industrial OOP concept in now a must to overcome the complexity but the pace is not so fast and the old habits still last. ;-)
I was introduced to the world of PLC programming and the programs that I was given were such a mess, with global variables, no structures, no case statements, etc... I always dread the days where I need to connect to a customer for support and I have to debug hundreds of lines of code in Main 😂. Beckhoff has some decent introduction to the world of OOP (which does steer you away from globals, in my opinion), but you put the nail in the coffin as far as programming practice goes, for me at least. I have a project coming where I will definitely start using TDD as well, so thanks for that! For me, the only global variable I use is system's time, which I do not want to pass explicitly to every block that might need it and is only assigned by 1 source, the time provider instance and a few others that really are only written from one source and are only a read for the rest of the program (EtherCAT error state, first cycle, control system's (HMI) connection state).
Very educational video. I did quit automation because the lack of these practices, I hope it gets better in the future and you are helping a lot bringing this information to the automation field. I feel you should start explaining as well what is a scope before talk about global variables. In automation there are a lot of people that don't even know that, I read in LinkedIn a guy who was confusing global variables with structs.
Hey Alberto! Thanks for your feedback. You're right, so to clarify; The scope of a variable defines where it’s accessible in a program based off of where it was created. Global variables scope is the complete program, while local variables scope can only be accessed in where they are defined (function, function block or method).
Well said! When I first started with PLC programming, I was surprised by how often global variables are used. In any other language it is discouraged. Just look at any Stackoverflow question/answer where someone tries to solve it with global variables :D. I hope you're also making a follow up video to show people how they can prevent using globals 99% of the time?
And unfortunately this is how most beginners are introduced to declaring variables, as even the official documentation is using globals like there is no tomorrow. I'll do my best to make more videos about this important topic!
Good points. Also you didn't mention the static and thread local variables. Including the ones declared at block scope such as function scope. Those types of variables are bad for unit testing too. A usual alternative to those variables is to just declare them at the call site and pass them by reference to the callee.
In general all variables in some PLCs are in reality still global eg. Siemens TIA, instance-FBs are accessible from everywhere 😞 I CoDeSys (Twincat) I still prefer the IO-map to be global, as it represents "physical the machine-structure" which I consider intrinsically "global". And then map using 'REF=', not at all needed but it just still sits right with me 🙂 I must also admit to using it in CodeSys 2.3 for HMI-tags, where the access symbolic setting was easy, and inherited hiracial, so less chance of error in the SYM-file as interface structures, in a certain area was automatically exported. I fully agree with the generel points in the video, though.
As of the last 13 years of programming in building automation and some industry applications i do agree with you as far as using globals while programming function blocks, functions etc. My main use of globals are inputs/outputs from the io-cards on the plc and controlvalues related to the specific regulator of a temperature probe etc. This is because it’s pretty much needed in order to have those important values saved as persistent data, used in the scada application and hmi application. I do however thus far never used them for individual functionblocks, functions or anywhere else unrelated, if i need to take the input of two values in order to make a change or similar before it sets a global variable for an output in my program i use a local variable in that program. The structure of program that we use means that i have several subprograms with actions for every subpart function for every system in a big building like a hospital and as you say, if i had used global variables for all that it would be a nightmare to troubleshoot and in my case it also could be lethal if i made a change in a wrong spot in that global variable list 😬 I think i understood your point as that i am programming as you are trying to explain in this video 😅
Very informative Jakob! Tracking the source of the change in the behavior of the program due to the use of global variables is pain in the a**. However I believe it’s up to the individuals’ preference how they want to use the variables. Some project are needed to be delivered quicker so just go with them. We avoid them mostly but handshaking between the modules is pretty easy and neat using them. Just pass them as input reference variables to the modules and the chain is easy to track.
As a lot of other people in the comment, I use GVL's for IO's (IN, OUT and IO_LINK). I also have a specific GVL for RETAIN and PERSISTENT. What I am more and more doing, is using the outputs of a PROGRAM. So you have a global variable but in read-only access. Usefull to pass the machine state or a particular recipe as input of a process
Great video, well explained What is also not done in pc coding is the Hungarian notation method It makes code less readable. Why is the plc world so persistent in this?
@JakobSagatowski its been a while since your global variable video. I really appreaciate it since i also struggle with the concept of global variables and in particular how beckhoff themselve use it in every example. We have developed an artifical layer where we model each terminals. Those terminals are instantiated in one PRG and in other programs we have interfaces to those terminals models to access the data. The linking of the IOs we do the via pragmas. Anyhow the list of modelled terminals is still global what i dont like. Starting a new project i am now thinking again about a better concept. I am very curious how you are tackling that thing and looking forward for the promised video ;)
AB 5000 coding. We use a set of Data called Mechanism, which includes the bool, dints, real, timers, etc, all used for that mechanism. We also use sequence step methodology, which to me is close to state logic. Basically, the mechanism, which can be either an actual part of the equipment we are integrating, or an idea/process, is the only thing that can modify the values in the data mechanism. Anything, that when planning, looks like it will be tough to separate, because that part of the equipment works depends so much on what another part is doing, is kept together in a mechanism, which might be included under a higher level mechanism, so it is easier to debug parts without integration until later. My opinion is that global is fine, as long as the control engineering group uses a good practice.
yeah but what about global variables that are read only (except of course in very specific places)? What im talking about is something like user preferences, especially in a user interface. thats both impossible to unit test, and unique to the application anyway. For example, i dont see the problem in storing the prefered color scheme in a (read only) variable, so that buttons and other ui elemts can easily get the color theyre supposed to be
6 місяців тому
Correct me If I am wrong, but the instance of the Fb need to be in global, so you can call the get and set From other parts of the code? Also the persistent and and retain for configuration parameters from the SCADA, I understand they need to be global? And what about status and commands from SCADA? I would love to see how you implement a real life example with all these details... Something simple like a DOL motor, I would love to see a video, showing this paradigm In a real industrial example, if you could make a live would be even better...
I agree, and it's a shame this still has to be mentioned. Sadly, every few months I see a new powerful loud ignorant that continues to do it wrong. Is there a list of studies, Pdf's, articles I can drop on them ?
A question, i do not like using variabels, but i program in many diffrent PLC systems. And for example GX Works from Mitsubishi. There you cant call a program and get the value. For example in twincat you can say POU1.Variabel1 in another program and get reading access to that Variabel. What would you do instead in these situations ?
Hey Jakob, awesome Video. What's your take on passing objects by Reference via VAR_INPUT or as implicit Reference via the good old VAR_IN_OUT ? Are there situations where one would opt for either one in the first place or does it even really matter ? I would argue using VAR_IN_OUT should be the standard way, but one could also argue that the introduction of references makes VAR_IN_OUT a legacy/deprecated feature.
Hey there! I generally prefer VAR_IN_OUT as it's mandatory to supply any parameters, while it for VAR_INPUT is optional (even for references, that's why you have this annoying ISVALIDREF() check that you don't need to use if you use VAR_IN_OUT).
Thanks for another great video. But as someone new to industrial automation, HOW would we go about passing data between routines without global variables. For example, ive written a metrics routine that gathers data like cycle time and fault counts from the other routines and don't know how to do that without writing that state to a global variable that can then be read by the metrics routine.
You dont need to use global vars to pass data if both routines have the same local scope. Lets say, for instance, you have 2 different routine instances that need to talk to each other, for that, write them inside the "main" routine and declare local variables in the "main" routine to interface between the two
AHA! I was right. I'm currently coding a trading algorithm and realized that I was having multiple issues with Global Variables. My project is currently over 14,000 lines of code wherein a large section of code was relying on an approximation of 20+ global variables. I was so stressed out to where I begin looking to see which GV's were causing the issues only arrive at the eureka moment that made me say to myself: "FUK GLOBAL VARIABLES" "THEY'RE BAD"! "I'M NEVER USING THEM AGAIN".
Tell one popular open source project which doesn't have global variables. Saying don't use global variable is like saying there should no knife in this world as it a can cut someones hand. Just like a knife a global variables like anything else can hurt if you use it in the wrong way. But it doesn't justify abolition of global variables.
We build a framework to configure, control and monitor devices in the field. For example, you can start, stop, reverse, reset a motor, and monitor for error, running status, stopped status, and other status flags throughout the entire program. This is done using global variables (of structs) and eliminates the need for passing status from block to block. Because of the way it is build, it is still quite easy to monitor (and control), and we haven't found an "easier", or more "elegant" way to achieve this. Also towards the customer, this make the code very easy readable and takes away much of clutter regarding timers and conversions, since this is taken care of by function blocks, compiled in a library. So in our opinion, global vars sill have their uses. However, for the passing of states and commands between program blocks, we no NOT use global vars and totally agree that there are much better ways to do that.
I declare all the hardware linked variables in a GVL listing (AT%I* & AT%Q*) - there's no particular reason for doing this, just been doing it this way since I was first introduced to this during a training session by Beckhoff Canada. Anyone else does it this way? I'd be curious to know how other programmers do it.
I would say that based on what I've seen in the industry, this is probably the most common way to do it (and one of the reasons why I've created this video in the first place). I'll be making a follow-up video where I will show how you can declare I/O in an alternative way where you don't use global variables.
@@JakobSagatowski I look forward to this video! Variables for IO mapping are one of the few things that I use GVLs for still. Maybe because I too did training at Beckhoff Canada haha
I use a function block and structs to declare my IO mapping and link them in the systems manager to the correaponding hardware IO. I then link my nececcary global variables for the IO on the funtionblock instead. Makes it a LOT easier and also gives the benefit that you can change witch value that comes in on what io without relink anything in the system manager and it just takes an online change to the plc program 🙂👍🏻
@@JakobSagatowski its been a while since your global variable video. I really appreaciate it since i also struggle with the concept of global variables and in particular how beckhoff themselve use it in every example. We have developed an atrifical layer where we model each terminal and then have pointers on the terminals to access teh data and do the linking via pragmas. Anyhow the list of modelled terminals is still global what i dont like. Starting a new project i am now thinking again about a better concept. I am very curious how you are tackling that thing and looking forward for the promised video ;)
Anyone here working with Siemens TIA Portal? I am looking for some source of information about proper metodology how to manage your code/project. So far it has been hard to find on the internet. I have not come across similar person ... like Jakob ... for siemens. Do you have some tips?
On MikroTik routers scripts are the same, each person use Global and Global.... I have no idea why all ppl use only them insted of `local` one. Ahh, people...
It is not only on low ressources where Global Variables make sense. We have big machines running with huge IPCs. However, if we don't structure the program with GVL online change etc. is a pain.
"Stop using global variables!".. In your opinion. Fifty years of programming and I just do NOT agree. In my opinion. But this I'm sure of... A mindless adherence to blanket rules is SILLY.
I come from a world (line-of-business applications written in higher programming languages (e.g. C#)) where this knowledge comes natural to most of the people I worked with. I've recently dipped my toes into PLC programming and I was amazed how some of the software best-practices seem to only now being slowly picked up in this PLC world. Thanks for spreading this knowledge!
Nice video Jakob, quite interesting topic and the concept you've mentioned is clear for the most.
I started programming PLC's from the late '80s.
At those days the situation was totally different.
The PLC's programming tools rely on IL/AWL , Ladder and few on FBD (Siemens).
The CPU's elaboration power was a lot less and the use of FBs and FCs slowed down the process capabilities, needless to say the low memory available.
Moreover some industries like automotive industries made the use of the Ladder the only one accepted (Allen Bradley for the most in the US and Siemens in Europe).
In the last decade the improvement of the CPUs performances and the IT technoligies introduced for the Industry 4.0 paradigm changed everything.
The Industrial OOP concept in now a must to overcome the complexity but the pace is not so fast and the old habits still last. ;-)
I was introduced to the world of PLC programming and the programs that I was given were such a mess, with global variables, no structures, no case statements, etc... I always dread the days where I need to connect to a customer for support and I have to debug hundreds of lines of code in Main 😂. Beckhoff has some decent introduction to the world of OOP (which does steer you away from globals, in my opinion), but you put the nail in the coffin as far as programming practice goes, for me at least. I have a project coming where I will definitely start using TDD as well, so thanks for that!
For me, the only global variable I use is system's time, which I do not want to pass explicitly to every block that might need it and is only assigned by 1 source, the time provider instance and a few others that really are only written from one source and are only a read for the rest of the program (EtherCAT error state, first cycle, control system's (HMI) connection state).
Very educational video. I did quit automation because the lack of these practices, I hope it gets better in the future and you are helping a lot bringing this information to the automation field.
I feel you should start explaining as well what is a scope before talk about global variables. In automation there are a lot of people that don't even know that, I read in LinkedIn a guy who was confusing global variables with structs.
Hey Alberto! Thanks for your feedback. You're right, so to clarify; The scope of a variable defines where it’s accessible in a program based off of where it was created. Global variables scope is the complete program, while local variables scope can only be accessed in where they are defined (function, function block or method).
Well said! When I first started with PLC programming, I was surprised by how often global variables are used. In any other language it is discouraged. Just look at any Stackoverflow question/answer where someone tries to solve it with global variables :D.
I hope you're also making a follow up video to show people how they can prevent using globals 99% of the time?
And unfortunately this is how most beginners are introduced to declaring variables, as even the official documentation is using globals like there is no tomorrow.
I'll do my best to make more videos about this important topic!
@@JakobSagatowski oh yes. A video as a follow up to show poeople how they can prevent using globals in 99% would be nice.
Good points. Also you didn't mention the static and thread local variables. Including the ones declared at block scope such as function scope. Those types of variables are bad for unit testing too. A usual alternative to those variables is to just declare them at the call site and pass them by reference to the callee.
In general all variables in some PLCs are in reality still global eg. Siemens TIA, instance-FBs are accessible from everywhere 😞
I CoDeSys (Twincat) I still prefer the IO-map to be global, as it represents "physical the machine-structure" which I consider intrinsically "global". And then map using 'REF=', not at all needed but it just still sits right with me 🙂
I must also admit to using it in CodeSys 2.3 for HMI-tags, where the access symbolic setting was easy, and inherited hiracial, so less chance of error in the SYM-file as interface structures, in a certain area was automatically exported.
I fully agree with the generel points in the video, though.
As of the last 13 years of programming in building automation and some industry applications i do agree with you as far as using globals while programming function blocks, functions etc.
My main use of globals are inputs/outputs from the io-cards on the plc and controlvalues related to the specific regulator of a temperature probe etc. This is because it’s pretty much needed in order to have those important values saved as persistent data, used in the scada application and hmi application.
I do however thus far never used them for individual functionblocks, functions or anywhere else unrelated, if i need to take the input of two values in order to make a change or similar before it sets a global variable for an output in my program i use a local variable in that program.
The structure of program that we use means that i have several subprograms with actions for every subpart function for every system in a big building like a hospital and as you say, if i had used global variables for all that it would be a nightmare to troubleshoot and in my case it also could be lethal if i made a change in a wrong spot in that global variable list 😬
I think i understood your point as that i am programming as you are trying to explain in this video 😅
Very informative Jakob!
Tracking the source of the change in the behavior of the program due to the use of global variables is pain in the a**. However I believe it’s up to the individuals’ preference how they want to use the variables. Some project are needed to be delivered quicker so just go with them.
We avoid them mostly but handshaking between the modules is pretty easy and neat using them. Just pass them as input reference variables to the modules and the chain is easy to track.
As a lot of other people in the comment, I use GVL's for IO's (IN, OUT and IO_LINK). I also have a specific GVL for RETAIN and PERSISTENT. What I am more and more doing, is using the outputs of a PROGRAM. So you have a global variable but in read-only access. Usefull to pass the machine state or a particular recipe as input of a process
Great video, well explained
What is also not done in pc coding is the Hungarian notation method
It makes code less readable. Why is the plc world so persistent in this?
I think the primary reason is historically bad/proprietary IDE's with bad highlighting.
And dont get me started about SCADA's like Indusoft, Zenon and the like - they import everything into a flat global scope.
@JakobSagatowski its been a while since your global variable video. I really appreaciate it since i also struggle with the concept of global variables and in particular how beckhoff themselve use it in every example. We have developed an artifical layer where we model each terminals. Those terminals are instantiated in one PRG and in other programs we have interfaces to those terminals models to access the data. The linking of the IOs we do the via pragmas. Anyhow the list of modelled terminals is still global what i dont like. Starting a new project i am now thinking again about a better concept. I am very curious how you are tackling that thing and looking forward for the promised video ;)
AB 5000 coding. We use a set of Data called Mechanism, which includes the bool, dints, real, timers, etc, all used for that mechanism. We also use sequence step methodology, which to me is close to state logic. Basically, the mechanism, which can be either an actual part of the equipment we are integrating, or an idea/process, is the only thing that can modify the values in the data mechanism. Anything, that when planning, looks like it will be tough to separate, because that part of the equipment works depends so much on what another part is doing, is kept together in a mechanism, which might be included under a higher level mechanism, so it is easier to debug parts without integration until later. My opinion is that global is fine, as long as the control engineering group uses a good practice.
Nice explanation. This information will be useful for many individuals 👍👍
yeah but what about global variables that are read only (except of course in very specific places)? What im talking about is something like user preferences, especially in a user interface. thats both impossible to unit test, and unique to the application anyway. For example, i dont see the problem in storing the prefered color scheme in a (read only) variable, so that buttons and other ui elemts can easily get the color theyre supposed to be
Correct me If I am wrong, but the instance of the Fb need to be in global, so you can call the get and set From other parts of the code? Also the persistent and and retain for configuration parameters from the SCADA, I understand they need to be global? And what about status and commands from SCADA? I would love to see how you implement a real life example with all these details... Something simple like a DOL motor, I would love to see a video, showing this paradigm In a real industrial example, if you could make a live would be even better...
I agree, and it's a shame this still has to be mentioned.
Sadly, every few months I see a new powerful loud ignorant that continues to do it wrong.
Is there a list of studies, Pdf's, articles I can drop on them ?
A question, i do not like using variabels, but i program in many diffrent PLC systems. And for example GX Works from Mitsubishi.
There you cant call a program and get the value. For example in twincat you can say POU1.Variabel1 in another program and get reading access to that Variabel. What would you do instead in these situations ?
Hey Jakob, awesome Video. What's your take on passing objects by Reference via VAR_INPUT or as implicit Reference via the good old VAR_IN_OUT ? Are there situations where one would opt for either one in the first place or does it even really matter ? I would argue using VAR_IN_OUT should be the standard way, but one could also argue that the introduction of references makes VAR_IN_OUT a legacy/deprecated feature.
Hey there! I generally prefer VAR_IN_OUT as it's mandatory to supply any parameters, while it for VAR_INPUT is optional (even for references, that's why you have this annoying ISVALIDREF() check that you don't need to use if you use VAR_IN_OUT).
Hello Jakob.
Then What is the best way to share Information between two functions bloks?
Properties
Does it really matter until we get some memory safe languages, like Rust?
Thanks for another great video. But as someone new to industrial automation, HOW would we go about passing data between routines without global variables. For example, ive written a metrics routine that gathers data like cycle time and fault counts from the other routines and don't know how to do that without writing that state to a global variable that can then be read by the metrics routine.
You dont need to use global vars to pass data if both routines have the same local scope. Lets say, for instance, you have 2 different routine instances that need to talk to each other, for that, write them inside the "main" routine and declare local variables in the "main" routine to interface between the two
AHA! I was right. I'm currently coding a trading algorithm and realized that I was having multiple issues with Global Variables. My project is currently over 14,000 lines of code wherein a large section of code was relying on an approximation of 20+ global variables. I was so stressed out to where I begin looking to see which GV's were causing the issues only arrive at the eureka moment that made me say to myself: "FUK GLOBAL VARIABLES" "THEY'RE BAD"! "I'M NEVER USING THEM AGAIN".
It would be great if you can add some examples when mentioning topics in video
Tell one popular open source project which doesn't have global variables.
Saying don't use global variable is like saying there should no knife in this world as it a can cut someones hand.
Just like a knife a global variables like anything else can hurt if you use it in the wrong way.
But it doesn't justify abolition of global variables.
We build a framework to configure, control and monitor devices in the field. For example, you can start, stop, reverse, reset a motor, and monitor for error, running status, stopped status, and other status flags throughout the entire program. This is done using global variables (of structs) and eliminates the need for passing status from block to block. Because of the way it is build, it is still quite easy to monitor (and control), and we haven't found an "easier", or more "elegant" way to achieve this. Also towards the customer, this make the code very easy readable and takes away much of clutter regarding timers and conversions, since this is taken care of by function blocks, compiled in a library. So in our opinion, global vars sill have their uses.
However, for the passing of states and commands between program blocks, we no NOT use global vars and totally agree that there are much better ways to do that.
I declare all the hardware linked variables in a GVL listing (AT%I* & AT%Q*) - there's no particular reason for doing this, just been doing it this way since I was first introduced to this during a training session by Beckhoff Canada. Anyone else does it this way? I'd be curious to know how other programmers do it.
I would say that based on what I've seen in the industry, this is probably the most common way to do it (and one of the reasons why I've created this video in the first place). I'll be making a follow-up video where I will show how you can declare I/O in an alternative way where you don't use global variables.
@@JakobSagatowski I look forward to this video! Variables for IO mapping are one of the few things that I use GVLs for still. Maybe because I too did training at Beckhoff Canada haha
I use a function block and structs to declare my IO mapping and link them in the systems manager to the correaponding hardware IO.
I then link my nececcary global variables for the IO on the funtionblock instead.
Makes it a LOT easier and also gives the benefit that you can change witch value that comes in on what io without relink anything in the system manager and it just takes an online change to the plc program 🙂👍🏻
@@JakobSagatowski its been a while since your global variable video. I really appreaciate it since i also struggle with the concept of global variables and in particular how beckhoff themselve use it in every example. We have developed an atrifical layer where we model each terminal and then have pointers on the terminals to access teh data and do the linking via pragmas. Anyhow the list of modelled terminals is still global what i dont like. Starting a new project i am now thinking again about a better concept. I am very curious how you are tackling that thing and looking forward for the promised video ;)
Anyone here working with Siemens TIA Portal? I am looking for some source of information about proper metodology how to manage your code/project. So far it has been hard to find on the internet. I have not come across similar person ... like Jakob ... for siemens. Do you have some tips?
On MikroTik routers scripts are the same, each person use Global and Global.... I have no idea why all ppl use only them insted of `local` one. Ahh, people...
No 😢
Rust makes it very hard to have a global variable
Haven't had the opportunity to try rust out, will definitely have to try it out!
It is not only on low ressources where Global Variables make sense. We have big machines running with huge IPCs. However, if we don't structure the program with GVL online change etc. is a pain.
Why you don't structure it with Function blocks, methods, inheritance, ...?
@@StefanImhof-w1q we do. but that doesn't exclude GVLs to operate efficiently.
"Stop using global variables!".. In your opinion. Fifty years of programming and I just do NOT agree. In my opinion. But this I'm sure of... A mindless adherence to blanket rules is SILLY.