在计时器回调完成之前,DiscardAsyncTimer返回

时间:2019-07-07 15:03:37

标签: multithreading labwindows cvi

我正在尝试使用异步计时器在LabWindows / CVI 2017中编写程序,但是遇到了DiscardAsyncTimer()函数的问题。从DiscardAsyncTimer()的文档中:

  

创建或放弃异步计时器的调用不会完成   (将阻止),直到所有未完成的异步回调返回为止。

但是,在调用DiscardAsyncTimer()之后,我在释放异步计时器线程中使用的内存时遇到了一些内存问题。我希望不再使用该内存,但是显然不是这样吗?我下面有一个示例程序,该程序重新创建了我的问题。运行时,此程序由于尝试访问释放的内存而生成“一般保护错误”。但是,如果我对文档以及文档本身的理解是正确的,那应该是不可能的,因为应该在所有回调返回之前阻塞DiscardAsyncTimer()。

我的问题:

  1. 我是否正确理解了文档?
  2. 在这个例子中我做些蠢事吗?
  3. 在释放内存之前,我应该如何验证我的异步计时器线程已完成运行?

示例程序:

#include <ansi_c.h>
#include <asynctmr.h>
#include <stdio.h>
#include <utility.h> 
#include <userint.h>


typedef struct {
    int* array;
} MyStruct;

int CVICALLBACK ShuffleValues (int reserved, int timerId, int event, void *callbackData, int eventData1, int eventData2)
{
    if (event == EVENT_TIMER_TICK) {
        printf("start\n");
        MyStruct* mystruct = callbackData;

        // Shuffle values
        for(int i = 0;i < 1000;i++) {
            mystruct->array[0] = mystruct->array[1];
            mystruct->array[1] = mystruct->array[2];
            Delay(0.01);
        }

        printf("finished\n");
    }

    return 0;
}

int main ()
{
    // Allocate memory
    MyStruct* mystruct = malloc(sizeof(MyStruct));
    mystruct->array = malloc(10 * sizeof(int));

    // Start Async Timer
    int timer = NewAsyncTimer(0.01, -1, 1, ShuffleValues, mystruct);

    // Wait a while to let the timer thread run a bit
    Delay(0.5);

    // Destroy Async Timer  
    printf("start destroying\n");
    int retval = DiscardAsyncTimer(timer);
    printf("finished destroying: %d\n", retval);

    // Free memory now that the timer thread is no longer running
    free(mystruct->array);
    free(mystruct);

    Delay(1);
    return 0;
}

我也在一周前的LabWindows / CVI论坛上问了这个问题,没有任何回应:https://forums.ni.com/t5/LabWindows-CVI/DiscardAsyncTimer-returning-before-timer-callback-is-complete/td-p/3943460

1 个答案:

答案 0 :(得分:1)

我在NI论坛上没有看到您的问题。快速的答案是异步计时器在单独的线程中运行,因此您需要采取通常的预防措施。 发生错误时,您可以使用[Windows] [Threads]看到该错误,并从一个线程跳转到另一个线程。 使用volatile变量进行同步,或者最好使用信号量。