最近在做美颜软件虚拟摄像头的功能,类似YY开播(我们获取笔记本的摄像头列表时,列表中除了自带的摄像头,以及usb的摄像头,还有YY开播),查了一下网络上实现的虚拟摄像头展示的都是本地的视频,但是我这边的美颜软件需要读取自带摄像头拍摄的画面,美颜后将视频帧数据传输到虚拟摄像头中,当我们调用虚拟摄像头的时候就能看到美颜后的画面,这个功能的实现就用到了美颜软件和虚拟摄像头的内存共享。
QMap<const char*, void*> mRegisterData; void* RegisterSendData(const char *name, const int buf_size) { if (mRegisterData.count(name) > 0) { return mRegisterData[name]; } HANDLE hMapFile2 = CreateFileMapping( INVALID_HANDLE_VALUE, // 物理文件句柄 NULL, // 默认安全级别 PAGE_READWRITE, // 可读可写 0, // 高位文件大小 buf_size, // 低位文件大小 name // 共享内存名称 ); // 映射缓存区视图 , 得到指向共享内存的指针 if (hMapFile2 != nullptr) { void* lpBase2 = MapViewOfFile( hMapFile2, // 共享内存的句柄 FILE_MAP_ALL_ACCESS, // 可读写许可 0, 0, buf_size ); mRegisterData.insert (name, lpBase2); return mRegisterData[name]; } else { return nullptr; } }
void set_buffer_data(LPVOID lpBase,unsigned char* buffer_data, int buffer_size) { if(lpBase) { memcpy((unsigned char*)lpBase, buffer_data, buffer_size); } }
LPVOID lpBase= register_data("FrameBuffer",800* 600 * 3); if(lpBase) { QImage map_img = effectImage.scaled(800,600);//effectImage即美颜后的图片 set_buffer_data(lpBase,map_img.bits(),BUF_SIZE); }
bool GetFrameBuffer(const char* name, unsigned char* buffer_data, const int buf_size) { if (buffer_data == nullptr) { return false; } 打开共享的文件对象 HANDLE hMapFile = OpenFileMapping(FILE_MAP_ALL_ACCESS, NULL, name); if (hMapFile) { LPVOID lpBase = MapViewOfFile(hMapFile, FILE_MAP_ALL_ACCESS, 0, 0, 0); // 将共享内存数据拷贝出来 if (lpBase != nullptr) { memcpy(buffer_data, lpBase, buf_size); // 解除文件映射 UnmapViewOfFile(lpBase); } // 关闭内存映射文件对象句柄 CloseHandle(hMapFile); return true; } else { // 打开共享内存句柄失败 std::cout << "打开共享失败!" << std::endl; return false; } }
unsigned char* szBuffer = nullptr; szBuffer = (unsigned char*)malloc(800*600*3*sizeof(char)); GetFrameBuffer("FrameBuffer", szBuffer, 800*600*3);