This thread has been locked.
If you have a related question, please click the "Ask a related question" button in the top right corner. The newly created question will be automatically linked to this question.
Hi! I'm using MSP430I2020's analog-to-digital converter in single ended configuration to read two analog voltages from a sensor.
I've connected the sensor outputs to A0.0+ and A1.0+, respectively. A0.0- and A1.0- are connected to GND. This is the code I'm using:
void SD24_initialize(void) { // Internal ref SD24_init(SD24_BASE, SD24_REF_INTERNAL); //Group with Channel 0 SD24_initConverterAdvancedParam param = {0}; param.converter = SD24_CONVERTER_0; param.conversionMode = SD24_CONTINUOUS_MODE; param.groupEnable = SD24_GROUPED; param.inputChannel = SD24_INPUT_CH_ANALOG; param.dataFormat = SD24_DATA_FORMAT_2COMPLEMENT; param.interruptDelay = SD24_FOURTH_SAMPLE_INTERRUPT; param.oversampleRatio = SD24_OVERSAMPLE_256; param.gain = SD24_GAIN_1; SD24_initConverterAdvanced(SD24_BASE, ¶m); //Group with Channel 1 param.converter = SD24_CONVERTER_1; param.conversionMode = SD24_CONTINUOUS_MODE; param.groupEnable = SD24_NOT_GROUPED; param.inputChannel = SD24_INPUT_CH_ANALOG; param.dataFormat = SD24_DATA_FORMAT_2COMPLEMENT; param.interruptDelay = SD24_FOURTH_SAMPLE_INTERRUPT; param.oversampleRatio = SD24_OVERSAMPLE_256; param.gain = SD24_GAIN_1; SD24_initConverterAdvanced(SD24_BASE, ¶m); // Enable interrupt SD24_enableInterrupt(SD24_BASE, SD24_CONVERTER_1, SD24_CONVERTER_INTERRUPT); // Delay ~200us for 1.2V ref to settle __delay_cycles(3200); } void SD24_read(float* op1, float* op2) { // Start conversion SD24_startConverterConversion(SD24_BASE, SD24_CONVERTER_1); // Enter LPM0 w/ interrupts __bis_SR_register(LPM0_bits | GIE); SD24_stopConverterConversion(SD24_BASE, SD24_CONVERTER_1); for (i = 0; i < NUM_OF_RESULTS; i++) { Ch0results_float[i] = Int24BitTwoComplementToSignedInt(Ch0results[i]) * LSB * 2.0; Ch1results_float[i] = Int24BitTwoComplementToSignedInt(Ch1results[i]) * LSB * 2.0; } if (NUM_OF_RESULTS > 1) // Mean value { float sum0 = 0, sum1 = 0; for (i = 0; i < NUM_OF_RESULTS; i++) { sum0 += Ch0results_float[i]; sum1 += Ch1results_float[i]; } *op1 = sum0 / NUM_OF_RESULTS; *op2 = sum1 / NUM_OF_RESULTS; } else { *op1 = Ch0results_float[0]; *op2 = Ch1results_float[0]; } } int32_t Int24BitTwoComplementToSignedInt(uint32_t val) { return (0x800000 & val ? (int)(0x7FFFFF & val) - 0x800000 : val); }
#if defined(__TI_COMPILER_VERSION__) || defined(__IAR_SYSTEMS_ICC__) #pragma vector=SD24_VECTOR __interrupt void SD24_ISR(void) { #elif defined(__GNUC__) void __attribute__ ((interrupt(SD24_VECTOR))) SD24_ISR (void) { #else #error Compiler not supported! #endif switch (__even_in_range(SD24IV,SD24IV_SD24MEM3)) { case SD24IV_NONE: break; case SD24IV_SD24OVIFG: break; case SD24IV_SD24MEM0: break; case SD24IV_SD24MEM1: // Save CH0 results (clears IFG) Ch0results[i] = SD24_getResults(SD24_BASE, SD24_CONVERTER_0); // Save CH1 results (clears IFG) Ch1results[i] = SD24_getResults(SD24_BASE, SD24_CONVERTER_1); i++; if (i == NUM_OF_RESULTS) { i = 0; __bic_SR_register_on_exit(LPM0_bits); // Wake up } break; case SD24IV_SD24MEM2: break; case SD24IV_SD24MEM3: break; default: break; } }
The values read by the ADC are half than those measured with a voltmeter. So I've multiplied them with 2.
My questions: is it expected for SD24 to return half values when working in single ended configuration, with 2's complement output format? Am I reading the values correctly?
Hello,
The SD24 will return a value that's between +Full Scale Range (FSR) and -FSR. Since you're using a gain of 1, the +/-FSR is +/-928mV. However, your input circuitry (probably voltage dividers) will impact how much of that FSR you're using at the maximum input voltage. Ideally, you'd want to use the full range. Basically, what you're seeing is normal, but you'll need to calibrate the device.
Let's assume the device measures an input voltage within +/-FSR. However, this isn't in human terms, but MCU terms. Thus, you need to calibrate it at a certain known input voltage, which would allow you to calculate the necessary scaling factor that translates the MCU results (e.g. 0x1234) into human results (e.g. 3300mV).
Does this make sense?
Regards,
James
MSP Customer Applications
Hi! Please help me understand this correctly. This is how I see things now:
+/-FSR is +/-1.16V (as I understand from the datasheet, since it uses internal reference).
+FSR means to me 1.16V (0x7FFFFF), FSR/2 means 0.58V (0x000000) and -FSR means 0 (0x800000). I am working in 2's complement, single ended configuration (negative input Connected to GND).
The voltage measured with the voltmeter is ~220mV. My code returned ~110mV, so it seems I have to multiply by 2 to get the real voltage (I forgot to mention in the code, LSB = 1.16/2^24)
Shouldn't it return 220mV or is there a problem with my code?
There is mathematical relation between ADC input voltage and output value. I can understand calibration if I work in another voltage range, let's day 1-2V.
Hello,
I don't see where you're setting the SD24LSBACC and SD24LSBTOG bits, which give access to the least significant bits of the digital filter output. I suspect that you're only reading 16 bits rather than 24 bits. Please read Section 13.2.7.2 Digital Filter Output in the User's Guide.
Regards,
James
MSP Customer Applications
Hi,
I think that SD24_getResults() does this for me. Please correct me if I am wrong.
//***************************************************************************** // //! \brief Returns the results for a converter //! //! This function gets the results from the SD24MEMx register for upper 16-bit //! and lower 16-bit results, and concatenates them to form a long. The actual //! result is a maximum 24 bits. //! //! \param baseAddress is the base address of the SD24 module. //! \param converter selects the converter who's results will be returned //! Valid values are: //! - \b SD24_CONVERTER_0 //! - \b SD24_CONVERTER_1 //! - \b SD24_CONVERTER_2 //! - \b SD24_CONVERTER_3 //! //! \return Result of conversion // //***************************************************************************** uint32_t SD24_getResults(uint16_t baseAddress, uint8_t converter) { volatile uint16_t OSR; // Calculate address of MEM results uint16_t address = baseAddress + (OFS_SD24MEM0 + (converter * 0x02)); // Get high word result HWREG16(baseAddress + (OFS_SD24CCTL0 + (converter * 0x02))) &= ~(SD24LSBACC); uint32_t highResult = (uint32_t)HWREG16(address); // Get low word result HWREG16(baseAddress + (OFS_SD24CCTL0 + (converter * 0x02))) |= SD24LSBACC; uint16_t lowResult = HWREG16(address); HWREG16(baseAddress + (OFS_SD24CCTL0 + (converter * 0x02))) &= ~(SD24LSBACC); // Determine the OSR and combine the high and low result words as appropriate OSR = HWREG16(baseAddress + (OFS_SD24CCTL0 + (converter * 0x02))) & (SD24OSR0 | SD24OSR1); if(OSR == SD24_OVERSAMPLE_256) { return (highResult << 8) | lowResult; } else if(OSR == SD24_OVERSAMPLE_128) { return (highResult << 5) | lowResult; } else if(OSR == SD24_OVERSAMPLE_64) { return (highResult << 2) | lowResult; } else // OSR = SD24_OVERSAMPLE_32 { return (highResult); } }
I've tried setting the output format to offset binary, which means that 0x000000 is 0V and 0xFFFFFF is 1.16V. Using the following conversion I get ~600mV, where it should be ~220mV.
Ch0results_float[i] = Ch0results[I] * LSB;
Where did you find this function? After a quick glance, it appears to be fine. Is your input voltage 220mV? Could it vary? Is it AC or DC? What do you get BEFORE multiplying the result with LSB? Can you try inputting a known positive voltage from a signal generator? Connect its positive and negative outputs to the SD24 +/- inputs.
Based on the MSP430 portion of your schematic (found in the thread below), I'd recommend removing the diodes and shorting the in-line resistors too. What's the sensor that you're measuring?
MSP432P401R: Problem with I2C pullups - MSP low-power microcontroller forum - MSP low-power microcontrollers...
Regards,
James
MSP Customer Applications
Where did you find this function? In driverlib. I'm using msp430ware_3_80_04_05.
After a quick glance, it appears to be fine. Is your input voltage 220mV? It is 223mV, to be precise (measured with a voltmeter).
Could it vary? It can vary max. 1mV.
Is it AC or DC? It is DC.
What do you get BEFORE multiplying the result with LSB? This is the reading: 0x0018632F. After multiplying with LSB*2.0, I get 0.222916409, where LSB is 1.17/16777216.0. Why do I have to multiply it with 2?
Can you try inputting a known positive voltage from a signal generator? Connect its positive and negative outputs to the SD24 +/- inputs. I've already tried that, but using a laboratory power supply, not a signal generator. I confirm that it measures fine. I've tested up to 700mV, because I do not need higher values.
Based on the MSP430 portion of your schematic (found in the thread below), I'd recommend removing the diodes and shorting the in-line resistors too. Thank you. I've taken this advice into account. However, I think it is good practice to provide a minimal protection to the MSP430. I will lower the values of these resistors. Since ADC input impedance (single end) is 200kohms I think that, let's say, 10ohms resistors will not noticeably impact the ADC reading.
What's the sensor that you're measuring? It is a gas sensor (OX-B431), which comes with its own signal conditioning circuit. Since I am not measuring in controlled conditions, the sensor output voltages cannot change.
Cristian Monea said:What do you get BEFORE multiplying the result with LSB? This is the reading: 0x0018632F. After multiplying with LSB*2.0, I get 0.222916409, where LSB is 1.17/16777216.0. Why do I have to multiply it with 2?
I think I just figured out the issue. It's because you're dividing by 2^24 instead of 2^23. Remember, for the maximum positive number, you'll only have 23 bits, not 24 bits. For example, take a look at Equation 1 in the Measuring Single-Ended 0- to 5-V Signals with Differential Delta-Sigma ADCs app note.
Cristian Monea said:Based on the MSP430 portion of your schematic (found in the thread below), I'd recommend removing the diodes and shorting the in-line resistors too. Thank you. I've taken this advice into account. However, I think it is good practice to provide a minimal protection to the MSP430. I will lower the values of these resistors. Since ADC input impedance (single end) is 200kohms I think that, let's say, 10ohms resistors will not noticeably impact the ADC reading.
I'm sorry if I wasn't clear. What I meant to say was just remove these for testing. I'm sure they have their place (ESD protection, current limiting, etc.) in your design.
Hope this helps!
Regards,
James
MSP Customer Applications
**Attention** This is a public forum