Skip to content

Instantly share code, notes, and snippets.

Show Gist options
  • Save kylemcdonald/1447925 to your computer and use it in GitHub Desktop.
Save kylemcdonald/1447925 to your computer and use it in GitHub Desktop.
BlinkM MaxM RGB color from a knob
#include <i2cmaster.h>
#include "Wire.h"
#include "BlinkM_funcs.h"
const float lowReading = 75;
const float highReading = 110;
const int blinkm_addr = 0;
const unsigned char separatorCharacter = 255;
void setup(){
Serial.begin(9600);
Serial.println("Setup...");
i2c_init(); //Initialise the i2c bus
PORTC = (1 << PORTC4) | (1 << PORTC5);//enable pullups
Serial.println("done.");
}
float normf(float x, float low, float high) {
float y = (x - low) * 255.f / (high - low);
if(y > 255) {
y = 255;
}
if(y < 0) {
y = 0;
}
return y;
}
void loop(){
int dev = 0x5A<<1;
int data_low = 0;
int data_high = 0;
int pec = 0;
i2c_start_wait(dev+I2C_WRITE);
i2c_write(0x07);
// read
i2c_rep_start(dev+I2C_READ);
data_low = i2c_readAck(); //Read 1 byte and then send ack
data_high = i2c_readAck(); //Read 1 byte and then send ack
pec = i2c_readNak();
i2c_stop();
//This converts high and low bytes together and processes temperature, MSB is a error bit and is ignored for temps
double tempFactor = 0.02; // 0.02 degrees per LSB (measurement resolution of the MLX90614)
double tempData = 0x0000; // zero out the data
int frac; // data past the decimal point
// This masks off the error bit of the high byte, then moves it left 8 bits and adds the low byte.
tempData = (double)(((data_high & 0x007F) << 8) + data_low);
tempData = (tempData * tempFactor)-0.01;
float celcius = tempData - 273.15;
float fahrenheit = (celcius*1.8) + 32;
//Serial.println(fahrenheit);
float state = normf(fahrenheit, lowReading, highReading);
Serial.write((unsigned int) state);
Serial.write(separatorCharacter);
// 165 is blue, 0 is red
BlinkM_fadeToHSB(blinkm_addr, map(state, 0, 255, 165, 0), 255, 255);
}
/*
* BlinkM_funcs.h -- Arduino 'library' to control BlinkM
* --------------
*
*
* Note: original version of this file lives with the BlinkMTester sketch
*
* Note: all the functions are declared 'static' because
* it saves about 1.5 kbyte in code space in final compiled sketch.
* A C++ library of this costs a 1kB more.
*
* 2007-11, Tod E. Kurt, ThingM, http://thingm.com/
*
* version: 20111201
*
* history:
* 20080101 - initial release
* 20080203 - added setStartupParam(), bugfix receiveBytes() from Dan Julio
* 20081101 - fixed to work with Arduino-0012, added MaxM commands,
* added test script read/write functions, cleaned up some functions
* 20090121 - added I2C bus scan functions, has dependencies on private
* functions inside Wire library, so might break in the future
* 20100420 - added BlinkM_startPower and _stopPower
* 20111201 - updated to work with Arduino 1.0 (breaks compatibility with Arduino <= 0023)
*
*/
#include <Arduino.h>
extern "C" {
#include "utility/twi.h" // from Wire library, so we can do bus scanning
}
// format of light script lines: duration, command, arg1,arg2,arg3
typedef struct _blinkm_script_line {
uint8_t dur;
uint8_t cmd[4]; // cmd,arg1,arg2,arg3
} blinkm_script_line;
// Call this first (when powering BlinkM from a power supply)
static void BlinkM_begin()
{
Wire.begin(); // join i2c bus (address optional for master)
}
/*
* actually can't do this either, because twi_init() has THREE callocs in it too
*
static void BlinkM_reset()
{
twi_init(); // can't just call Wire.begin() again because of calloc()s there
}
*/
//
// each call to twi_writeTo() should return 0 if device is there
// or other value (usually 2) if nothing is at that address
//
static void BlinkM_scanI2CBus(byte from, byte to,
void(*callback)(byte add, byte result) )
{
byte rc;
byte data = 0; // not used, just an address to feed to twi_writeTo()
for( byte addr = from; addr <= to; addr++ ) {
rc = twi_writeTo(addr, &data, 0, 1);
callback( addr, rc );
}
}
//
//
static int8_t BlinkM_findFirstI2CDevice()
{
byte rc;
byte data = 0; // not used, just an address to feed to twi_writeTo()
for( byte addr=1; addr < 120; addr++ ) { // only scan addrs 1-120
rc = twi_writeTo(addr, &data, 0, 1);
if( rc == 0 ) return addr; // found an address
}
return -1; // no device found in range given
}
// FIXME: make this more Arduino-like
static void BlinkM_startPowerWithPins(byte pwrpin, byte gndpin)
{
pinMode( pwrpin, OUTPUT);
pinMode( gndpin, OUTPUT);
digitalWrite( pwrpin, HIGH );
digitalWrite( gndpin, LOW );
/*
DDRC |= _BV(pwrpin) | _BV(gndpin); // make outputs
PORTC &=~ _BV(gndpin);
PORTC |= _BV(pwrpin);
*/
}
// FIXME: make this more Arduino-like
static void BlinkM_stopPowerWithPins(byte pwrpin, byte gndpin)
{
//DDRC &=~ (_BV(pwrpin) | _BV(gndpin));
digitalWrite( pwrpin, LOW );
digitalWrite( gndpin, LOW );
}
//
static void BlinkM_startPower()
{
BlinkM_startPowerWithPins( A3, A2 );
}
//
static void BlinkM_stopPower()
{
BlinkM_stopPowerWithPins( A3, A2 );
}
// General version of BlinkM_beginWithPower().
// Call this first when BlinkM is plugged directly into Arduino
static void BlinkM_beginWithPowerPins(byte pwrpin, byte gndpin)
{
BlinkM_startPowerWithPins(pwrpin,gndpin);
delay(100); // wait for things to stabilize
Wire.begin();
}
// Call this first when BlinkM is plugged directly into Arduino
// FIXME: make this more Arduino-like
static void BlinkM_beginWithPower()
{
BlinkM_beginWithPowerPins( A3, A2 );
}
// sends a generic command
static void BlinkM_sendCmd(byte addr, byte* cmd, int cmdlen)
{
Wire.beginTransmission(addr);
for( byte i=0; i<cmdlen; i++)
Wire.write(cmd[i]);
Wire.endTransmission();
}
// receives generic data
// returns 0 on success, and -1 if no data available
// note: responsiblity of caller to know how many bytes to expect
static int BlinkM_receiveBytes(byte addr, byte* resp, byte len)
{
Wire.requestFrom(addr, len);
if( Wire.available() ) {
for( int i=0; i<len; i++)
resp[i] = Wire.read();
return 0;
}
return -1;
}
// Sets the I2C address of the BlinkM.
// Uses "general call" broadcast address
static void BlinkM_setAddress(byte newaddress)
{
Wire.beginTransmission(0x00); // general call (broadcast address)
Wire.write('A');
Wire.write(newaddress);
Wire.write(0xD0);
Wire.write(0x0D); // dood!
Wire.write(newaddress);
Wire.endTransmission();
delay(50); // just in case
}
// Gets the I2C address of the BlinKM
// Kind of redundant when sent to a specific address
// but uses to verify BlinkM communication
static int BlinkM_getAddress(byte addr)
{
Wire.beginTransmission(addr);
Wire.write('a');
Wire.endTransmission();
Wire.requestFrom(addr, (byte)1); // general call
if( Wire.available() ) {
byte b = Wire.read();
return b;
}
return -1;
}
// Gets the BlinkM firmware version
static int BlinkM_getVersion(byte addr)
{
Wire.beginTransmission(addr);
Wire.write('Z');
Wire.endTransmission();
Wire.requestFrom(addr, (byte)2);
if( Wire.available() ) {
byte major_ver = Wire.read();
byte minor_ver = Wire.read();
return (major_ver<<8) + minor_ver;
}
return -1;
}
// Demonstrates how to verify you're talking to a BlinkM
// and that you know its address
static int BlinkM_checkAddress(byte addr)
{
//Serial.print("Checking BlinkM address...");
int b = BlinkM_getAddress(addr);
if( b==-1 ) {
//Serial.println("No response, that's not good");
return -1; // no response
}
//Serial.print("received addr: 0x");
//Serial.print(b,HEX);
if( b != addr )
return 1; // error, addr mismatch
else
return 0; // match, everything okay
}
// Sets the speed of fading between colors.
// Higher numbers means faster fading, 255 == instantaneous fading
static void BlinkM_setFadeSpeed(byte addr, byte fadespeed)
{
Wire.beginTransmission(addr);
Wire.write('f');
Wire.write(fadespeed);
Wire.endTransmission();
}
// Sets the light script playback time adjust
// The timeadj argument is signed, and is an additive value to all
// durations in a light script. Set to zero to turn off time adjust.
static void BlinkM_setTimeAdj(byte addr, byte timeadj)
{
Wire.beginTransmission(addr);
Wire.write('t');
Wire.write(timeadj);
Wire.endTransmission();
}
// Fades to an RGB color
static void BlinkM_fadeToRGB(byte addr, byte red, byte grn, byte blu)
{
Wire.beginTransmission(addr);
Wire.write('c');
Wire.write(red);
Wire.write(grn);
Wire.write(blu);
Wire.endTransmission();
}
// Fades to an HSB color
static void BlinkM_fadeToHSB(byte addr, byte hue, byte saturation, byte brightness)
{
Wire.beginTransmission(addr);
Wire.write('h');
Wire.write(hue);
Wire.write(saturation);
Wire.write(brightness);
Wire.endTransmission();
}
// Sets an RGB color immediately
static void BlinkM_setRGB(byte addr, byte red, byte grn, byte blu)
{
Wire.beginTransmission(addr);
Wire.write('n');
Wire.write(red);
Wire.write(grn);
Wire.write(blu);
Wire.endTransmission();
}
// Fades to a random RGB color
static void BlinkM_fadeToRandomRGB(byte addr, byte rrnd, byte grnd, byte brnd)
{
Wire.beginTransmission(addr);
Wire.write('C');
Wire.write(rrnd);
Wire.write(grnd);
Wire.write(brnd);
Wire.endTransmission();
}
// Fades to a random HSB color
static void BlinkM_fadeToRandomHSB(byte addr, byte hrnd, byte srnd, byte brnd)
{
Wire.beginTransmission(addr);
Wire.write('H');
Wire.write(hrnd);
Wire.write(srnd);
Wire.write(brnd);
Wire.endTransmission();
}
//
static void BlinkM_getRGBColor(byte addr, byte* r, byte* g, byte* b)
{
Wire.beginTransmission(addr);
Wire.write('g');
Wire.endTransmission();
Wire.requestFrom(addr, (byte)3);
if( Wire.available() ) {
*r = Wire.read();
*g = Wire.read();
*b = Wire.read();
}
}
//
static void BlinkM_playScript(byte addr, byte script_id, byte reps, byte pos)
{
Wire.beginTransmission(addr);
Wire.write('p');
Wire.write(script_id);
Wire.write(reps);
Wire.write(pos);
Wire.endTransmission();
}
//
static void BlinkM_stopScript(byte addr)
{
Wire.beginTransmission(addr);
Wire.write('o');
Wire.endTransmission();
}
static void BlinkM_off(uint8_t addr)
{
BlinkM_stopScript( addr );
BlinkM_setFadeSpeed(addr,10);
BlinkM_setRGB(addr, 0,0,0 );
}
//
static void BlinkM_setScriptLengthReps(byte addr, byte script_id,
byte len, byte reps)
{
Wire.beginTransmission(addr);
Wire.write('L');
Wire.write(script_id);
Wire.write(len);
Wire.write(reps);
Wire.endTransmission();
}
// Fill up script_line with data from a script line
// currently only script_id = 0 works (eeprom script)
static void BlinkM_readScriptLine(byte addr, byte script_id,
byte pos, blinkm_script_line* script_line)
{
Wire.beginTransmission(addr);
Wire.write('R');
Wire.write(script_id);
Wire.write(pos);
Wire.endTransmission();
Wire.requestFrom(addr, (byte)5);
while( Wire.available() < 5 ) ; // FIXME: wait until we get 7 bytes
script_line->dur = Wire.read();
script_line->cmd[0] = Wire.read();
script_line->cmd[1] = Wire.read();
script_line->cmd[2] = Wire.read();
script_line->cmd[3] = Wire.read();
}
//
static void BlinkM_writeScriptLine(byte addr, byte script_id,
byte pos, byte dur,
byte cmd, byte arg1, byte arg2, byte arg3)
{
#ifdef BLINKM_FUNCS_DEBUG
Serial.print("writing line:"); Serial.print(pos,DEC);
Serial.print(" with cmd:"); Serial.print(cmd);
Serial.print(" arg1:"); Serial.println(arg1,HEX);
#endif
Wire.beginTransmission(addr);
Wire.write('W');
Wire.write(script_id);
Wire.write(pos);
Wire.write(dur);
Wire.write(cmd);
Wire.write(arg1);
Wire.write(arg2);
Wire.write(arg3);
Wire.endTransmission();
}
//
static void BlinkM_writeScript(byte addr, byte script_id,
byte len, byte reps,
blinkm_script_line* lines)
{
#ifdef BLINKM_FUNCS_DEBUG
Serial.print("writing script to addr:"); Serial.print(addr,DEC);
Serial.print(", script_id:"); Serial.println(script_id,DEC);
#endif
for(byte i=0; i < len; i++) {
blinkm_script_line l = lines[i];
BlinkM_writeScriptLine( addr, script_id, i, l.dur,
l.cmd[0], l.cmd[1], l.cmd[2], l.cmd[3]);
delay(20); // must wait for EEPROM to be programmed
}
BlinkM_setScriptLengthReps(addr, script_id, len, reps);
}
//
static void BlinkM_setStartupParams(byte addr, byte mode, byte script_id,
byte reps, byte fadespeed, byte timeadj)
{
Wire.beginTransmission(addr);
Wire.write('B');
Wire.write(mode); // default 0x01 == Play script
Wire.write(script_id); // default 0x00 == script #0
Wire.write(reps); // default 0x00 == repeat infinitely
Wire.write(fadespeed); // default 0x08 == usually overridden by sketch
Wire.write(timeadj); // default 0x00 == sometimes overridden by sketch
Wire.endTransmission();
}
//
static void BlinkM_setStartupParamsDefault( byte addr )
{
BlinkM_setStartupParams( addr, 0x01, 0x00, 0x00, 0x08, 0x00 );
}
// Gets digital inputs of the BlinkM
// returns -1 on failure
static int BlinkM_getInputsO(byte addr)
{
Wire.beginTransmission(addr);
Wire.write('i');
Wire.endTransmission();
Wire.requestFrom(addr, (byte)1);
if( Wire.available() ) {
byte b = Wire.read();
return b;
}
return -1;
}
// Gets digital inputs of the BlinkM
// stores them in passed in array
// returns -1 on failure
static int BlinkM_getInputs(byte addr, byte inputs[])
{
Wire.beginTransmission(addr);
Wire.write('i');
Wire.endTransmission();
Wire.requestFrom(addr, (byte)4);
while( Wire.available() < 4 ) ; // FIXME: wait until we get 4 bytes
inputs[0] = Wire.read();
inputs[1] = Wire.read();
inputs[2] = Wire.read();
inputs[3] = Wire.read();
return 0;
}
//
static int BlinkM_doFactoryReset()
{
BlinkM_setAddress( 0x09 );
delay(30);
BlinkM_setStartupParamsDefault( 0x09 );
delay(30);
// the example script we're going to write
blinkm_script_line script1_lines[] = {
{ 1, {'f', 10, 00, 00}}, // set fade speed to 10
{ 100, {'c', 0xff,0xff,0xff}}, // white
{ 50, {'c', 0xff,0x00,0x00}}, // red
{ 50, {'c', 0x00,0xff,0x00}}, // green
{ 50, {'c', 0x00,0x00,0xff}}, // blue
{ 50, {'c', 0x00,0x00,0x00}}, // off
};
int script1_len = 6; // number of script lines above
BlinkM_writeScript( 0x09, 0, script1_len, 0, script1_lines);
/*
BlinkMScript script = new BlinkMScript();
script.add( new BlinkMScriptLine( 1, 'f', 10, 0, 0) );
script.add( new BlinkMScriptLine(100, 'c', 0xff,0xff,0xff) );
script.add( new BlinkMScriptLine( 50, 'c', 0xff,0x00,0x00) );
script.add( new BlinkMScriptLine( 50, 'c', 0x00,0xff,0x00) );
script.add( new BlinkMScriptLine( 50, 'c', 0x00,0x00,0xff) );
script.add( new BlinkMScriptLine( 50, 'c', 0x00,0x00,0x00) );
for( int i=0; i< 49-6; i++ ) { // FIXME: make this length correct
script.add( new BlinkMScriptLine( 0, 'c', 0,0,0 ) );
}
writeScript( addr, script);
*/
}
int recent = 10;
int minAdapt = 2;
int maxAdapt = 100;
class Graph extends ArrayList {
float maxValue, minValue;
boolean watching;
Graph(boolean watching) {
this.watching = watching;
this.maxValue = Float.NEGATIVE_INFINITY;
this.minValue = Float.POSITIVE_INFINITY;
}
void add(float value) {
if(watching) {
if(value == Float.NEGATIVE_INFINITY ||
value == Float.POSITIVE_INFINITY ||
value != value)
return;
if(value > maxValue)
maxValue = value;
if(value < minValue)
minValue = value;
}
super.add(value);
}
float getFloat(int i) {
if(size() == 0)
return 0;
return ((Float) super.get(i)).floatValue();
}
float getLastFloat() {
return getFloat(size() - 1);
}
float normalize(float x) {
return constrain(norm(x, minValue, maxValue), 0, 1);
}
float getNorm(int i) {
return normalize(getFloat(i));
}
float getLastNorm() {
return getNorm(size() - 1);
}
float getLinear(int i) {
return sqrt(1. / getNorm(i));
}
float getLastLinear() {
return getLinear(size() - 1);
}
float mean() {
float sum = 0;
for(int i = 0; i < size(); i++)
sum += getFloat(i);
return sum / size();
}
float recentMean() {
float mean = 0;
int n = min(size(), recent);
for(int i = 0; i < n; i++)
mean += getFloat(size() - i - 1);
return mean / n;
}
float recentVarianceWeighted() {
float mean = recentMean();
float recentVariance = 0;
int n = min(size(), recent);
float weights = 0;
for(int i = 0; i < n; i++) {
float w = 1. - ((float) i / (float) n);
recentVariance += abs(getFloat(size() - i - 1) - mean) * w;
weights += w;
}
return recentVariance / weights;
}
float recentAdaptive(float adapt) {
float sum = 0;
float weights = 0;
float curRecent = map(adapt, 0, 1, minAdapt, maxAdapt);
int n = min(size(), 1 + (int) curRecent);
for(int i = 0; i < n; i++) {
float w = 1. - ((float) i / (float) n);
sum += getFloat(size() - i - 1) * w;
weights += w;
}
println(sum + " " + weights + " " + n);
return sum / weights;
}
void draw(int width, int height) {
fill(getNorm(size() - 1) * 255);
//rect(0, 0, width, height);
fill(0);
stroke(0);
textAlign(LEFT, CENTER);
text(nf(getLastFloat(), 0, 0), 10, height - normalize(recentMean()) * height);
textAlign(LEFT, TOP);
text(nf(minValue, 0, 0), 200, height - 20);
//noFill();
fill(0, 128);
beginShape();
vertex(0, height);
for(int i = 0; i < width && i < size(); i++) {
int position = size() - i - 1;
vertex(i, height - getNorm(position) * height);
}
vertex(width, height);
endShape();
fill(0);
textAlign(LEFT, BOTTOM);
text(nf(maxValue, 0, 0), 200, 20);
}
void save(String filename) {
String[] out = new String[size()];
for(int i = 0; i < size(); i++)
out[i] = nf(getFloat(i), 0, 0);
saveStrings(filename + ".csv", out);
}
}
import processing.serial.*;
PFont font;
int curSensor = 0;
boolean ready = false;
int totalSensors = 1;
int baudRate = 9600; //115200
float lowReading = 75;
float highReading = 110;
Serial serial;
Vector<Graph> graphs;
void setup() {
size(1280, 720, P2D);
font = createFont("", 18);
textFont(font);
//textMode(SCREEN);
serial = new Serial(this, Serial.list()[0], baudRate);
reset();
}
void reset() {
graphs = new Vector<Graph>();
for(int i = 0; i < totalSensors; i++) {
graphs.add(new Graph(true));
}
}
void draw() {
background(255);
readData();
for(int i = 0; i < totalSensors; i++) {
graphs.get(i).draw(width, height / totalSensors);
translate(0, height / totalSensors);
}
}
int separatorCharacter = 255;
void readData() {
while(serial.available() > 0) {
int in = (int) serial.readChar();
if(in == separatorCharacter) {
ready = true;
curSensor = 0;
} else {
if(!pause && ready && curSensor < totalSensors) {
graphs.get(curSensor).add(map(in, 0, 255, lowReading, highReading));
}
curSensor++;
}
}
}
void mousePressed() {
reset();
}
boolean pause = false;
String prefix = "boots";
void keyPressed() {
if(key == 's') {
for(int i = 0; i < graphs.size(); i++) {
graphs.get(i).save(prefix + "-" + i);
}
}
if(key == 'p') {
pause = !pause;
}
}
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment