When using software SW1 (efi_davide_20160131_v1), it is possible to request information to the ECU in real time, using the serial communication provided by the Arduino Micro Serial1 port.
SW1 supports 8 bytes ASCII format commands (from PC, service tool, or logger), and they are interpreted by the following function, which is called every time 8 characters are received on the serial communication.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 |
int evaluate_parameter_read_writing_request() { int temp_num=0; int sum_rpm=0; int sum_throttle=0; int sum_value=0; int i=0; temp_num=serial_inbyte[1]-'0'; sum_rpm+=10*temp_num; temp_num=serial_inbyte[2]-'0'; sum_rpm+=temp_num; temp_num=serial_inbyte[3]-'0'; sum_throttle+=10*temp_num; temp_num=serial_inbyte[4]-'0'; sum_throttle+=temp_num; temp_num=serial_inbyte[5]-'0'; sum_value+=100*temp_num; temp_num=serial_inbyte[6]-'0'; sum_value+=10*temp_num; temp_num=serial_inbyte[7]-'0'; sum_value+=temp_num; if(sum_rpm>=0 && sum_rpm<16){ if(sum_throttle>=0 && sum_throttle<16){ String singleMessageString = ""; if(sum_value>=0 && sum_value<MAX_PERCENTAGE_INJ){ if (serial_inbyte[0]=='w'){ incrementi_rpm[sum_rpm]=sum_value; EEPROM.write(sum_rpm, sum_value); //for redundancy EEPROM.write(sum_rpm+16, sum_value+16); //for redundancy sum_value=EEPROM.read(sum_rpm); singleMessageString+="w"; if (sum_rpm<10) singleMessageString+="0"; singleMessageString+=sum_rpm; singleMessageString+="00"; if (sum_value<10) singleMessageString+="0"; if (sum_value<100) singleMessageString+="0"; singleMessageString+=sum_value; Serial1.print(singleMessageString); // manda fuori il valore appena scritto in EEPROM return 0; } if (serial_inbyte[0]=='r' && sum_value==0){ singleMessageString+="r"; if (sum_rpm<10) singleMessageString+="0"; singleMessageString+=sum_rpm; singleMessageString+="00"; sum_value=incrementi_rpm[sum_rpm]; if (sum_value<10) singleMessageString+="0"; if (sum_value<100) singleMessageString+="0"; singleMessageString+=sum_value; Serial1.print(singleMessageString); // manda fuori il valore appena letto da EEPROM return 0; } } } } if(sum_rpm==16 && sum_throttle==16 && sum_value==0 && serial_inbyte[0]=='r'){ //r1616000 = read all values EEPROM_read(); return 0; } if(sum_rpm==16 && sum_throttle==16 && sum_value==1 && serial_inbyte[0]=='r'){ //r1616001 = EEPROM check if (EEPROM_check()){ // =1 means OK Serial1.print("r1616000"); //EEPROM OK }else{ // =0 means NG (EEPROM has been rewritten to standard values) Serial1.print("w1616000"); //EEPROM ERROR, write was needed and performed } return 0; } if(sum_rpm==16 && sum_throttle==16 && sum_value==0 && serial_inbyte[0]=='w'){ //w1616000 = write standard values EEPROM_write_standard_values(); Serial1.print("w1616000"); //write was performed return 0; } if(sum_rpm==16 && sum_throttle==16 && sum_value==0 && serial_inbyte[0]=='d'){ //d1616000 = read current data shapshot data_send_req=1; // requires the loop to send the data return 0; } return 1; } |
The supported ASCII commands are written below:
- "rxxyy000": reads the injection correction parameter inside the ECU 2D map at rpm position "xx" (0-15 range) and throttle position "yy" (0-15 range). At the moment, instead of a 2D map, a 1D array is used, and the injection time correcting parameter value depends only to the rpm; for this reason, any value can be put as "yy" (from 0 to 15). When the ECU receives this command, it reads the increment value inside the RAM memory and outputs an 8 byte ASCII reply in the format "rxxyyzzz", in which "zzz" is the value of increment at rpm "xx". Example: in case it is asked the value at 3000rpm, and the increment is 20% at 3000rpm, the request will be "r0300000", and the reply will be "r0300020".
- "r1616000": when the ECU receives this command, it outputs the complete info inside the array, in the following order: "r0000xyz", "r0100xyz", "r0200xyz", ..., "r1500xyz", in which "xyz" is the increment value for each array component, in percentage.
- "r1616001": performs EEPROM check, to verify if there is any corruption inside the data written in EEPROM memory. In case any error is found, the standard values are written in EEPROM.
- "w1616000": used to write standard values (array incrementi_rpm_std[]) in the EEPROM.
- "wxxyyzzz": it writes the increment value "zzz" on rpm index "xx" (0-15 range) and throttle index "yy" (0-15 range) of the 2D map. The value is also saved in EEPROM.
In addition, a special command is described below. This command is "d1616000" and it is used to read the real time data snapshot from the ECU. When the ECU receives this command, it replies with a series of 16 bytes, which have the following structure:
- Byte 0. Constant value "d" (data).
- Byte 1. Constant value binary "14" (the following data are 14 bytes)
- Byte 2-5. "delta_time_buffer" variable. It is the time, in microseconds, between one injection and the following (2 crankshaft rotations). These 4 bytes have little endian format.
- Byte 6-9. "delta_inj_buffer" variable. It is the time, in microseconds, of the injection commanded by the Original Honda ECU. These 4 bytes have little endian format.
- Byte 10-11. "throttle_buffer" variable. It is the throttle position sensor value acquired by the ADC (0-1023). These 2 bytes have little endian format.
- Byte 12-15. "crc_somma" variable. It is the CRC check value to determine if previous data can be considered correct or not (there might have been any data corruption due to noise in serial communication, for example). It is the sum of the previous data fields. These 4 bytes have little endian format.
This command can be used to log data continuously when the motorcycle is running, for example by polling this command once every 100ms. The logged data can be then analyzed using other tools, such as Office Excel or Mathworks Matlab.
Hi, I'm ibrahim from Turkey .I'm interested in motorcycles and programmable cards. I've seen your article now and I wonder what the result would be if we used a microcontroller from the pic family instead of arduino. I'd be happy if you'd come back.
i want to read ecu data from my bike yamaha yzf-250 / R25, can u teach me? read to arduino and send to android via bluetooth
David can u please send me files that I can send to have a PCB made I tried the files I can get here n there but they don't seam to work for some reason please help I'll purchase a board from u I only need it to run don't want to piggyback it off another ecu