Skip to main content

Per room heating control with smart thermostats

This post is a continuation of my last post about KNX thermostats, which is part of my series about the troubleshooting of our heating system. In the last post I gave a "software sided" intro on our smart thermostats, but I've found the control theory behind these devices even more fascinating.

But first of all, let me introduce you to our thermostats:

As you can see, there are some leds, a push button and a rotary control labeled from -3 ... +3. The leds indicate the current operating mode (comfort, standby & night mode), as well as frost and heat protection. The rotary control allows the user to customize the temperature in the room: the mid-point is set with KNX parameters and the user may subtract from or add to maximum 3 degrees to that value.

I mentioned in the last article that the midpoints were not set to the same value, which could explain some of the temperature differences. I also mentioned that you can set an adjustment to the temperature measurement, which was also incorrect in one case.

What I wanted to explain in this post how this thermostat controls the valves by setting them on or off.

My initial assumption was that nothing could be simpler: whenever the temperature falls below a threshold (e.g. setpoint minus 0.5C), heating would turn on, and then when it reaches setpoint plus 0.5C it would turn it off. The difference between the lower and higher threshold is the hysteresis, as described nicely in this article. This way room temperature would always be around the setpoint, but would very rarely be the exact temperature that was set.

This assumption proved false, at least with the thermostat on display above.

I had to learn that there are more sophisticated control mechanisms, ones that can set match the setpoints to 0.1C accuracy, even if all the control they have is an on-off valve. A more thorough article about P-I-D (or PID for short) can be found here.

If you are not that patient, a short summary follows.

A PID controller is a combination (e.g. sum) of three terms (e.g. components). The last one or two components can be omitted, e.g. we can talk about a P-only and a PI controller, the first only using one component, the second omitting only the "D" component. The full concept is delivered when all the components are in use.

The goal of the PID controller is to determine how much we want to turn on heating, represented as a percentage from 0 to 100%. I know that I mentioned fully on or fully off valves that are unable to open halfway, much less to a random value, but more on that portion later. Let's just say for now, that the output of the PID function is a percentage: the capacity of heating we want enable. 0% means no heating, 100% means all we have.

Back to PID:
  • P for Proportional: proportional to the difference between setpoint and the current temperature (e.g. let's say 5C below the setpoint means 100% capacity, 2.5C would mean 50%)
  • I for Integral: in addition to P, it also takes the past into account, e.g. if the error is small (thus the P component would not increase capacity much) but has been around for a long time, we would still increase capacity as the time goes.
  • D for Derivative: in addition to PI, the derivative component takes the steepness of the change into account. E.g. it also matters how fast our increase of capacity results in the increase of temperature.
My thermostats are running PI, so they don't have the derivative component. But if you need to control temperature (or something else), look for a PID implementation, you can find one for Arduino here.

The tricky part of setting up PID is to set up the tuning parameters, I can't provide good advice there, if you are in that situation, just read the articles and do measurements to validate them back.

Back to the question of how we apply a percentage based capacity setting to an on-off valve? I already explained this in my last post, but let me repeat a bit: in every 10 minutes, keep them open for a time proportional to the capacity setting. If that setting is 70%, keep the valve open for seven minutes and closed for three minutes. If you need more juice, keep them open longer. At 100% you'll simply be opening the valves completely, without closing them off. This is pretty similar to how PWM is applied to leds, but at a much slower pace. And there's a reason for being slower: valve driving motors may not appreciate if you'd try to open the valve 50 per second :)

Comments

Popular posts from this blog

syslog-ng fun with performance

I like christmas for a number of reasons: in addition to the traditional "meet and have fun with your family", eat lots of delicious food and so on, I like it because this is the season of the year when I have some time to do whatever I feel like. This year I felt like doing some syslog-ng performance analysis. After reading Ulrich Deppert's series about stuff "What every programmer should know about memory" on LWN, I thought I'm more than prepared to improve syslog-ng performance. Before going any further, I'd recommend this reading to any programmer, it's a bit long but every second reading it is worth it. As you need to measure performance in order to improve it, I wrote a tool called "loggen". This program generates messages messages at a user-specifyable rate. Apart from the git repository you can get this tool from the latest syslog-ng snapshots. Loggen supports TCP, UDP and UNIX domain sockets, so really almost everything can be me...

syslog-ng contributions redefined

syslog-ng has been around for about 12 years now, but I think the biggest change in the project's life is imminent: with the upcoming release of syslog-ng OSE 3.2, syslog-ng will become an independent entity. Until now, syslog-ng was primarily maintained & developed by BalaBit, copyrights needed to be reassigned in order to grant BalaBit special privileges. BalaBit used her privileges to create a dual-licensed fork of syslog-ng, named "syslog-ng Premium Edition". The value we offer over the Open Source Edition of syslog-ng are things that larger enterprises require: support on a large number of UNIX platforms (27 as of 3.1), smaller and larger feature differences (like the encrypted/digitally signed logfile feature) better test coverage and release management longer term support Although perfectly legal, this business model was not welcome in various Free Software communities, and has caused friction and harm, because BalaBit has enjoyed a privilege that no others cou...

syslog-ng message parsing

Earlier this month, I announced the new syslog-ng 3.0 git tree, adding a lot of new features to syslog-ng Open Source Edition. I thought it'd be useful to describe the new features with some more details, so this time I'd write about message parsing. First of all, the message structure was a bit generalized in syslog-ng. Earlier it was encapsulating a syslog message and had little space to anything beyond that. That is, every log message that syslog-ng handled had date, host , program and message fields, but syslog-ng didn't care about message contents. This has changed, a LogMessage became a set of name-value pairs , with some "built-in" pairs that correspond to the parts of a syslog message. The aim with this change is: new name-value pairs can be associated with messages through the use of a parsing. It is now possible to parse non-syslog logs and use the columns the same way you could do it with syslog fields. Use them in the name of files, SQL tables or c...