Multiple Modes

As a first example application I will use the reset counter to switch between different knight rider modes. We just combine the previous sketches like this.

//
//	www.blinkenlight.net
//
//	Copyright 2011 Udo Klein
//
//	This program is free software: you can redistribute it and/or modify
//	it under the terms of the GNU General Public License as published by
//	the Free Software Foundation, either version 3 of the License, or
//	(at your option) any later version.
//
//	This program is distributed in the hope that it will be useful,
//	but WITHOUT ANY WARRANTY; without even the implied warranty of
//	MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
//	GNU General Public License for more details.
//
//	You should have received a copy of the GNU General Public License
//	along with this program. If not, see http://www.gnu.org/licenses/


#include <EEPROM.h>

uint8_t get_next_count(const uint8_t count_limit) {
	// n cells to use --> 1/n wear per cll --> n times the life time
	const uint16_t cells_to_use = 128;

	// default cell to change
	uint8_t change_this_cell  = 0;
	// value of the default cell
	uint8_t change_value = EEPROM.read(change_this_cell);

	// will be used to aggregate the count_limit
	// must be able to hold values up to cells_to_use*255 + 1
	uint32_t count = change_value;

	for (uint16_t cell = 1; cell < cells_to_use; ++cell) {
		uint8_t value = EEPROM.read(cell);

		// determine current count by cummulating all cells
		count += value;

		if (value != change_value ) {
			// at the same time find at least one cell that differs
			change_this_cell = cell;
		}
	}

	// Either a cell differs from cell 0 --> change it
	// Otherwise no cell differs from cell 0 --> change cell 0

	// Since a cell might initially hold a value of -1 the % operator must be applied twice
	EEPROM.write(change_this_cell, (EEPROM.read(change_this_cell) % count_limit + 1) % count_limit);

	// return the new count
	return (count + 1) % count_limit;
}

uint8_t brightness_pattern_1(const int8_t led, const int8_t pos) {
	switch (abs(led-pos)) {
		case 0: 	return 32;
		case 1: 	return 16;
		case 2: 	return 6;
		case 3: 	return 2;
		default:	return 1;
	}
}
uint8_t brightness_pattern_2(const int8_t led, const int8_t pos) {
	switch (min(abs(led-pos),abs(19-led-pos))) {
		case 0: 	return 32;
		case 1: 	return 16;
		case 2: 	return 6;
		case 3: 	return 2;
		default:	return 1;
	}
}

uint8_t brightness_pattern_3(const int8_t led, const int8_t pos) {
	int8_t tmp = pos - abs(2*led-19)+2;
	return (tmp>0? tmp+tmp>>1: 0);
}


typedef uint8_t (*brightness_pattern)(const int8_t, const int8_t);

brightness_pattern pattern[] = {  brightness_pattern_1,
								  brightness_pattern_2,
								  brightness_pattern_3 };

brightness_pattern brightness;

void setup() {
	for (uint8_t pin=0; pin<20; ++pin) {
		pinMode(pin, OUTPUT);	
	}

	brightness = pattern[get_next_count(sizeof(pattern)/sizeof(pattern[0]))];
}

void pulse_width_modulation(const uint8_t pos) {
	for(uint8_t times=0; times<10; ++times) {
		for (uint8_t pass=0; pass<32; ++pass) {
			for (int8_t led=0; led<20; ++led) {
				digitalWrite(led, (brightness(led, pos) > pass));
			}
		}
	}
}

void loop() {
	static uint8_t pos=0;

	while(pos<20) {
		pulse_width_modulation(pos);
		++pos;
	}

	while(pos>0) {
		--pos;
		pulse_width_modulation(pos);
	}
}

Notice that this sketch does not include a case statement for the different brightness modes. Instead the mode selection is caused by the following pieces of code.

typedef uint8_t (*brightness_pattern)(const int8_t, const int8_t);

brightness_pattern pattern[] = {  brightness_pattern_1,
								  brightness_pattern_2,
								  brightness_pattern_3 };

brightness_pattern brightness;

This sequence defines a function pointer type for functions returning a uint8_t parameter and consuming two int8_t parameters. Then the 3 brightness pattern functions are put into an array. Selection of the desired brightness function happens during setup with the line.

	brightness = pattern[get_next_count(sizeof(pattern)/sizeof(pattern[0]))];

I used this slightly complicated approach for several reasons.

  1. I do not like case statements for things that can be done by table lookup.
  2. The Arduino has no execution pipeline that might get stalled by this approach.
  3. This approach allows to add new patterns in a declarative manner without changing the „real“ code.

As promised this setup now uses the reset switch for mode selection. See the video about our success.

The next step is not to deal with undesirable flicker. That is I investigate how to remove flicker.

Advertisement

2 Responses to Multiple Modes

  1. al says:

    Hello,

    First, I’d like to say thank-you for the blog. While I’m fairly new to the Arduino, I have learned much from reading your posts. This one in particular I’ve used to add LEDs to a RC helicopter for night-flying.

    Is it possible to count >7 entries in the brightness_pattern array? After adding an 8th entry, my patterns are replaced by a single repeating pattern. Hitting the reset button only restarts this same pattern again. Commenting out the entry returns things to normal, though.

    • This is strange. As far as I implemented it there is no magic limit to 7 patterns. Can you put Serial.print statements into your code and tell me the values of “count_limit” (the argument to get_next_count) as well as the result of get_next_count()?

      If this shows nothing special then the only other thing that I can think of is that you might be running out of SRAM. This happens sometimes if your programs get bigger and if you do not take enough care to keep the number of global variables low. Especially the use of strings without PROGMEM macros exhausts SRAM easily. To say more I would need to see your code. I suggest that you upload your code such that I can have a look.

      Actually it might be a very good idea to put your code to the Arduino forum such that others can help as well as benefit from it.

Leave a Reply

Fill in your details below or click an icon to log in:

WordPress.com Logo

You are commenting using your WordPress.com account. Log Out /  Change )

Facebook photo

You are commenting using your Facebook account. Log Out /  Change )

Connecting to %s

This site uses Akismet to reduce spam. Learn how your comment data is processed.