diff --git a/UI/assets/css/main.css b/UI/assets/css/main.css index 7002140..c6b12b9 100644 --- a/UI/assets/css/main.css +++ b/UI/assets/css/main.css @@ -66,12 +66,14 @@ input::-moz-focus-inner { padding: 0; } +/* input, select, textarea { -moz-appearance: none; -webkit-appearance: none; -ms-appearance: none; appearance: none; } +*/ /* Colummn */ diff --git a/UI/index.html b/UI/index.html index cab8a8c..837624d 100644 --- a/UI/index.html +++ b/UI/index.html @@ -80,6 +80,30 @@ +
+
Enable compression:
+
+ +
+
+
+
Configuration:
+
+ +
+
+
+
Compression RPM:
+
+ +
+
+

diff --git a/UI/renderer.js b/UI/renderer.js index aa56ede..a06d17b 100644 --- a/UI/renderer.js +++ b/UI/renderer.js @@ -6,7 +6,7 @@ const ByteLengthParser = require('@serialport/parser-byte-length') const {ipcRenderer} = require("electron") var port = new serialport('/dev/tty-usbserial1', { autoOpen: false }) -var CONFIG_SIZE = 13; +var CONFIG_SIZE = 15; var onConnectIntervalConfig; var onConnectIntervalWheels; var isConnected=false; @@ -215,13 +215,16 @@ function requestConfig() function receiveConfig(data) { console.log("Received config: " + data); - console.log("Mode: " + data[0]); - - document.getElementById("rpmSelect").value = data[4]; - document.getElementById("fixedRPM").value = (((data[6] & 0xff) << 8) | (data[5] & 0xff)); - document.getElementById("rpmSweepMin").value = (((data[8] & 0xff) << 8) | (data[7] & 0xff)); - document.getElementById("rpmSweepMax").value = (((data[10] & 0xff) << 8) | (data[9] & 0xff)); - document.getElementById("rpmSweepSpeed").value = (((data[12] & 0xff) << 8) | (data[11] & 0xff)); + console.log("Mode: " + data[2]); + + document.getElementById("rpmSelect").value = data[2]; + document.getElementById("fixedRPM").value = (((data[4] & 0xff) << 8) | (data[3] & 0xff)); + document.getElementById("rpmSweepMin").value = (((data[6] & 0xff) << 8) | (data[5] & 0xff)); + document.getElementById("rpmSweepMax").value = (((data[8] & 0xff) << 8) | (data[7] & 0xff)); + document.getElementById("rpmSweepSpeed").value = (((data[10] & 0xff) << 8) | (data[9] & 0xff)); + document.getElementById("compressionEnable").value = data[11]; + document.getElementById("compressionMode").value = data[12]; + document.getElementById("compressionRPM").value = (((data[14] & 0xff) << 8) | (data[13] & 0xff)); port.unpipe(); @@ -623,6 +626,7 @@ window.onload = function () refreshSerialPorts(); redrawGears(toothPatterns[0]); window.location.hash = '#connect'; + //window.location.hash = '#live'; checkForUpdates(); //animateGauges(); diff --git a/ardustim/ardustim/ardustim.h b/ardustim/ardustim/ardustim.h index 97dc509..292a95d 100644 --- a/ardustim/ardustim/ardustim.h +++ b/ardustim/ardustim/ardustim.h @@ -29,6 +29,8 @@ void reset_new_OCR1A(uint32_t); uint8_t get_bitshift_from_prescaler(uint8_t *); void get_prescaler_bits(uint32_t *, uint8_t *, uint8_t *); void setRPM(uint16_t); +uint16_t calculateCompressionModifier(); +uint16_t calculateCurrentCrankAngle(); /* Prototypes */ diff --git a/ardustim/ardustim/ardustim.ino b/ardustim/ardustim/ardustim.ino index 101c6d1..b10bc09 100644 --- a/ardustim/ardustim/ardustim.ino +++ b/ardustim/ardustim/ardustim.ino @@ -29,6 +29,7 @@ #include struct configTable config; +struct status currentStatus; /* Sensistive stuff used in ISR's */ volatile uint16_t adc0; /* POT RPM */ @@ -39,12 +40,13 @@ volatile uint8_t analog_port = 0; volatile bool adc0_read_complete = false; volatile bool adc1_read_complete = false; volatile bool reset_prescaler = false; -volatile bool normal = true; volatile uint8_t output_invert_mask = 0x00; /* Don't invert anything */ volatile uint8_t prescaler_bits = 0; volatile uint8_t last_prescaler_bits = 0; volatile uint16_t new_OCR1A = 5000; /* sane default */ volatile uint16_t edge_counter = 0; +volatile uint32_t cycleStartTime = micros(); +volatile uint32_t cycleDuration = 0; uint32_t sweep_time_counter = 0; uint8_t sweep_direction = ASCENDING; @@ -213,7 +215,7 @@ void setup() { // Set ADSC in ADCSRA (0x7A) to start the ADC conversion ADCSRA |= B01000000; /* Make sure we are using the DEFAULT RPM on startup */ - reset_new_OCR1A(config.rpm); + reset_new_OCR1A(currentStatus.rpm); } // End setup @@ -252,22 +254,17 @@ ISR(ADC_vect){ /* Pumps the pattern out of flash to the port * The rate at which this runs is dependent on what OCR1A is set to */ -ISR(TIMER1_COMPA_vect) { +ISR(TIMER1_COMPA_vect) +{ /* This is VERY simple, just walk the array and wrap when we hit the limit */ PORTB = output_invert_mask ^ pgm_read_byte(&Wheels[config.wheel].edge_states_ptr[edge_counter]); /* Write it to the port */ - /* Normal direction overflow handling */ - if (normal) - { - edge_counter++; - if (edge_counter == Wheels[config.wheel].wheel_max_edges) { - edge_counter = 0; - } - } - else + + edge_counter++; + if (edge_counter == Wheels[config.wheel].wheel_max_edges) { - if (edge_counter == 0) - edge_counter = Wheels[config.wheel].wheel_max_edges; - edge_counter--; + edge_counter = 0; + cycleDuration = micros() - cycleStartTime; + cycleStartTime = micros(); } /* The tables are in flash so we need pgm_read_byte() */ @@ -301,18 +298,17 @@ void loop() } else if (config.mode == LINEAR_SWEPT_RPM) { - //if(millis() > (sweep_time_counter + sweep_interval_ms)) if(micros() > (sweep_time_counter + config.sweep_interval)) { sweep_time_counter = micros(); if(sweep_direction == ASCENDING) { - tmp_rpm = config.rpm + 1; + tmp_rpm = currentStatus.rpm + 1; if(tmp_rpm >= config.sweep_high_rpm) { sweep_direction = DESCENDING; } } else { - tmp_rpm = config.rpm - 1; + tmp_rpm = currentStatus.rpm - 1; if(tmp_rpm <= config.sweep_low_rpm) { sweep_direction = ASCENDING; } } } @@ -321,7 +317,55 @@ void loop() { tmp_rpm = config.fixed_rpm; } - setRPM(tmp_rpm); + currentStatus.base_rpm = tmp_rpm; + currentStatus.compressionModifier = calculateCompressionModifier(); + + if(currentStatus.compressionModifier >= currentStatus.base_rpm ) { currentStatus.compressionModifier = 0; } + setRPM( (currentStatus.base_rpm - currentStatus.compressionModifier) ); +} + +uint16_t calculateCompressionModifier() +{ + if( (currentStatus.rpm > config.compressionRPM) || (config.useCompression != true) ) { return 0; } + //if( currentStatus.base_rpm > 400 ) { return 0;} + + uint16_t crankAngle = calculateCurrentCrankAngle(); + uint16_t modAngle = crankAngle; + + uint16_t compressionModifier = 0; + switch(config.compressionType) + { + case COMPRESSION_TYPE_2CYL_4STROKE: + modAngle = modAngle / 2; + compressionModifier = pgm_read_byte(&sin_100_180[modAngle]); + case COMPRESSION_TYPE_4CYL_4STROKE: + modAngle = (crankAngle % 180) ; + compressionModifier = pgm_read_byte(&sin_100_180[modAngle]); + break; + case COMPRESSION_TYPE_6CYL_4STROKE: + modAngle = crankAngle % 120; + compressionModifier = pgm_read_byte(&sin_100_120[modAngle]); + break; + case COMPRESSION_TYPE_8CYL_4STROKE: + modAngle = crankAngle % 90; + compressionModifier = pgm_read_byte(&sin_100_90[modAngle]); + break; + } + + return compressionModifier; +} + +uint16_t calculateCurrentCrankAngle() +{ + if(cycleDuration == 0) { return 0; } + + uint32_t cycleTime = micros() - cycleStartTime; + if( pgm_read_byte(&Wheels[config.wheel].wheel_degrees) == 720 ) { cycleTime = cycleTime / 2; } + + uint16_t tmpCrankAngle = ((cycleTime * 360U) / cycleDuration); + while(tmpCrankAngle > 360) { tmpCrankAngle -= 360; } + + return tmpCrankAngle; } /*! @@ -331,8 +375,8 @@ void setRPM(uint16_t newRPM) { if (newRPM < 10) { return; } - if(config.rpm != newRPM) { reset_new_OCR1A(newRPM); } - config.rpm = newRPM; + if(currentStatus.rpm != newRPM) { reset_new_OCR1A( newRPM ); } + currentStatus.rpm = newRPM; } diff --git a/ardustim/ardustim/comms.cpp b/ardustim/ardustim/comms.cpp index 3020ce0..d37e026 100644 --- a/ardustim/ardustim/comms.cpp +++ b/ardustim/ardustim/comms.cpp @@ -132,7 +132,7 @@ void commandParser() break; case 'R': //Send the current RPM - Serial.println(config.rpm); + Serial.println(currentStatus.rpm); break; case 's': //Set the high and low RPM for sweep mode @@ -199,7 +199,7 @@ void toggle_invert_secondary_cb() void display_new_wheel() { - reset_new_OCR1A(config.rpm); + reset_new_OCR1A(currentStatus.rpm); edge_counter = 0; // Reset to beginning of the wheel pattern */ } diff --git a/ardustim/ardustim/globals.h b/ardustim/ardustim/globals.h index f8eb1c0..cd812fc 100644 --- a/ardustim/ardustim/globals.h +++ b/ardustim/ardustim/globals.h @@ -28,18 +28,37 @@ #define TMP_RPM_CAP 9000 /* MAX RPM via pot control. Adjusted to 9,000rpm max from 16,384rpm to match the GUI */ #define EEPROM_LAST_MODE 100 -struct configTable { - uint8_t version; - uint16_t rpm = 6000; +#define COMPRESSION_TYPE_1CYL_4STROKE 0 //Not initiallity supported +#define COMPRESSION_TYPE_2CYL_4STROKE 1 +#define COMPRESSION_TYPE_3CYL_4STROKE 2 //Not initiallity supported +#define COMPRESSION_TYPE_4CYL_4STROKE 3 +#define COMPRESSION_TYPE_6CYL_4STROKE 4 +#define COMPRESSION_TYPE_8CYL_4STROKE 5 + +struct configTable +{ + uint8_t version; uint8_t wheel = FOUR_TWENTY_A; uint8_t mode; uint16_t fixed_rpm = 2500; uint16_t sweep_low_rpm = 250; uint16_t sweep_high_rpm = 4000; uint16_t sweep_interval = 1000; + + bool useCompression = false; + uint8_t compressionType = 0; + uint16_t compressionRPM = 400; }; extern struct configTable config; +struct status +{ + uint16_t base_rpm; //RPM excluding compression modifier + uint16_t compressionModifier; + uint16_t rpm; //Final RPM +}; +extern struct status currentStatus; + /* Tie things wheel related into one nicer structure ... */ typedef struct _wheels wheels; struct _wheels { @@ -51,4 +70,47 @@ struct _wheels { const uint16_t wheel_degrees; }; +//A sin wave of amplitude 100 with a complete cycle in 180 degrees (1 entry per degree). +const uint8_t sin_100_180[] PROGMEM = +{ + 0,0,0,0,0,1,1,1,2,2,3,4,4,5,6,7,8,9,10,11, + 12,13,14,15,17,18,19,21,22,24,25,27,28,30, + 31,33,35,36,38,40,41,43,45,47,48,50,52,53, + 55,57,59,60,62,64,65,67,69,70,72,73,75,76, + 78,79,81,82,83,85,86,87,88,89,90,91,92,93, + 94,95,96,96,97,98,98,99,99,99,100,100,100, + 100,100,100,100,100,100,99,99,99,98,98,97, + 96,96,95,94,93,92,91,90,89,88,87,86,85,83, + 82,81,79,78,76,75,73,72,70,69,67,65,64,62, + 60,59,57,55,53,52,50,48,47,45,43,41,40,38, + 36,35,33,31,30,28,27,25,24,22,21,19,18,17, + 15,14,13,12,11,10,9,8,7,6,5,4,4,3,2,2,1,1, + 1,0,0,0,0 +}; + +//A sin wave of amplitude 100 with a complete cycle in 90 degrees (1 entry per degree). +const uint8_t sin_100_90[] PROGMEM = +{ + 0,0,0,1,2,3,4,6,8,10,12,14,17,19,22,25,28, + 31,35,38,41,45,48,52,55,59,62,65,69,72,75, + 78,81,83,86,88,90,92,94,96,97,98,99,100,100, + 100,100,100,99,98,97,96,94,92,90,88,86,83,81, + 78,75,72,69,65,62,59,55,52,48,45,41,38,35,31, + 28,25,22,19,17,14,12,10,8,6,4,3,2,1,0,0 +}; + +//A sin wave of amplitude 100 with a complete cycle in 120 degrees + +const uint8_t sin_100_120[] PROGMEM = +{ + 0,0,0,1,1,2,2,3,4,5,7,8,10,11,13,15,17,19, + 21,23,25,27,30,32,35,37,40,42,45,47,50,53, + 55,58,60,63,65,68,70,73,75,77,79,81,83,85, + 87,89,90,92,93,95,96,97,98,98,99,99,100,100, + 100,100,100,99,99,98,98,97,96,95,93,92,90,89, + 87,85,83,81,79,77,75,73,70,68,65,63,60,58,55, + 53,50,47,45,42,40,37,35,32,30,27,25,23,21,19, + 17,15,13,11,10,8,7,5,4,3,2,2,1,1,0,0 +}; + #endif diff --git a/ardustim/ardustim/storage.h b/ardustim/ardustim/storage.h index f5880ed..0a8fecc 100644 --- a/ardustim/ardustim/storage.h +++ b/ardustim/ardustim/storage.h @@ -9,6 +9,9 @@ #define EEPROM_SWEEP_RPM_MAX 8 //Note this is 2 bytes #define EEPROM_SWEEP_RPM_INT 10 //Note this is 2 bytes #define EEPROM_FIXED_RPM 12 //Note this is 2 bytes +#define EEPROM_USE_COMPRESSION 14 +#define EEPROM_COMPRESSION_TYPE 15 +#define EEPROM_COMPRESSION_RPM 16 //Note this is 2 bytes void loadConfig(); void saveConfig(); diff --git a/ardustim/ardustim/storage.ino b/ardustim/ardustim/storage.ino index 7ab964f..a4799b6 100644 --- a/ardustim/ardustim/storage.ino +++ b/ardustim/ardustim/storage.ino @@ -11,7 +11,7 @@ void loadConfig() { //New arduino config.wheel = 5; //36-1 - config.rpm = 3000; + currentStatus.rpm = 3000; config.mode = POT_RPM; config.fixed_rpm = 3500; @@ -28,7 +28,7 @@ void loadConfig() byte highByte = EEPROM.read(EEPROM_CURRENT_RPM); byte lowByte = EEPROM.read(EEPROM_CURRENT_RPM+1); - config.rpm = word(highByte, lowByte); + currentStatus.rpm = word(highByte, lowByte); highByte = EEPROM.read(EEPROM_FIXED_RPM); lowByte = EEPROM.read(EEPROM_FIXED_RPM+1); @@ -52,10 +52,19 @@ void loadConfig() if(config.sweep_low_rpm >= config.sweep_high_rpm) { config.sweep_low_rpm = config.sweep_high_rpm - 100; } + config.useCompression = EEPROM.read(EEPROM_USE_COMPRESSION); + config.compressionType = EEPROM.read(EEPROM_COMPRESSION_TYPE); + highByte = EEPROM.read(EEPROM_COMPRESSION_RPM); + lowByte = EEPROM.read(EEPROM_COMPRESSION_RPM+1); + config.compressionRPM = word(highByte, lowByte); + //config.compressionType = COMPRESSION_TYPE_6CYL_4STROKE; + //Error checking if(config.wheel >= MAX_WHEELS) { config.wheel = 5; } if(config.mode >= MAX_MODES) { config.mode = FIXED_RPM; } - if(config.rpm > 15000) { config.rpm = 4000; } + if(currentStatus.rpm > 15000) { currentStatus.rpm = 4000; } + if(config.compressionType > COMPRESSION_TYPE_8CYL_4STROKE) { config.compressionType = COMPRESSION_TYPE_4CYL_4STROKE; } + if(config.compressionRPM > 1000) { config.compressionRPM = 400; } } } @@ -65,8 +74,8 @@ void saveConfig() EEPROM.update(EEPROM_RPM_MODE, config.mode); EEPROM.update(EEPROM_VERSION, EEPROM_CURRENT_VERSION); - byte highByte = highByte(config.rpm); - byte lowByte = lowByte(config.rpm); + byte highByte = highByte(currentStatus.rpm); + byte lowByte = lowByte(currentStatus.rpm); EEPROM.update(EEPROM_CURRENT_RPM, highByte); EEPROM.update(EEPROM_CURRENT_RPM+1, lowByte); @@ -90,4 +99,10 @@ void saveConfig() EEPROM.update(EEPROM_SWEEP_RPM_INT, highByte); EEPROM.update(EEPROM_SWEEP_RPM_INT+1, lowByte); + EEPROM.update(EEPROM_USE_COMPRESSION, config.useCompression); + EEPROM.update(EEPROM_COMPRESSION_TYPE, config.compressionType); + highByte = highByte(config.compressionRPM); + lowByte = lowByte(config.compressionRPM); + EEPROM.update(EEPROM_COMPRESSION_RPM, highByte); + EEPROM.update(EEPROM_COMPRESSION_RPM+1, lowByte); } \ No newline at end of file