代码改变世界

纠结的CLI C++与Native C++的交互

2014-08-03 21:20  hailants  阅读(1518)  评论(2编辑  收藏  举报

最近在写点东西,涉及到了CLR C++与Native C++的互相调用的问题,结果...........纠结啊。

交互原型

交互原型是这样的:

void* avio_alloc_context(
                  unsigned char *buffer,
                  int buffer_size,
                  int write_flag,
                  void *opaque,
                  int (*read_packet)(void *opaque, uint8_t *buf, int buf_size),
                  int (*write_packet)(void *opaque, uint8_t *buf, int buf_size),
                  int64_t (*seek)(void *opaque, int64_t offset, int whence));

第一次尝试

public ref class test
{
  int Writebuffer(void *opaque,libffmpeg::uint8_t* buf, int buf_size){    
}

main()
{
test testobj=gcnew test();
    libffmpeg::avio_alloc_context(outbuffer, 32768,0,NULL ,NULL,testobj->Writebuffer ,NULL);    
}

结果报错,要求创建指向成员的指针。

第二次尝试

使用了委托,然后使用  Marshal::GetFunctionPointerForDelegate获取函数指针结果....

编译不报错了,运行时回调倒是有了,可回调函数return后就报错..........估计是对象回收或者是非法访问导致。

第三次尝试

直接构造了静态函数

int static Writebuffer(void *opaque,libffmpeg::uint8_t* buf, int buf_size)

这个可以正常回调了,运行也不报错,可我需要的对象在静态函数中可获取不到(为了多线程考虑,不能用全局对象)。

于是需要传入 void *opaque 参数。

但是这个要传的是一个托管对象,咋办呢?使用pin_ptr,嗯,可以传进去。

呃,新的问题来了,void *怎转回托管对象呢?好吧,这个我是没找到合适的办法,

再次失败.........

第四次尝试

找到了gcroot<>这个模版,必须有这个:#include<vcclr.h> 

代码:

public ref class test
{
gcroot<BinaryWriter^> _writedStream;
  int static Writebuffer(void *opaque,libffmpeg::uint8_t* buf, int buf_size){    
}

好吧,斜体部分报错,说是托管类中不能使用非托管对象........

再改:

gcroot<BinaryWriter^>* _writedStream;

这次不报错了。

继续:

Open( BinaryWriter^ writer)
{
*_writedStream=writer;
}

结果运行时告诉我找不到对象....好吧,还需要初始化:

_writedStream=new gcroot<BinaryWriter ^>;
    *_writedStream=writer;

这次终于木有问题了............

avio_alloc_context(outbuffer, 32768,0,this->_writedStream ,NULL,this->Write_buffer ,NULL);

Write_buffer(void *opaque,libffmpeg::uint8_t* buf, int buf_size){  
    array<unsigned char>^ mbuf=gcnew array<unsigned char>(buf_size);
    System::Runtime::InteropServices::Marshal::Copy((IntPtr)buf,mbuf,0,buf_size);
    gcroot<BinaryWriter ^>* wr=(gcroot<BinaryWriter ^>*) opaque;
    BinaryWriter ^writeStream= *wr;
    writeStream->Write(mbuf,0,buf_size);
    writeStream->Flush();
    return 0;
}

总结

就俩字:坑爹