TI论坛
直播中

张秀兰

8年用户 1341经验值
私信 关注
[问答]

如何写出ADC的分时复用?


  • 我想问一下如何写出ADC的分时复用,我要用来量测不同引脚/物理路的连续AD值
    我用的是CC1310 LAUNCHPAD,文挡是用ADCBUF......
    我遇到的问题大致有:
    1.    continuousConversion.adcChannel = 1  //通道-对应引脚
           这个参数是建构在continuousConversion里面,那我到底该怎么更新这个参数?在哪里更新?来改变采样引脚
    2.    我的想法是采样30个AD值后,便切换引脚,但我只能让他采样完30个数字后转换停止,无法切换到下个引脚,
           我的程式该怎么写,求助~~~~
    //****************************************************************************
    //adc test
    //
    #include
    #include
    #include
    #include
    #include
    #if defined(CC2650DK_7ID) || defined(CC1310DK_7XD)
    #include
    #endif
    #include "Board.h"
    #define ADCBUFFERSIZE (1)

    uint16_t sampleBufferOne[ADCBUFFERSIZE];
    uint16_t sampleBufferTwo[ADCBUFFERSIZE];
    uint32_t microVoltBuffer[ADCBUFFERSIZE];
    uint32_t buffersCompletedCounter = 0;
    char uartTxBuffer[(10 * ADCBUFFERSIZE) + 25];

    UART_Handle uart;
    int c =0; //一组adc转换的次数
    int q =0; //第几组/第几pin 做adc转换
    void adcBufCallback(ADCBuf_Handle handle, ADCBuf_Conversion *conversion, void *completedADCBuffer, uint32_t completedChannel) {
    uint_fast16_t i;
    uint_fast16_t uartTxBufferOffset;
    ADCBuf_adjustRawValues(handle, completedADCBuffer, ADCBUFFERSIZE, completedChannel);
    ADCBuf_convertAdjustedToMicroVolts(handle, completedChannel, completedADCBuffer, microVoltBuffer, ADCBUFFERSIZE);
    uartTxBufferOffset = sprintf(uartTxBuffer, "rnBuffer %u finished:rn", (unsigned int)buffersCompletedCounter++);
    for (i = 0; i < ADCBUFFERSIZE; i++)
    {
    uartTxBufferOffset += sprintf(uartTxBuffer + uartTxBufferOffset, "%u,", (unsigned int)microVoltBuffer);
    }
    uartTxBuffer[uartTxBufferOffset] = 'n';
    UART_write(uart, uartTxBuffer, uartTxBufferOffset + 1);

    //********************************************************************
    c++;
    //转换30次 : 0-29
    if(c%30==0)
    {
    ADCBuf_convertCancel(handle);
    }
    //********************************************************************
    }
    void uartCallback(UART_Handle handle, void *buf, size_t count)
    {
    return;
    }

    void *mainThread(void *arg0)
    {
    UART_Params uartParams;
    ADCBuf_Handle adcBuf;
    ADCBuf_Params adcBufParams;
    ADCBuf_Conversion continuousConversion;
    ADCBuf_init();
    UART_init();
    #if defined(CC2650DK_7ID) || defined(CC1310DK_7XD)
    PIN_State pinState;
    PIN_Config AlsPinTable[] =
    {
    Board_ALS_PWR | PIN_GPIO_OUTPUT_EN | PIN_GPIO_HIGH | PIN_PUSHPULL, // Turn on ALS power /
    PIN_TERMINATE // Terminate list /
    };

    PIN_open(&pinState, AlsPinTable);
    #endif

    UART_Params_init(&uartParams);
    uartParams.writeDataMode = UART_DATA_BINARY;
    uartParams.writeMode = UART_MODE_CALLBACK;
    uartParams.writeCallback = uartCallback;
    uartParams.baudRate = 115200;
    uart = UART_open(Board_UART0, &uartParams);
    ADCBuf_Params_init(&adcBufParams);
    adcBufParams.callbackFxn = adcBufCallback;
    adcBufParams.recurrenceMode = ADCBuf_RECURRENCE_MODE_CONTINUOUS;
    adcBufParams.returnMode = ADCBuf_RETURN_MODE_CALLBACK;
    adcBufParams.samplingFrequency = 0.2;
    adcBuf = ADCBuf_open(CC1310_LAUNCHXL_ADCBUF0, &adcBufParams);
    continuousConversion.arg = NULL; //NULL
    continuousConversion.adcChannel = q; //6
    continuousConversion.sampleBuffer = sampleBufferOne;
    continuousConversion.sampleBufferTwo = sampleBufferTwo;
    continuousConversion.samplesRequestedCount = ADCBUFFERSIZE;
    //********************************************************************
    for(q=0;q<7;q++)
    {
    ADCBuf_convert(adcBuf, &continuousConversion, 1);
    //while(ADCBuf_convertCancel(adcBuf))
    }
    //********************************************************************
    while(1)
    {
    sleep(1000);
    }

    }

回帖(2)

潘千

2024-12-20 09:51:38
你应该尽量使用硬件的功能, 
简单的, 可考虑使用 ADC 中断, 这中断服务中切换通道再启动.
当然, 还可以使用事件触发, 加 FIFO 或者 DMA 来实现连续转换
举报

康桃花

2024-12-21 17:25:24
要实现ADC的分时复用,您需要在CC1310 LAUNCHPAD上编写一个程序,以便在不同的引脚/物理路径上连续测量AD值。以下是实现这一功能的步骤和代码示例:

1. 首先,您需要包含必要的头文件和初始化ADC模块。

```c
#include "CC1310_LAUNCHXL.h"

void initADC(void) {
    ADCBuf_configureAdc(ADCBUF_BASE, ADC_0, ADC_TRIGGER_PROCESSOR, 0);
    ADCBuf_enableConversion(ADCBUF_BASE, ADC_0);
}
```

2. 接下来,您需要创建一个函数来设置ADC通道。这个函数将允许您在不同的引脚之间切换。

```c
void setADCChannel(uint8_t channel) {
    ADCBuf_setAdcChannel(ADCBUF_BASE, ADC_0, channel);
}
```

3. 现在,您需要编写一个函数来读取ADC值。这个函数将从ADC缓冲区中读取值,并在读取后清除缓冲区。

```c
uint16_t readADCValue(void) {
    uint16_t adcValue;

    while (ADCBuf_getResults(ADCBUF_BASE, ADC_0, &adcValue) == ADCBUF_DONE) {
        // 等待直到ADC缓冲区有数据
    }

    ADCBuf_clearInterrupt(ADCBUF_BASE, ADC_0);
    return adcValue;
}
```

4. 最后,您可以编写主函数来实现分时复用。在这个示例中,我们将在三个引脚之间切换,并在每个引脚上采样30个AD值。

```c
int main(void) {
    uint8_t channels[] = {1, 2, 3}; // ADC通道数组
    uint8_t numChannels = sizeof(channels) / sizeof(channels[0]); // 通道数量
    uint8_t numSamples = 30; // 每个通道的采样数量

    // 初始化设备和ADC模块
    initADC();

    while (1) {
        for (uint8_t i = 0; i < numChannels; i++) {
            setADCChannel(channels[i]); // 设置ADC通道

            for (uint8_t j = 0; j < numSamples; j++) {
                uint16_t adcValue = readADCValue(); // 读取ADC值
                // 处理ADC值(例如,将其存储在数组中或发送到另一个模块)
            }
        }
    }
}
```

这个示例程序将在三个引脚之间切换,并在每个引脚上采样30个AD值。您可以根据需要修改通道数组和采样数量。希望这对您有所帮助!
举报

更多回帖

×
20
完善资料,
赚取积分