首页
学习
活动
专区
圈层
工具
发布
社区首页 >问答首页 >如何在linux上的驱动程序中执行回调函数?

如何在linux上的驱动程序中执行回调函数?
EN

Stack Overflow用户
提问于 2022-01-14 11:57:14
回答 2查看 618关注 0票数 1

我必须修改在linux上运行的驱动程序,以添加从外部应用程序调用的回调函数。我已经实现了代码,但是当它在计算机启动时执行时,系统会出现错误并被阻塞。

这是我在驱动端的新代码:

代码语言:javascript
复制
typedef void (*callbackFunctionNoParams) ();

typedef struct T_EXI_CONFIGURE_BUS_
{
    T_mode mode;
    unsigned short NumeroRT;
    T_SA_Enable SA_Enable;
    unsigned short MINOR_CYCLE;
    callbackFunctionNoParams callback;

} T_EXI_CONFIGURE_BUS;

typedef struct PciExiDev_
{
   /**
    * It represents a char device to read/write
    */
   struct cdev charDevice;
   /**
    * IRQ assigned
    */
   unsigned int irq;

   /**
    * Callback function to be invoked
    */

   callbackFunctionNoParams callback;

   /**
    * Device control block
    */
   EXI_DCB theDCB;
} PciExiDev;

驱动端的执行代码:

代码语言:javascript
复制
static long exi_ioctl( struct file * filep, unsigned int cmd, unsigned long arg )
{
   PciExiDev * aPciExiDev = (PciExiDev *) filep->private_data;

   int result = SUCCESS;
   int i, j;
   long ret = 0;
   //printk("Ioctl received %d.\n",cmd);
   switch( cmd )
   {
      case FIO_EXI_CONFIGURE_BUS:
      {
         T_EXI_CONFIGURE_BUS config;
        T_LISTA_TRANS *auxTrans1, *auxTrans2;
        T_TRANSACTION_DCB *transDCB1;
         T_OPI opi;
         T_EXIS exis;
         unsigned short dato;
         unsigned short datolong[2];
         unsigned short ControlBlock[12];
//       printk("Exi configure bus initiated.\n");
         printk("TNB. Exi ioctl CONFIGURE BUS.\n");

         copy_from_user( &config, (T_EXI_CONFIGURE_BUS *) arg, sizeof(T_EXI_CONFIGURE_BUS) );

         LeerDatos( &aPciExiDev->theDCB, OPI_ADDRESS, 1, (unsigned short *) &opi, 1 );

         aPciExiDev->callback = config.callback;
         aPciExiDev->theDCB.modo = config.mode;
         aPciExiDev->theDCB.CicloMenor = config.MINOR_CYCLE;
         
         (*aPciExiDev->callback)();

...

客户端的新代码:

代码语言:javascript
复制
   if( theHWConfiguration.existExi() )
   {
      T_EXI_CONFIGURE_BUS bus_config;

      // Configura la tarjega exi en modo Bus Controller.
      bus_config.mode = BC;
      bus_config.NumeroRT = 28;
      bus_config.MINOR_CYCLE = 20;
      bus_config.callback = &bcInterruptHandler2;
      status = ioctl( A_fd_exi, FIO_EXI_CONFIGURE_BUS, reinterpret_cast<long>( &bus_config ) );
   }
   return status;
}

void C_EXI::bcInterruptHandler2()
{
   std::cout<< "bcInterruptHandler2" << endl;
}

这是执行代码的结果:

碰撞图像

如果有人能帮助我或提出另一种方法来做这件事,我将非常感激。

EN

回答 2

Stack Overflow用户

回答已采纳

发布于 2022-01-14 14:14:16

您的callback必须在内核空间运行,然后将其写入std::cout。在遍历代码时,它告诉我们内核模式地址空间和用户端进程地址空间之间存在冲突。这意味着,如果callback函数是在用户端声明的,而是在内核空间中调用的,则会出现错误。

票数 0
EN

Stack Overflow用户

发布于 2022-01-14 13:42:39

崩溃图像表明,在代码中的某个地方,您正试图访问一个无效的指针。恐怕我不能用提供的小上下文来调试您的代码,但是我可以给您一些建议:

  • 尽量避免铸造,直到严格必要。
  • 当您转换到指针时,请反复检查这正是您需要做的。
  • 在错误消息中,还有调用堆栈:查看它,以确定错误在哪里。
  • 您可以简单地在代码中添加一些printk("%p",指针)来调试变量的内容。
票数 0
EN
页面原文内容由Stack Overflow提供。腾讯云小微IT领域专用引擎提供翻译支持
原文链接:

https://stackoverflow.com/questions/70710212

复制
相关文章

相似问题

领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档