Author Topic: DIY Modbus RTU fan controller, coils vs. registers  (Read 2562 times)

0 Members and 1 Guest are viewing this topic.

Offline LomaxTopic starter

  • Frequent Contributor
  • **
  • Posts: 587
  • Country: eu
  • Minimalist
DIY Modbus RTU fan controller, coils vs. registers
« on: July 14, 2023, 03:04:24 pm »
Hi all,

I'm building a Modbus RTU slave (my first ever) based on the ATMega328p, which is meant to control four fans. Each fan can run at one of three pre-set speeds, which are defined as 8-bit PWM duty cycles in three registers per fan. All registers are stored in EEPROM for persistence. The desired speed is selected with a 2-bit value, representing off, low, medium and high speed. These 2-bit values are also output on individual AVR pins, and used for driving LED indicators.

In my initial version these values are also Modbus registers, and fan speed is selected by writing to those - but this approach has a drawback: since I write to the EEPROM every time a fan's speed is changed, this will eventually wear out. If each fan has its speed changed a handful of times per day the life-time of 100k writes suggested by the AVR datasheet would be exhausted in about a decade. This seems unnecessary, since my application has no need for the selected speeds to persist over power cycles - in fact it may be preferable for all fans to be off at every power on, rather than having them return to the previous level.

My first thought was to simply exclude those registers when writing to EEPROM, and to not trigger any writes when the speed is changed, but this seems a little messy and confusing. I generally tend to think of Modbus registers as contiguous and persistent, though I know there are no hard rules and operation varies wildly between RTUs. So I had a crazy thought: how about I control each fan's level through two "coils" instead? Activating an individual coil selects low or medium level, while activating both selects the high level. Seems like a great idea to me, but I have learned to be wary of great ideas, and thought I'd check what others think - maybe there's a better way? Perhaps I've missed something obvious? Questions & feedback appreciated!
« Last Edit: July 14, 2023, 03:48:51 pm by Lomax »
 

Offline RoGeorge

  • Super Contributor
  • ***
  • Posts: 6806
  • Country: ro
Re: DIY Modbus RTU fan controller, coils vs. registers
« Reply #1 on: July 14, 2023, 04:00:18 pm »
I would only save to EEPROM the last values found in existence, right before the power down.

To detect the shutdown moment, use the brown-out feature of the microcontroller.  The brown-out is a hardware feature of the microcontroller, brown-out detector can generate an interrupt when the power supply falls under a certain voltage.  The brown-out interrupt will write the last values into EEPROM.  Make sure the onboard electrolytic capacitors can still supply energy for long enough (when the board is un-powered), such that the EEPROM save operation will have the time to write before running out of energy.
 
The following users thanked this post: Lomax

Offline LomaxTopic starter

  • Frequent Contributor
  • **
  • Posts: 587
  • Country: eu
  • Minimalist
Re: DIY Modbus RTU fan controller, coils vs. registers
« Reply #2 on: July 14, 2023, 04:02:47 pm »
I would only save to EEPROM the last values found in existence, right before the power down.

That's a neat idea, thank you!

Edit: It won't prevent the selected levels from persisting though, so I'm still interested to hear what Modbus nerds think about using multiple coils to control a single device...
« Last Edit: July 14, 2023, 04:08:09 pm by Lomax »
 

Offline ajb

  • Super Contributor
  • ***
  • Posts: 2735
  • Country: us
Re: DIY Modbus RTU fan controller, coils vs. registers
« Reply #3 on: July 14, 2023, 06:45:33 pm »
If necessary, you can buy more time for a shutdown by providing a localized power domain for just the MCU, isolated from the rest of the logic supply via a diode (or transistor used for the same purpose), with its own bulk capacitance.  That way, if you have a bunch of other logic or control circuitry, you don't need enough capacitance to hold ALL of it up for the shutdown time, just the MCU.  The MCU can monitor the main logic supply voltage via an IO pin (or analog comparator if available) that triggers an interrupt to begin the shutdown procedure once the main voltage drops a little bit, rather than waiting for the MCU supply voltage to hit brownout levels.  You'll need to take care to ensure that the other circuitry doesn't end up pulling power from the MCU supply during shutdown via a GPIO, and prematurely discharge the MCU reserve power. 

Is there generally a convention for modbus registers to be non-volatile?  I would think that would be highly application- and device-dependent.  I have very limited experience in this area, but I believe I've also seen devices where some registers are and some registers aren't non-volatile.  As long as you document what the behavior is, I don't think either one would be an issue.  I could also see an argument for a configuration option to control whether the speed setting is non-volatile, or perhaps two different registers where one is the non-volatile default speed at power on and the other is the volatile current speed setting.  Lots of options to consider just depending on what behavior makes sense for the application. 

As far as using coils -- again, I have limited experience here -- it seems like that would be harder to deal with, since the coils aren't really independent from each other if they are combined to a single 2-bit value.  If it were one coil for on/off and one coil for high/low speed, that would be different, but if you have off/low/med/high then you have to write both coils to set the speed instead of just writing to one register. 
 
The following users thanked this post: Lomax

Offline Siwastaja

  • Super Contributor
  • ***
  • Posts: 8888
  • Country: fi
Re: DIY Modbus RTU fan controller, coils vs. registers
« Reply #4 on: July 14, 2023, 06:55:33 pm »
There are very few conventions in modbus. Persistence can exist or it does not, and may be implemented in various ways, it would be not weird to require the user to write to another register triggering a "save" operation!

If you don't need persistence, just don't implement it; instruct the users to update the desired state regularly.

Some simple wear-leveling wouldn't be too hard to implement either.
« Last Edit: July 14, 2023, 06:57:46 pm by Siwastaja »
 
The following users thanked this post: Lomax

Offline AndyC_772

  • Super Contributor
  • ***
  • Posts: 4284
  • Country: gb
  • Professional design engineer
    • Cawte Engineering | Reliable Electronics
Re: DIY Modbus RTU fan controller, coils vs. registers
« Reply #5 on: July 14, 2023, 07:10:07 pm »
Decide what you'd like the end-user facing interface to look like, and implement that. If that means using a register for each fan, that's fine. If you'd rather use a couple of separate, individual bits ("coils"), then use those. Think of the user experience before anything else; it's why the original iPhone was so successful.

Unless there's a really compelling reason, the interface you present shouldn't be tied to the underlying implementation - in this case, what, how and where you store data in EEPROM.

If the registers need to be non-volatile, and you're concerned about the write endurance, then implement wear levelling. Divide your available EEPROM space into blocks, and in each one, store a unique sequence number along with a copy of the settings. Each time settings change, write them to a new EEPROM area along with the next sequence number. When your board powers up, search the EEPROM for the newest sequence number and use the settings associated with it. You will need to think about how to cope with wrapping around when the EEPROM fills up, as well as what happens if power fails during a write. I'll leave these as an interesting exercise for now - but just remember, you probably have a lot more EEPROM space than you need, so make use of it.
 
The following users thanked this post: Lomax

Offline LomaxTopic starter

  • Frequent Contributor
  • **
  • Posts: 587
  • Country: eu
  • Minimalist
Re: DIY Modbus RTU fan controller, coils vs. registers
« Reply #6 on: July 16, 2023, 10:42:13 am »
Many thanks to everyone for their input! Since no one raised any compelling reason why using two "coils" to represent a 2-bit level might be bad I went ahead and re-wrote my code to do so. This ended up taking longer than I expected, but I now have something that's working. Please pardon my C - it is not my first language.

Code: (C) [Select]
#include <avr/io.h>
#include <avr/interrupt.h>
#include <avr/wdt.h>
#include <avr/eeprom.h>
#include "yaMBSiavr.h"

// why is BAUD definition here ignored by #ifndef in yaMBSiavr.h?
// #define BAUD 9600L

#define channels 4
#define regsize 16
#define read_eeprom_array(address,value_p,length) eeprom_read_block ((void *)value_p, (const void *)address, length)
#define write_eeprom_array(address,value_p,length) eeprom_write_block ((const void *)value_p, (void *)address, length)

volatile uint8_t instate = 0;
volatile uint8_t outstate = 0;
volatile uint16_t EEMEM eeprom[regsize];
volatile uint16_t registers[regsize];
volatile uint8_t levels[channels];

static uint8_t const BTN0 = 0b00000100; //PORTC
static uint8_t const BTN1 = 0b00100000; //PORTC
static uint8_t const BTN2 = 0b00100000; //PORTB
static uint8_t const BTN3 = 0b00000001; //PORTD

void read_eeprom(void) {
read_eeprom_array(eeprom, registers, regsize * 2);
}
void write_eeprom(void) {
write_eeprom_array(eeprom, registers, regsize * 2);
}

void setup(void) {
// output on B4,B5 (LEDs ch4) & B1,B2,B3 (PWM ch1-3)
  DDRB |= 0b00111110;
// output on C0,C1 & C3,C4 (LEDs ch1 & 2)
  DDRC |= 0b00011011;
// output on D6,D7 (LEDs ch3) & D3 (PWM ch4)
  DDRD |= 0b11001000;
// pull-up on B0 (button input ch4)
PORTB |= BTN3;
// pull-up on C2,C5 (button inputs ch1 & 2)
PORTC |= BTN0 | BTN1;
// pull-up on D5 (button input ch3)
PORTD |= BTN2;

// UART clock on timer0
TCCR0B |= (1<<CS01); //prescaler 8
TIMSK0 |= (1<<TOIE0);

// 8-bit Fast PWM on timer1 (on B1,B2)
TCCR1A |= _BV(COM1A1) | _BV(COM1B1) | _BV(WGM10);
TCCR1B |= _BV(CS12) | _BV(WGM12); // 30.52 Hz (prescaler 256)

// 8-bit Fast PWM on timer2 (on B3,D3)
TCCR2A = _BV(COM2A1) | _BV(COM2B1) | _BV(WGM21) | _BV(WGM20);
TCCR2B = _BV(CS21) | _BV(CS22); // 30.52 Hz (prescaler 256)

// set all PWM to level 0
OCR1A = registers[0];
OCR1B = registers[4];
OCR2A = registers[8];
OCR2B = registers[12];
}

ISR(TIMER0_OVF_vect) {
modbusTickTimer();
}

void setLevel(uint8_t channel, uint8_t level) {
levels[channel] = level;
switch(channel) {
case 0: {
OCR1A = registers[0 + level];
PORTC = (levels[0] | (levels[1]<<3)) | BTN0 | BTN1;
}
break;
case 1: {
OCR1B = registers[4 + level];
PORTC = (levels[0] | (levels[1]<<3)) | BTN0 | BTN1;
}
break;
case 2: {
OCR2A = registers[8 + level];
PORTD = levels[2]<<6 | BTN2;
}
break;
case 3: {
OCR2B = registers[12 + level];
PORTB = levels[3]<<4 | BTN3;
}
break;
}
}

void applyOutstate(void) {
for (int i = 0; i < channels; i++) {
uint8_t state = (outstate >> (i*2)) & 0b00000011;
setLevel(i, state);
}
}

void incrChannel(uint8_t channel) {
uint8_t new_level = (levels[channel] + 1) % 4;
outstate &= ~(0b00000011<<(channel*2)); // Clear channel
outstate |= new_level<<(channel*2); // Set channel
applyOutstate();
}

void modbusGet(void) {
if (modbusGetBusState() & (1<<ReceiveCompleted)) {
switch(rxbuffer[1]) {
case fcReadCoilStatus: {
modbusExchangeBits(&outstate,0,8);
}
break;
case fcForceSingleCoil: {
modbusExchangeBits(&outstate,0,8);
applyOutstate();
}
break;
case fcForceMultipleCoils: {
modbusExchangeBits(&outstate,0,8);
applyOutstate();
}
break;
case fcReadHoldingRegisters: {
modbusExchangeRegisters(registers,0,regsize);
}
break;
case fcPresetSingleRegister: {
modbusExchangeRegisters(registers,0,regsize);
write_eeprom();
applyOutstate();
}
break;
case fcPresetMultipleRegisters: {
modbusExchangeRegisters(registers,0,regsize);
write_eeprom();
applyOutstate();
}
break;
default: {
modbusSendException(ecIllegalFunction);
}
break;
}
}
}

int main(void) {
read_eeprom();
setup();
sei();
modbusSetAddress(0x01);
modbusInit();
wdt_enable(7);

// TODO: enforce limits on register values
// TODO: make PWM phase configurable
// TODO: make PWM frequency configurable
// TODO: make slave address configurable
// TODO: make comm parameters configurable
// TODO: move some stuff to a header file
// TODO: button debouncing
while(1) {
wdt_reset();
modbusGet();
if((PINC & BTN0) == 0) {
incrChannel(0);
while((PINC & BTN0) == 0); // Wait until the button is released
}
if((PINC & BTN1) == 0) {
incrChannel(1);
while((PINC & BTN1) == 0);
}
if((PIND & BTN2) == 0) {
incrChannel(2);
while((PIND & BTN2) == 0);
}
if((PINB & BTN3) == 0) {
incrChannel(03);
while((PINB & BTN3) == 0);
}
}
}

I'm running this in Simulide, with a virtual serial port that I can talk to from a simple Python program:

Code: (Python) [Select]
#!/usr/bin/env python3
import minimalmodbus

instrument = minimalmodbus.Instrument("/dev/pts/7", 1)
instrument.serial.baudrate = 9600
instrument.serial.timeout = 2

while True:
cmd = input("C/R:")
if cmd == "r" or cmd == "R":
register = input("Register:")
if register:
register = int(register)
value = int(input("Value:"))
instrument.write_register(register, value)
val = instrument.read_registers(0, 16)  # start, count
print(val)
elif cmd == "c" or cmd == "C":
coil =input("Coil:")
if coil:
coil = int(coil)
on = int(input("1/0:"))
instrument.write_bit(coil, on)
val = instrument.read_bits(0, 8, 1)  # start, count, fc
print(val)

# set up two connected virtual serial ports with:
# socat -d -d pty,link=/tmp/vtty1,raw,echo=0 pty,link=/tmp/vtty2,raw,echo=0



I have lowered the AVR CPU frequency to 2 MHz and adjusted the clock prescalers accordingly - everything still works beautifully!


« Last Edit: July 16, 2023, 08:32:05 pm by Lomax »
 

Offline LomaxTopic starter

  • Frequent Contributor
  • **
  • Posts: 587
  • Country: eu
  • Minimalist
Re: DIY Modbus RTU fan controller, coils vs. registers
« Reply #7 on: July 16, 2023, 07:27:06 pm »
I would only save to EEPROM the last values found in existence, right before the power down.

I still think this is a very elegant solution, but will save it for another project, which is much more complex (and has been on hold for years).

If necessary, you can buy more time for a shutdown by providing a localized power domain for just the MCU, isolated from the rest of the logic supply via a diode (or transistor used for the same purpose), with its own bulk capacitance.

Nice - and controlling this transistor from the MCU could have several advantages.

Is there generally a convention for modbus registers to be non-volatile?

I really don't know, but like you I also have quite limited experience. AFAICR the four other devices (from three manufacturers) on the bus that the fan controller will connnect to only have persistent r/w or persistent read only registers. I can't recall ever seeing anything else, but again I have limited experience. Or actually, one exception is a counter with a reset register which if you write to the counter will reset. Still, to my mind it seems natural that if I write a value to a configuration register then that value will automatically persist.

As far as using coils -- again, I have limited experience here -- it seems like that would be harder to deal with, since the coils aren't really independent from each other if they are combined to a single 2-bit value.  If it were one coil for on/off and one coil for high/low speed, that would be different, but if you have off/low/med/high then you have to write both coils to set the speed instead of just writing to one register.

It's a little more complicated to deal with on the code side, but we're only talking about some bitwise-operation-magic - the hard part is designing those operations; bit of a mindbender for a C novice like me. The actual code is only a handful of extra lines, none of which will tax the CPU. Internally the coils are handled as bytes anyway, into which the eight I'm using fit very nicely (see code above if curious). I like the semantics of your suggestion to use one register for on/off and one for high/low, but I would have to forgoe the medium level which is not an option. And you would still have to "force multiple coils" to set the speed in those cases where the second bit was not already set to the desired speed.

Think of the user experience before anything else

Yeah, that's what I try to do, though the implementation is always guided by the language - I know it's good when the code doesn't feel forced, or arbitrarily contrived. I'm also very much a C novice, so have to keep it simple for that reason too.

If the registers need to be non-volatile, and you're concerned about the write endurance, then implement wear levelling.

Some simple wear-leveling wouldn't be too hard to implement either.

Another nice idea I hadn't thought of. Makes perfect sense, but seems rather overkill for this application. The registers are only really written to during commissioning, to configure the device and its channels. Since the active level of each channel is now set using coils the registers will only rarely need to be touched post deployment - certainly not with a frequency that could cause any significant wear.
« Last Edit: July 16, 2023, 07:42:34 pm by Lomax »
 

Offline ajb

  • Super Contributor
  • ***
  • Posts: 2735
  • Country: us
Re: DIY Modbus RTU fan controller, coils vs. registers
« Reply #8 on: July 16, 2023, 09:10:10 pm »
Nice!

As far as using coils -- again, I have limited experience here -- it seems like that would be harder to deal with, since the coils aren't really independent from each other if they are combined to a single 2-bit value.  If it were one coil for on/off and one coil for high/low speed, that would be different, but if you have off/low/med/high then you have to write both coils to set the speed instead of just writing to one register.

It's a little more complicated to deal with on the code side, but we're only talking about some bitwise-operation-magic - the hard part is designing those operations; bit of a mindbender for a C novice like me. The actual code is only a handful of extra lines, none of which will tax the CPU. Internally the coils are handled as bytes anyway, into which the eight I'm using fit very nicely (see code above if curious). I like the semantics of your suggestion to use one register for on/off and one for high/low, but I would have to forgoe the medium level which is not an option. And you would still have to "force multiple coils" to set the speed in those cases where the second bit was not already set to the desired speed.

I was actually thinking about the other side of the MODBUS interface, whatever is sending messages to control the fan controller.  The register method allows a pretty direct translation from the intended operation to the required message: "set fan 1 to medium" simply becomes "fan1_register = value_that_represents_medium".  On the other hand, with two coils representing three speeds and 'off', the desired state must be decomposed into two separate binary values.  So "set fan 1 to medium" becomes "fan1_low_coil = off; fan1_high_coil = on" or whatever, where the boolean values have a less clear relationship to the intended state.  Reading back that state, if you need to do that, is similarly more complicated, since you can't just look at one value to see if the fan is on.  There are ways to deal with that, of course, but it seems like an unnecessary complication. 
 
The following users thanked this post: Lomax

Offline LomaxTopic starter

  • Frequent Contributor
  • **
  • Posts: 587
  • Country: eu
  • Minimalist
Re: DIY Modbus RTU fan controller, coils vs. registers
« Reply #9 on: July 17, 2023, 12:08:38 am »
I was actually thinking about the other side of the MODBUS interface, whatever is sending messages to control the fan controller.

Understood. I expect most "force multiple coils" implementations will take a start coil, a count and the bits for those, though accepted datatypes for the bits will vary. I use the node-red-contrib-modbus modbus-flex-write node and it wants an array of bits, which this JS one-liner will deliver:

Code: [Select]
value = 1;
arr = new Array(2).fill().map((x, i) => { return (value >> i) & 1 });
console.log(arr);

[1,0]
« Last Edit: July 17, 2023, 12:12:26 am by Lomax »
 

Offline ajb

  • Super Contributor
  • ***
  • Posts: 2735
  • Country: us
Re: DIY Modbus RTU fan controller, coils vs. registers
« Reply #10 on: July 17, 2023, 02:59:59 am »
It's not often we C programmers get to give the JS/Python crowd shit for being obtuse and not expressive  :box:
« Last Edit: July 17, 2023, 03:04:42 am by ajb »
 

Offline RoGeorge

  • Super Contributor
  • ***
  • Posts: 6806
  • Country: ro
Re: DIY Modbus RTU fan controller, coils vs. registers
« Reply #11 on: July 17, 2023, 06:54:18 am »
First, congrats for making it work!  :-+

About coils vs register, if you know your application can work OK without a register, then use coils.  In theory, a register would be preferred.

For the industrial process, a register would be safer than two coils (no matter it's written in Python or C), because writing a register is an atomic operation, all bits are changed at once.

Why being atomic (which is usually a software consideration) would be important for a fan?  Because, for example, you may want to increase the ventilation from fan=1 to fan=2 (in binary from 01 to 10).  Notice you need to change both bits.  If one of the two set-coil commands gives an error, the fan might end in 11, or in 00, instead of the desired 10.  Trying to increase a fan, and ending with fan-off instead, could be very bad for the industrial process.

If the fan values are meant to only increment/decrement by 1, then you can use Gray Code to encode the fan states.  Grey code has the property that only one bit at a time changes (at an increment/decrement).  In this case, would mean you will never need to send 2 set-coils commands.  By choosing Grey code, the fan can be controlled with a single set-coil instead of two.

If the fan states are not incremental, then the Gray code can not fix atomic problems, and to make sure both bits arrived well, you'll have to read back the state of the coils after each set-fan operation.
« Last Edit: July 17, 2023, 07:13:47 am by RoGeorge »
 

Offline Siwastaja

  • Super Contributor
  • ***
  • Posts: 8888
  • Country: fi
Re: DIY Modbus RTU fan controller, coils vs. registers
« Reply #12 on: July 17, 2023, 08:19:53 am »
Now that I think about it, having one fan per one modbus register would be easier to user, than mapping one fan across two registers. (Using register here as a generic term for something that has unique address.)

Modbus devices are often controlled from all sorts of industrial programmable logic devices and while it is possible to make multi-register writes etc., it would be easier to explain and implement "write value 0-2 to register 20005" than to "write first bit to coil 10001 and second bit to coil 10002".

In practice you can and will use write multiple coils operation for atomicity and communication efficiency, but how do you force users to always do that?
« Last Edit: July 17, 2023, 08:21:59 am by Siwastaja »
 

Offline LomaxTopic starter

  • Frequent Contributor
  • **
  • Posts: 587
  • Country: eu
  • Minimalist
Re: DIY Modbus RTU fan controller, coils vs. registers
« Reply #13 on: July 17, 2023, 10:54:40 am »
Ok, you've convinced me with good arguments - I will revert back to using registers  :-+

It's not often we C programmers get to give the JS/Python crowd shit for being obtuse and not expressive  :box:

No I agree, C definitely seems a lot more bit friendly! That JS one-liner is awful, but it's the briefest possible way to perform this operation - with the possible exception of

Code: [Select]
value = 1;
arr = [0,0].map((x, i) => { return (value >> i) & 1 });
console.log(arr);

But then you lose the parameterisation of the bit-length. That said, I'm not sure there is a much prettier way to convert an integer to an array of bits in C?
 

Offline LomaxTopic starter

  • Frequent Contributor
  • **
  • Posts: 587
  • Country: eu
  • Minimalist
Re: DIY Modbus RTU fan controller, coils vs. registers
« Reply #14 on: July 17, 2023, 11:52:27 am »
Ok, here's the updated code:

Code: (C) [Select]
#include <avr/io.h>
#include <avr/interrupt.h>
#include <avr/wdt.h>
#include <avr/eeprom.h>
#include "yaMBSiavr.h"

// why is BAUD definition here ignored by #ifndef in yaMBSiavr.h?
// #define BAUD 9600L

#define channels 4
#define regsize 20
#define read_register_array(address,value_p,length) eeprom_read_block ((void *)value_p, (const void *)address, length)
#define write_register_array(address,value_p,length) eeprom_write_block ((const void *)value_p, (void *)address, length)

volatile uint8_t instate = 0;
volatile uint8_t outstate = 0;
volatile uint16_t EEMEM eeprom[regsize];
volatile uint16_t registers[regsize];

static uint8_t const BTN0 = 0b00000100; //PORTC
static uint8_t const BTN1 = 0b00100000; //PORTC
static uint8_t const BTN2 = 0b00100000; //PORTB
static uint8_t const BTN3 = 0b00000001; //PORTD

void loadRegisters(void) {
read_register_array(eeprom, registers, regsize * 2);
}
void saveRegisters(void) {
write_register_array(eeprom, registers, regsize * 2);
}

void setup(void) {
// output on B4,B5 (LEDs ch4) & B1,B2,B3 (PWM ch1-3)
  DDRB |= 0b00111110;
// output on C0,C1 & C3,C4 (LEDs ch1 & 2)
  DDRC |= 0b00011011;
// output on D6,D7 (LEDs ch3) & D3 (PWM ch4)
  DDRD |= 0b11001000;
// pull-up on B0 (button input ch4)
PORTB |= BTN3;
// pull-up on C2,C5 (button inputs ch1 & 2)
PORTC |= BTN0 | BTN1;
// pull-up on D5 (button input ch3)
PORTD |= BTN2;

// UART clock on timer0
TCCR0B |= (1<<CS01); //prescaler 8
TIMSK0 |= (1<<TOIE0);

// 8-bit Fast PWM on timer1 (on B1,B2)
TCCR1A |= _BV(COM1A1) | _BV(COM1B1) | _BV(WGM10);
//TCCR1B |= _BV(CS20) | _BV(WGM12); // 31.25 KHz (prescaler 1)
TCCR1B |= _BV(CS12) | _BV(WGM12); // 30.52 Hz (prescaler 256)

// 8-bit Fast PWM on timer2 (on B3,D3)
TCCR2A = _BV(COM2A1) | _BV(COM2B1) | _BV(WGM21) | _BV(WGM20);
//TCCR2B = _BV(CS20); // 31.25 KHz (prescaler 1)
TCCR2B = _BV(CS21) | _BV(CS22); // 30.52 Hz (prescaler 256)
}

ISR(TIMER0_OVF_vect) {
modbusTickTimer();
}

void setLevel(uint8_t channel, uint8_t level) {
registers[channel] = level;
switch(channel) {
case 0: {
OCR1A = registers[4 + level];
PORTC = (registers[0] | (registers[1]<<3)) | BTN0 | BTN1;
}
break;
case 1: {
OCR1B = registers[8 + level];
PORTC = (registers[0] | (registers[1]<<3)) | BTN0 | BTN1;
}
break;
case 2: {
OCR2A = registers[12 + level];
PORTD = registers[2]<<6 | BTN2;
}
break;
case 3: {
OCR2B = registers[16 + level];
PORTB = registers[3]<<4 | BTN3;
}
break;
}
}

void updateLevels(void) {
for(int i = 0; i < channels; i++) {
setLevel(i, registers[i]);
}
}

void incrChannel(uint8_t channel) {
uint8_t new_level = (registers[channel] + 1) % 4;
setLevel(channel, new_level);
}

void modbusGet(void) {
if (modbusGetBusState() & (1<<ReceiveCompleted)) {
switch(rxbuffer[1]) {
case fcReadHoldingRegisters: {
modbusExchangeRegisters(registers,0,regsize);
}
break;
case fcPresetSingleRegister: {
modbusExchangeRegisters(registers,0,regsize);
updateLevels();
saveRegisters();
}
break;
case fcPresetMultipleRegisters: {
modbusExchangeRegisters(registers,0,regsize);
updateLevels();
saveRegisters();
}
break;
default: {
modbusSendException(ecIllegalFunction);
}
break;
}
}
}

int main(void) {
loadRegisters();
setup();
sei();
modbusSetAddress(0x01);
modbusInit();
wdt_enable(7);
updateLevels();

// TODO: do not write EEPROM for registers 0-3
// TODO: assemble LED pins into a byte that can be written to? virtual port?
// TODO: simplify PORTC = (levels[0] | (levels[1]<<3)) | BTN0 | BTN1;
// TODO: enforce limits on register values
// TODO: make PWM phase configurable
// TODO: make PWM frequency configurable
// TODO: make slave address configurable
// TODO: make comm parameters configurable
// TODO: move some things to a header file
// TODO: button debouncing (idelly non-blocking)
// TODO: add reset button (factory defaults)
while(1) {
wdt_reset();
modbusGet();
if((PINC & BTN0) == 0) {
incrChannel(0);
while((PINC & BTN0) == 0); // Wait until the button is released
}
if((PINC & BTN1) == 0) {
incrChannel(1);
while((PINC & BTN1) == 0);
}
if((PIND & BTN2) == 0) {
incrChannel(2);
while((PIND & BTN2) == 0);
}
if((PINB & BTN3) == 0) {
incrChannel(3);
while((PINB & BTN3) == 0);
}
}
}

That's 23 lines less actual code, and it feels a lot more elegant - though I haven't figured out a nice way to "unpersist" registers 0-3. The level doesn't persist from button presses, but it does when modifying registers via Modbus. But then it might actually be desirable to be able to set the power-on level for each channel; perhaps the best route is to have a "save settings" register to write to which triggers the save (as suggested by @Siwastaja).

I must have made a mistake somewhere though, because I now get blips in the level 0 PWM signal (255) that weren't there before:


 
I've looked and looked but can't see where I'm doing some repeated call that might cause this interruption. The main loop does nothing unless there is an input (button press or Modbus command), so I don't understand  :-//
« Last Edit: July 17, 2023, 12:09:57 pm by Lomax »
 

Offline ledtester

  • Super Contributor
  • ***
  • Posts: 3248
  • Country: us
Re: DIY Modbus RTU fan controller, coils vs. registers
« Reply #15 on: July 17, 2023, 12:41:49 pm »
Quote
In my initial version these values are also Modbus registers, and fan speed is selected by writing to those - but this approach has a drawback: since I write to the EEPROM every time a fan's speed is changed, this will eventually wear out. ...

Another idea... save the fan settings to the EEPROM but only when a certain MODBUS command is received (or register is written). That is, the user decides when the fan settings are saved.
« Last Edit: July 17, 2023, 12:43:20 pm by ledtester »
 
The following users thanked this post: Lomax

Offline ajb

  • Super Contributor
  • ***
  • Posts: 2735
  • Country: us
Re: DIY Modbus RTU fan controller, coils vs. registers
« Reply #16 on: July 18, 2023, 02:18:25 am »
That said, I'm not sure there is a much prettier way to convert an integer to an array of bits in C?

Well, fundamentally any integer IS an array of bits.  C provides minimal abstraction of this, so you just have to pick (or construct) the integer that gives you the right pattern of bits.  You'd probably do something like:

Code: [Select]
#define FAN_MODE_OFF    0x0
#define FAN_MODE_LOW    0x1
#define FAN_MODE_MED    0x2
#define FAN_MODE_HIGH   0x3

or if you're feeling fancy:

Code: [Select]
typedef enum {
    FAN_MODE_OFF,
    FAN_MODE_LOW,
    FAN_MODE_MED,
    FAN_MODE_HIGH,
} fan_mode_enum;

It doesn't matter what numeric representation you use, but hex is conventional for values where the bit pattern is important.  Enumerated values in C are guaranteed to start at zero and increment by one, and represent literal values just like #defines do, so the result is the same.  The enum is just a convenient way to define the values.  More complicated values might be constructed from bitwise operations, but those expressions are typically reduced to simple literal values during compilation.

You'd expect to hand it to a function that looks something like:

Code: [Select]
void modbus_writeCoils(uint8_t stationAddress, uint16_t startCoil, uint16_t coilCount, uint32_t coilBits);

// or if you want/have to write one at a time:
modbus_writeCoil(addr, FAN_MODE_LOW_COIL, FAN_MODE_LOW);
modbus_writeCoil(addr, FAN_MODE_HIGH_COIL, (FAN_MODE_LOW>>1) );

Okay, I'll grant that none of these are really arrays in the sense that you mean, so if you REALLY want an array of bits, you could do:

Code: [Select]
bool fan_mode_low[2] = {0,1};
but each entry takes up at least one byte, so it's very storage inefficient--well, inefficient compared to bitfields.  It's still way more efficient than an array of bits in python or JS  8) :P
« Last Edit: July 18, 2023, 04:26:40 am by ajb »
 

Offline WattsThat

  • Frequent Contributor
  • **
  • Posts: 778
  • Country: us
Re: DIY Modbus RTU fan controller, coils vs. registers
« Reply #17 on: July 18, 2023, 03:06:20 am »
Quote
In my initial version these values are also Modbus registers, and fan speed is selected by writing to those - but this approach has a drawback: since I write to the EEPROM every time a fan's speed is changed, this will eventually wear out. ...

Another idea... save the fan settings to the EEPROM but only when a certain MODBUS command is received (or register is written). That is, the user decides when the fan settings are saved.

This +1. Make the write to eeprom dependent upon a Modbus register write with a specific value. Think of it as a “save config” command.
 

Offline LomaxTopic starter

  • Frequent Contributor
  • **
  • Posts: 587
  • Country: eu
  • Minimalist
Re: DIY Modbus RTU fan controller, coils vs. registers
« Reply #18 on: July 18, 2023, 06:39:55 am »
Quote
In my initial version these values are also Modbus registers, and fan speed is selected by writing to those - but this approach has a drawback: since I write to the EEPROM every time a fan's speed is changed, this will eventually wear out. ...

Another idea... save the fan settings to the EEPROM but only when a certain MODBUS command is received (or register is written). That is, the user decides when the fan settings are saved.

This +1. Make the write to eeprom dependent upon a Modbus register write with a specific value. Think of it as a “save config” command.

I guess this makes it +4? ;)

perhaps the best route is to have a "save settings" register to write to which triggers the save (as suggested by @Siwastaja).
 
The following users thanked this post: ledtester

Offline Wiljan

  • Regular Contributor
  • *
  • Posts: 230
  • Country: dk
Re: DIY Modbus RTU fan controller, coils vs. registers
« Reply #19 on: July 18, 2023, 03:10:07 pm »
I did make another arduino project where I needed to save some variable's when the power goes off, so they could be restored next time the power comes back.

It's not an good idea to save often to the eeprom since it will fail then over time

I did use the comparator input and did make a resistor divider between the PSU +24v (before regulated to 5v) so the resistor divider then feed in a voltage on the comparator and if it goes below the threshold it generate an interrupt saving the variables to eeprom and stops.

It's far faster than using the ADC to detect the power goes down, when thats said if you have many variables you want to store it takes time and you need the power to drop slowly.

So in another project I did use FRAM with SPI (you can also have them with I2C) this is absolutly the best solution very fast and they remember without power so you can use them directly as variables...  job done  ;D

https://learn.adafruit.com/adafruit-i2c-fram-breakout/wiring-and-test
https://learn.adafruit.com/adafruit-spi-fram-breakout

 
The following users thanked this post: Lomax

Offline RoGeorge

  • Super Contributor
  • ***
  • Posts: 6806
  • Country: ro
Re: DIY Modbus RTU fan controller, coils vs. registers
« Reply #20 on: July 18, 2023, 03:59:11 pm »
About FRAM, while not of a practical concern, it still wears out.  Much slower than an EEPROM, but what's worst is that the FRAM wears out not only when writing, but it wears out when reading, too.  :scared:

In an endless loop of executing a JMP $ from FRAM, at 16MHz it will wear out in only 2 years.  :o
https://e2e.ti.com/support/microcontrollers/msp-low-power-microcontrollers-group/msp430/f/msp-low-power-microcontroller-forum/132022/fram-endurance

Offline Wiljan

  • Regular Contributor
  • *
  • Posts: 230
  • Country: dk
Re: DIY Modbus RTU fan controller, coils vs. registers
« Reply #21 on: July 19, 2023, 07:55:02 am »
About FRAM, while not of a practical concern, it still wears out.  Much slower than an EEPROM, but what's worst is that the FRAM wears out not only when writing, but it wears out when reading, too.  :scared:

In an endless loop of executing a JMP $ from FRAM, at 16MHz it will wear out in only 2 years.  :o
https://e2e.ti.com/support/microcontrollers/msp-low-power-microcontrollers-group/msp430/f/msp-low-power-microcontroller-forum/132022/fram-endurance

Interesting information  :-+

If you only write to the FRAM when a variable actually does change in your code and only read it back when your system is powered on, then FRAM should be safe for more power-cycles than the power switch can handle  :)
 


Share me

Digg  Facebook  SlashDot  Delicious  Technorati  Twitter  Google  Yahoo
Smf