/* * Color Light Calibration * Gogh Bike by Ben Cohen, Ryan Greenberg, Joyce Tsai * * Connect a photosensor under different colored films to analog inputs 0, 1, 2. * Take calibration values when instructed by user via serial commands and store in Arduino EEPROM. * These values are used by color_light_sensing. */ #include // EEPROM Read/Write helper from http://www.arduino.cc/playground/Code/EEPROMWriteAnything template int EEPROM_writeAnything(int ee, const T& value) { const byte* p = (const byte*)(const void*)&value; int i; for (i = 0; i < sizeof(value); i++) EEPROM.write(ee++, *p++); return i; } template int EEPROM_readAnything(int ee, T& value) { byte* p = (byte*)(void*)&value; int i; for (i = 0; i < sizeof(value); i++) *p++ = EEPROM.read(ee++); return i; } int redPin = 1; int yellowPin = 2; int bluePin = 0; int photoPins[] = {0, 1, 2}; // Arduino pins that photosensors are connected to const int numPins = sizeof(photoPins) / sizeof(int); // Number of photosensors connected to Arduino const int sampleWindow = 10; // Number of readings to use to evaluate the high or low value for a sensor int sampleDelay = 50; // ms to delay between sample readings int sensingDelay = 500; // ms to delay between readings to determine pour color. Used to handle macro jitter int vals[numPins] = {0,0,0}; // Single reading from each sensor float current[numPins] = {0,0,0}; // Averaged reading from each sensor using sampleWindow and sampleDelay int photoStates[numPins] = {0,0,0}; // Current HIGH (1) or LOW (0) state for a given photosensor int sampleVals[numPins * sampleWindow]; // Sample of readings from each sensor determined by sampleWindow and numPins // Calibration values for each sensor float lows[numPins] = {0,0,0}; float whites[numPins] = {0,0,0}; float reds[numPins] = {0,0,0}; float yellows[numPins] = {0,0,0}; float blues[numPins] = {0,0,0}; float ranges[numPins] = {0,0,0}; float thresholds[numPins] = {0,0,0}; // Direction to scale the initial reading when creating a threshold. If the initial // reading is taken in a LOW state (no light), this should be 1, indicating that the // threshold is created in a positive direciton. If the initial reading is in a HIGH // state (light activated), this should be -1, indicating that the threshold is created // in a negative direction. float thresholdScale = .10; // pecentage to use when scaling initial reading to set threshold level // TODO It might be useful to use map for reading test values // http://arduino.cc/en/Reference/Map void setup() { Serial.begin(9600); // Do an interactive calibration to set photosensor values for different light levels Serial.println("Interactive Calibration\n\n"); // LOW/AMBIENT LIGHT // Set low light value for sensor Serial.println("Recording low/ambient light values..."); samplePhotoSensors(sampleVals); photoSensorAverages(lows, sampleVals); // WHITE LIGHT // Set white light value for sensor Serial.println("Send serial to set white light values..."); waitForSerialConfirmation(); Serial.println("Recording white light values..."); samplePhotoSensors(sampleVals); photoSensorAverages(whites, sampleVals); // RED LIGHT // Set red light value for sensor Serial.println("Send serial to set red light values..."); waitForSerialConfirmation(); Serial.println("Recording red light values..."); samplePhotoSensors(sampleVals); photoSensorAverages(reds, sampleVals); // YELLOW LIGHT // Set yellow light value for sensor Serial.println("Send serial to set yellow light values..."); waitForSerialConfirmation(); Serial.println("Recording yellow light values..."); samplePhotoSensors(sampleVals); photoSensorAverages(yellows, sampleVals); // BLUE LIGHT // Set blue light value for sensor Serial.println("Send serial to set blue light values..."); waitForSerialConfirmation(); Serial.println("Recording blue light values..."); samplePhotoSensors(sampleVals); photoSensorAverages(blues, sampleVals); // Make sure that all calibration values were recorded properly Serial.println("CALIBRATION VALUES"); Serial.println("\t0\t1\t2"); // TODO It would be nice to have an array of pointers to the color to make this less verbose //float *colors[5] = { &lows, &whites, &reds, &yellows, &blues }; // The post calibration output should look something like this: // 0 1 2 // LOW 412 532 800 // WHITE 412 532 800 // RED 412 532 800 // YELLOW 412 532 800 // BLUE 412 532 800 Serial.print("LOW\t"); for (int i = 0; i < numPins; i++) { Serial.print(lows[i]); Serial.print("\t"); } Serial.println(); Serial.print("WHITE\t"); for (int i = 0; i < numPins; i++) { Serial.print(whites[i]); Serial.print("\t"); } Serial.println(); Serial.print("RED\t"); for (int i = 0; i < numPins; i++) { Serial.print(reds[i]); Serial.print("\t"); } Serial.println(); Serial.print("YELLOW\t"); for (int i = 0; i < numPins; i++) { Serial.print(yellows[i]); Serial.print("\t"); } Serial.println(); Serial.print("BLUE\t"); for (int i = 0; i < numPins; i++) { Serial.print(blues[i]); Serial.print("\t"); } Serial.println(); // Set threshold and range values for (int i = 0; i < numPins; i++) { ranges[i] = abs(whites[i] - lows[i]); thresholds[i] = abs(whites[i] - lows[i]) * thresholdScale; } Serial.println("Calibration complete. Send serial to copy calibration data to EEPROM."); waitForSerialConfirmation(); saveToEEPROM(); Serial.println("Data copied to EEPROM"); } void loop() { } // Take a series of readings from each photosensor determined by numPins and sampleWindow void samplePhotoSensors (int *readings) { for (int n = 0; n < sampleWindow * numPins;) { for (int i = 0; i < numPins; i++) { readings[n] = analogRead(photoPins[i]); n++; } delay(sampleDelay); } // Serial.println("sampled sensors: "); // for (int n = 0; n < sampleWindow * numPins; n++) { // Serial.println(readings[n]); // } } void photoSensorAverages(float *destination, int *readings) { for (int i = 0; i < numPins; i++) { // Serial.print("Average for pin "); // Serial.println(i); // destination[i] = average(readings, i); } } // Returns the average for a specified pin in a set of readings // For three photosensors using a 3 reading sample window, the // readings array looks like this: // [300,525,600,315,545,580,290,530,605] // 1 2 3 1 2 3 1 2 3 float average (int *readings, int pin) { int sum = 0; for (int i = 0; i < numPins * sampleWindow; i++) { if (i % numPins == pin) { // i = 0 corresponds to pin 0 // E.g. 0 % 3 = 0 // Serial.print("\t"); // Serial.println(readings[i]); sum += readings[i]; } } // Serial.print("AVERAGE:\t"); // Serial.println((float) sum / sampleWindow); return ((float) sum / sampleWindow); } // Loops until any data is sent via serial void waitForSerialConfirmation() { Serial.flush(); while(Serial.available() == 0) { } } // Write values for lows, whites, reds, yellows, and blues to EEPROM void saveToEEPROM() { Serial.println("saving..."); int start; start = EEPROM_writeAnything(0, 'C'); // Calibrated start += EEPROM_writeAnything(start, lows); start += EEPROM_writeAnything(start, whites); start += EEPROM_writeAnything(start, reds); start += EEPROM_writeAnything(start, yellows); start += EEPROM_writeAnything(start, blues); Serial.print("Done! Wrote "); Serial.print(start); Serial.println(" bytes."); }