-
Notifications
You must be signed in to change notification settings - Fork 192
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
MLX90640_GetTa() and MLX90640_CalculateTo calculate weird values #80
Comments
Figured out the Ta difference. That's just the shift that isn't subtracted from MLX90640_GetTa() when called inside MLX90640_CalculateTo(). Can't figure out where the temperature calculations are going wrong though. |
Frame data seems OK. Can you please share the dump of the EEPROM? Some remarks regarding the function calls:
Best regards |
MLX90640 const arrays.txt I attached the const arrays I calculated after getting the EEPROM data stored as a constant. The alpha, offset, kta, and kv had to calculated separately due to memory constraints. There is also an example of the data I'm getting. I'm going to redump the EEPROM and try out your suggestions. Really appreciate the help.
|
refresh rate is 2Hz, I2C is 400k register 0x800D returns 0x1901 |
Everything seems OK at first glimpse. Best regards |
redid the EEPROM dump and I get the same values for all 832 words |
The 24.2C is just my own reference for what my cheap laser temperature gun gives me so I can compare the temperatures. |
My current code. It just runs once on each key press.
Output: fTempResult[0] 5.7969722E8 0x4E0A35E9 the calculated results vary across 768 words from 1.99E8 to 1.18E9 with the sensor facing a warm light with a ceiling behind it. |
The EEPROM data and the FrameData both seem OK. The calculated Ta value also looks OK. I do not see where you call the CalculateTo function. Normally, the GetImage function is intended to give back a temperature image faster than the calculateTo function, but it does not calculate the temperatures - you only have normalized values. If you want to have a frame with calculated object temperatures you need to call MLX90640_CalculateTo() instead of MLX90640_GetImage(). |
Okay I will give that a try. I was rechecking my parameters and noticed the KsTo[5] was coming back with infinity and a couple others were being calculated properly. I fixed the other ones to calculate close to the example the but KsTo doesn't want to calculate write. Which gives me this now: With KsTo calculated with my EEPROM values (Note: not the exact same pixels) So I think I'm close. My EEPROM values are stored in ROM and not read on startup do to memory limitations of the PIC18. |
I think I have an idea what might be wrong - probably the int type of the PIC18 is 16-bit and the kstoScale variable which is specified as int simply overflows and becomes 0 - that is why you get the infinity in ksto. Could you please update your MLX90640_API.cpp and give it a try? Best regards |
I changed KsToScale to int32_t but still got infinity for KsTo[0..3] and KsTo[4] is -0.0002 Some of my output with KsTo as inifnity: some of my output with KsTo[0..4] set as -0.0002 |
That is strange: mlx90640->ksTo[i] = mlx90640->ksTo[i] / KsToScale; - looks like the only way to get infinity is if kstoScale is 0. Best regards |
Here's the parameter dump just after MLX90640_ExtractParameters(). I have definitely been moving some things around as I'm well over my memory limit pushing the kv, kva, alpha, offset, and eeprom arrays as constants. But I am calculating Kva, kv, and alpha on powerup just not the parts that push to the arrays. Just the alphaScale, for example. In ExtractKsToParameters() I am doing an I2C read to get the EEPROM fresh from 0x2461 and 0x2462. But everything else is from my const EEPROM array. |
The parameters also look OK. The only strange thing is the ksto. For your device ksto[0]=0, ksto[1] = -0.000198, ksto[2] = -0.0009 and ksto[3] = -0.001495, but -0.0002 is a good approximation and should not give too big of an error. I would still like to sort that ksto = infinity issue. Are you sure you are correctly passing the parameters to the functions? Best regards |
I only have the one variable for the parameters structure. MLX90640_ExtractParameters(EEPROM_Dump,¶mMLX90640); void ExtractKsToParameters(uint16_t *eeData, paramsMLX90640 *mlx90640)
} |
I did not realize you changed the code. I think you should modify your code: MLX90640_Read(TEMP_ADDR, 0x243D, 2, eeKsTo); // The EEPROM address is 0x243D (61st EEPROM address). Also I hope this '2' in the parameters means 2 address = 4 bytes mlx90640->ksTo[0] = eeKsTo[0] & 0x00FF; //You should use the data you just read from the device - other wise you never write anything in the eeData and you get some random values Best regards |
I will give that a shot. Double checking that address I definitely added 61d to 2400h. No wonder the ksTo is off. I only have enough memory to read a couple EEPROM values on each power up. Next iteration I plan on a more powerful microcontroller. Edit: Yes the 2 is actually reading 4 bytes. |
Still getting infinity values for ksTo. MLX90640_Read(TEMP_ADDR, 0x243D, 2, eeKsTo); //gets two words
|
Register 0x800D, in sensor RAM, is consistently 0x9901 now. Before it was 0x1901. What is the 16th bit for? It's not in the data sheet. |
KsToScale uses data from eeData[63} (address 0x243F). Can you check what is the value you have there? What is the KsToScale value that you get. Reading the MS bit as '1' in the register at address 0x800D means that you are writing it. Note that setting that bit to 1 should not influence the calculations, but could mean that you are writing to it instead of some write you wanted to do to some other register. Best regards |
at 0x243f i get 0x2558 |
The EEPROM value you get is correct, but is that same value in eeData[63] in your code?
What is the KsToScale value after executing those 2 lines? Best regards |
DEBUG: Extracting Parameters...please wait I'm still getting a large variation in temperatures across the sensor. Is there a formula to calculate my Celcius values from the raw sensor data? Like 0xFFab == x in celcius? I need to prove to my professor, this is a school capstone project, that the Celsius output is accurate. Latest data (this is just the bottom half or so of the output) held against a paperbox. Would my fingers touching it throw off the temperature that much? |
The formula is implemented in the driver. If you want to do the calculations externally, you can refer to the MLX90640 datasheet : https://www.melexis.com/-/media/files/documents/datasheets/mlx90640-datasheet-melexis.pdf . Touching the sensor is not a good idea as it creates thermal gradients and spoils the performance. How much time after POR are you taking your measurements? Did you put the appropriate decoupling capacitors as described in the datasheet? I wonder why KsToScale=0 after executing KsToScale = 1 << KsToScale;. Did you make sure that KsToScale is 32 bit int? Best regards |
My temperature average seems to stay about 2C from expected. I'm going to run the code on a beefier chip to see if it's just this micro that can't calculate fast enough between frames. |
KsToScale = 1 << KsToScale; The type of the expression You might want to use something similar to: KsToScale = 1ul << KsToScale;
my 2 cents. |
I get data similar, see below, but when I call GetTa() or CalculateTo() I get garbage data that definitely doesn't match me pointing the sensor at a desk or piece of paper that should be close to the same temp. Using the example data I get most of the same parameters calculated back out except for alpha (uint16), kv (int8) and kta (int8) are different because the example data are in decimals. My ksTo doesn't calculate properly for some reason other but even hard-coding -0.0002 hasn't helped. Ambient Ta comes out around 22.5, which makes sense but the Ta inside MLX906490_To() is 30.6?
My order of function calls is:
MLX90640_SynchFrame()
MLX90640_GetFrameData()
MLX90640_GetVdd()
MLX90640_ClearNewDataBit() //my own
MLX90640_GetTa()
MLX90640_CalculateTo()
while(!MLX90640_CheckNewFrame) //my own
//loop through twice as per the Melexis data sheet Measurement flow
RAM data
0xff94,0xffaf,0xffac,0xffac,0xffa8,0xffa3,0xff9f,0xff9e,0xff9c,0xff95,0xff8e,0xff8f,0xff8d,0xff8c,0xff80,0xff87,0xff81,0xff81,0xff79,0xff86,0xff7c,0xff83,0xff7a,0xff88,0xff7f,0xff89,0xff7d,0xff8f,0xff86,0xff92,0xff89,0xffa3,0xff9f,0xffbb,0xffae,0xff9f,0xff9c,0xff8f,0xff99,0xff91
Calculated values from MLX90640_GetImage() display with printf("%6.1f,", fTempResult[i]);
592.0,338272480.0,362236960.0,451677376.0,385740224.0,357841664.0,352839072.0,475474240.0,376040288.0,442351360.0,466023392.0,582827584.0,461466912.0,553721088.0,547306240.0,868710400.0,2073485824.0,1457174016.0,1639923584.0,1062499648.0,1117048576.0,1007729472.0,1176669312.0,879932992.0,835734784.0,817753600.0,914897472.0,799963840.0,703102208.0,733174976.0,809205696.0,751047872.0,584132416.0,718254464.0,727529920.0,732689984.0,592247680.0,722966720.0,742125504.0,791069504.0,623380992.0,879549504.0,863016000.0,849186432.0,711014208.0,1036032896.0,1065976704.0,1397301376.0,1953519232.0,717029248.0,930322368.0,625610176.0,823806528.0,471457376.0,657431424.0,563241280.0,568958144.0,422298112.0,595941056.0,526872288.0,473833376.0,384111200.0,546840512.0,500078720.0,364202208.0,400743552.0,427827680.0,505806528.0,415574816.0,424173664.0,465272864.0,537820032.0,476969216.0,450454080.0,583294464.0,651005760.0,669964096.0,614491712.0,835286528.0,1300244608.0,5021978112.0,3191158272.0,2957620480.0,1885567488.0,1862405376.0,1625331584.0,2040611968.0,1472667904.0,1337811456.0,1350845952.0,1630382464.0,1237255168.0,1087828608.0,1179745408.0,1401025536.0,1229205760.0,901572352.0,1152956672.0,1189396608.0,1176895488.0,904684224.0,1178945664.0,1352280704.0,1265989760.0,935620992.0,1309102336.0,1523960960.0,1398909568.0,1321256064.0,1681935104.0,2320309248.0,323784160.0
Calculated values from MLX90640_CalculateTo() display with printf("%6.1f,", fTempResult[i]);
-273.1, -273.1, -273.1, -273.1, -273.1, -273.1, -273.1, -273.1, -273.1, -273.1, -273.1, -273.1, -273.1, -273.1, -273.1, -273.1, -273.1, -273.1, -273.1, -273.1, 9291.8, -273.1, -273.1, -273.1, 11478.3, -273.1, -273.1, -273.1, -273.1, -273.1, -273.1, 5429.8, -273.1, -273.1, -273.1, -273.1, -273.1, -273.1, -273.1, -273.1, -273.1, -273.1, -273.1, -273.1, -273.1, -273.1, -273.1, -273.1, -273.1, -273.1
The text was updated successfully, but these errors were encountered: