将静态方法作为函数指针传递给 taskSpawn 函数

Pass a static method to taskSpawn function as a function pointer

提问人:Dark Sorrow 提问时间:7/14/2023 更新时间:7/16/2023 访问量:53

问:

taskSpawn 引用

typedef int         (*FUNCPTR) (...);     /* ptr to function returning int */

在发送函数指针时,我收到以下提到的错误。
我已经尝试了所有排列组合。

this->readerTaskId = taskSpawn("test1", 90, VX_FP_TASK, 20000, SerialPort::reader, (_Vx_usr_arg_t)this, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL);  
this->readerTaskId = taskSpawn("test1", 90, VX_FP_TASK, 20000, &SerialPort::reader, (_Vx_usr_arg_t)this, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL); 
this->readerTaskId = taskSpawn("test1", 90, VX_FP_TASK, 20000, SerialPort::reader, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL);
this->readerTaskId = taskSpawn("test1", 90, VX_FP_TASK, 20000, &SerialPort::reader, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL);
this->readerTaskId = taskSpawn("test1", 90, VX_FP_TASK, 20000, reader, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL); 
this->readerTaskId = taskSpawn("test1", 90, VX_FP_TASK, 20000, SerialPort::task_one, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL);
this->readerTaskId = taskSpawn("test1", 90, VX_FP_TASK, 20000, &SerialPort::task_one, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL);
this->readerTaskId = taskSpawn("test1", 90, VX_FP_TASK, 20000, task_two, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL);
this->readerTaskId = taskSpawn("test1", 90, VX_FP_TASK, 20000, &task_two, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL);

错误:

D:/Test/source/SerialPort.cpp:108:24: error: no matching function for call to 'taskSpawn'
                this->readerTaskId = taskSpawn("test1", 90, VX_FP_TASK, 20000, task_two, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL);
                                     ^~~~~~~~~
D:/Test/../primary_source/krnl/h/public\taskLibCommon.h:277:16: note: candidate function not viable: no known conversion from 'int (_Vx_usr_arg_t)' (aka 'int (long)') to 'FUNCPTR' (aka 'int (*)(...)') for 5th argument
extern TASK_ID  taskSpawn       (char * name, int priority, int options,
                ^
1 error generated.

页眉:

#include <taskLib.h>
#include "vxWorks.h"

namespace TaskTest
{
    int task_two(_Vx_usr_arg_t args){return 0;} //For Testing
    class SerialPort
    {
    public:
        bool read(void);
    private:
        static int reader(_Vx_usr_arg_t args);
        TASK_ID readerTaskId;
        static int task_one(_Vx_usr_arg_t args){return 0;} //For Testing
        
    };
}

源:

namespace TaskTest
{
    bool SerialPort::read(void)
    {
        this->readerTaskId = taskSpawn(this->taskName.c_str(), 90, VX_FP_TASK, 20000, SerialPort::reader, (_Vx_usr_arg_t)this, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL);   
        //this->readerTaskId = taskSpawn(this->taskName.c_str(), 90, VX_FP_TASK, 20000, &SerialPort::reader, (_Vx_usr_arg_t)this, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL);    
        //this->readerTaskId = taskSpawn(this->taskName.c_str(), 90, VX_FP_TASK, 20000, reader, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL);    
        //this->readerTaskId = taskSpawn(this->taskName.c_str(), 90, VX_FP_TASK, 20000, SerialPort::reader, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL);
        //this->readerTaskId = taskSpawn(this->taskName.c_str(), 90, VX_FP_TASK, 20000, SerialPort::task_one, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL);
        //this->readerTaskId = taskSpawn("test1", 90, VX_FP_TASK, 20000, task_two, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL);
        if(this->readerTaskId == TASK_ID_ERROR)
        {
            printf("taskSpawn failed for Task 1\n");
        }
            
    }
    
    int SerialPort::reader(_Vx_usr_arg_t args)
    {
        SerialPort *sp = (SerialPort *) args;
        printf("Serial Port Read Started. Waiting for data...\n");
        while (true)
        {
            try //Exception is inside while statement
            {
                //Do Something
            }
            catch (const std::exception& ex)
            {
                //TODO Log Error
            }
        }
        return 0;
    }
}
C++ VXWORKS公司

评论


答:

1赞 Fabio A. 7/14/2023 #1

您链接的参考资料说

入口地址 entryPt 是 任务。一旦设置了 C 环境,就会调用该例程 SPH公司指定的例程将使用十个给定的参数进行调用。

因此,其签名必须如下:

void entry(int a1, int a2, int a3, int a4, int a5, int a6, int a7, int a8, int a9, int a10);

若要将其作为参数传递给 taskSpawn,必须将其强制转换为 FUNCPTR。像这样的东西:

this->readerTaskId = taskSpawn("test1", 90, VX_FP_TASK, 20000, FUNCPTR(entry), 0, 0, 0, 0, 0, 0, 0, 0, 0, 0);
-1赞 Dark Sorrow 7/16/2023 #2

使用reinterprt_cast和const_cast解决了问题。

this->readerTaskId = taskSpawn(const_cast<char *>(this->taskName.c_str()), 90, VX_FP_TASK, 20000, reinterprt_cast<FUNCPTR>(SerialPort::reader), reinterprt_cast<_Vx_usr_arg_t>(this), NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL);

评论

0赞 Fabio A. 7/16/2023
这就是未定义行为:vxWorks 使用指定类型的所有 10 个参数调用该函数。如果它有效,那就靠运气了。我昨天给了你解决方案作为答案。
0赞 Dark Sorrow 7/16/2023
@FabioA。你能详细说明一下吗?我无法理解你想传达什么。
0赞 Fabio A. 7/16/2023
您引用的引用指出“将使用十个给定参数调用指定的例程”,但您正在向 taskSpawn 传递一个指向仅接受一个参数的例程的指针。根据您使用的体系结构和所使用的调用约定,这可能有效,也可能无效。例如,如果在调用函数之前将参数放在堆栈上,但使用的调用约定规定函数有责任在返回之前还原堆栈,则在返回时不会还原堆栈,因为只会弹出一个参数。