下载本插件
本插件读取音频和视频文件,输出音频样本和视频样本,音频样本为16位PCM,采样率48000;视频样本为RGB32。大部分音频和视频文件格式都可以读取。本插件类型为DLL。
本插件是通过创建媒体基础“源读取器”对象实现读取音视频文件。使用MFCreateSourceReaderFromURL函数创建源读取器,源读取器接口为IMFSourceReader,如果读取的是音频文件,只创建音频线程,如果读取视频文件,将创建视频线程和音频线程,在线程中使用IMFSourceReader接口的ReadSample方法读取样本,样本为媒体基础样本;使用IMFSourceReader接口的SetCurrentPosition方法更改当前位置。定义了7个导出函数,用于对本读取器的操作。
使用方法
首先,加载本读取器DLL,获得读取器模块句柄。
HMODULE hSReader = LoadLibrary(L"SReader.dll");
定义媒体信息和样本信息结构,定义接收当前位置信息的函数(如果不需要,也可以不定义),定义视频音频样本接收函数:
struct INFO
{int VideoWidth=0;//视频宽,单位像素int VideoHeight=0;//视频高,单位像素LONGLONG FrameCur=0;//帧持续时间,单位100纳秒LONGLONG DUR=0;//媒体时长,单位100纳秒DWORD AudioSamplePerSec=0;//音频样本采样率DWORD StreamCount=0;//流数量
};struct SAMPLE_INFO//样本信息
{BOOL B;//为TRUE,样本为第1个样本DWORD STAR;//运行开始时间,单位毫秒LONGLONG star;//样本开始时间,单位100纳秒LONGLONG end;//样本结束时间,单位100纳秒BYTE* pB;//样本缓冲区指针int len;//样本的字节大小HANDLE hRun;//“运行”事件句柄HANDLE hSeek;//“定位”事件句柄
};void SliderSetPosition(LONGLONG pos)
{
}int ReceiveVideoSample(SAMPLE_INFO SampleInfo)//视频样本接收函数
{return 0;
}int ReceiveAudioSample(SAMPLE_INFO SampleInfo)//音频样本接收函数
{return 0;
}
定义函数指针:
typedef int(__thiscall *MYPROC_ReceiveSample)(SAMPLE_INFO info);typedef void(__thiscall *MYPROC_GetPos)(LONGLONG pos);typedef INFO(__cdecl *MYPROC_SReader_Init)(wchar_t* Path);typedef int(__cdecl *MYPROC_SReader_Run)(MYPROC_ReceiveSample VideoSample, MYPROC_ReceiveSample AudioSample, MYPROC_GetPos Pos);typedef int(__cdecl *MYPROC_SReader_Pause)();typedef int(__cdecl *MYPROC_SReader_Stop)();typedef int(__cdecl *MYPROC_SReader_Seek)(LONGLONG SeekPos);typedef int(__cdecl *MYPROC_SReader_GetState)();typedef int(__cdecl *MYPROC_SReader_Exit)();
获取本读取器导出函数的地址:
MYPROC_SReader_Init SReader_Init = NULL;MYPROC_SReader_GetState SReader_GetState = NULL;MYPROC_SReader_Run SReader_Run = NULL;MYPROC_SReader_Pause SReader_Pause = NULL;MYPROC_SReader_Stop SReader_Stop = NULL;MYPROC_SReader_Seek SReader_Seek = NULL;MYPROC_SReader_Exit SReader_Exit = NULL;MYPROC_GetPos GetPos=NULL;MYPROC_ReceiveSample ReceiveVideo=NULL;MYPROC_ReceiveSample ReceiveAudio=NULL;if (hSReader){SReader_Init = (MYPROC_SReader_Init)GetProcAddress(hSReader, "Init");SReader_GetState = (MYPROC_SReader_GetState)GetProcAddress(hSReader, "GetState");SReader_Run = (MYPROC_SReader_Run)GetProcAddress(hSReader, "Run");SReader_Pause = (MYPROC_SReader_Pause)GetProcAddress(hSReader, "Pause");SReader_Stop = (MYPROC_SReader_Stop)GetProcAddress(hSReader, "Stop");SReader_Seek = (MYPROC_SReader_Seek)GetProcAddress(hSReader, "Seek");SReader_Exit = (MYPROC_SReader_Exit)GetProcAddress(hSReader, "Exit");GetPos = (MYPROC_GetPos)(&SliderSetPosition); ReceiveVideo=(MYPROC_ReceiveSample)(&ReceiveVideoSample);ReceiveAudio=(MYPROC_ReceiveSample)(&ReceiveAudioSample);}
初始化本读取器,获取文件媒体信息:
INFO info;wchar_t* path = L"某视频文件.mp4";if (SReader_Init != NULL)info = SReader_Init(path);
为读取器指定视频样本和音频样本接收函数,和接收位置信息函数(不需要位置信息时,GetPos可以为NULL);运行读取器:
if (SReader_Run){SReader_Run(ReceiveVideo, ReceiveAudio, GetPos);}
这样视频音频样本接收函数将被反复调用,参数提供样本的数据和信息。其它导出函数用于读取器的暂停,停止和退出;获取读取器当前状态,调用SReader_GetState(),返回状态值。
本读取器为后续文章“播放视频”而设计,读者可以在本代码的基础上添加自己想要的功能,也可以进行更改。
本读取器的全部代码
#include "windows.h"
#include "mmsystem.h"
#pragma comment(lib, "winmm")
#include "mfapi.h"
#include "mfidl.h"
#include "mfreadwrite.h"
#pragma comment(lib, "mfplat")
#pragma comment(lib, "mfreadwrite")
#pragma comment(lib, "mfuuid")template <class T> void SafeRelease(T** ppT)
{if (*ppT){(*ppT)->Release();*ppT = NULL;}
}struct INFO//媒体信息
{int VideoWidth;//视频宽,单位像素int VideoHeight;//视频高,单位像素LONGLONG FrameCur;//帧持续时间,单位100纳秒LONGLONG DUR;//媒体时长,单位100纳秒DWORD AudioSamplePerSec;//音频样本采样率DWORD StreamCount = 0;//流数量
};struct SAMPLE_INFO//样本信息
{BOOL B;//为TRUE,样本为第1个样本DWORD STAR;//运行开始时间,单位毫秒LONGLONG star;//样本开始时间,单位100纳秒LONGLONG end;//样本结束时间,单位100纳秒BYTE* pB;//样本缓冲区指针int len;//样本的字节大小HANDLE hRun;//“运行”事件句柄HANDLE hSeek;//“定位”事件句柄
};typedef int(__cdecl *MYPROC_SendVideo)(SAMPLE_INFO SampleInfo);//定义函数指针
typedef int(__cdecl *MYPROC_SendAudio)(SAMPLE_INFO SampleInfo);
typedef int(__cdecl *MYPROC_SendPos)(LONGLONG pos);class SReader
{
public:SReader(){HRESULT hr = MFStartup(MF_VERSION);if (hr != S_OK){MessageBox(NULL, L"初始化媒体基础失败", L"SourceReader", MB_OK);}hRun = CreateEvent(NULL, TRUE, FALSE, NULL);//手动重置,初始无信号hExit = CreateEvent(NULL, TRUE, FALSE, NULL);//手动重置,初始无信号hSeek = CreateEvent(NULL, TRUE, FALSE, NULL);//手动重置,初始无信号}~SReader(){SafeRelease(&pIMFSourceReader); CloseHandle(hRun); CloseHandle(hExit); CloseHandle(hSeek);MFShutdown();//关闭媒体基础}HANDLE hRun, hExit, hSeek;IMFSourceReader* pIMFSourceReader = NULL;//源读取器接口指针UINT32 Width, Height;//视频宽高LONGLONG DUR;//媒体持续时间,100纳秒单位LONGLONG CUR;//当前位置,100纳秒单位LONGLONG SeekPos;//定位位置DWORD STAR;//开始时间BOOL VFirst, AFirst;//为TRUE标明第1个样本int mState;//状态;0停止,1运行,2暂停INFO info;//媒体信息对象MYPROC_SendVideo SendVideo = NULL;//视频样本输出函数指针MYPROC_SendAudio SendAudio = NULL;//音频样本输出函数指针MYPROC_SendPos SendPos = NULL;//当前位置输出函数指针
};HRESULT GetSample(IMFSourceReader* pIMFSourceReader, UINT SOURCE, LONGLONG& Cur, LONGLONG& Dur)//获取样本
{IMFSample* pMFSample = NULL; DWORD flags;HRESULT hr = pIMFSourceReader->ReadSample(SOURCE, 0, NULL, &flags, NULL, &pMFSample);//读取视频样本hr = pMFSample->GetSampleTime(&Cur);//获取显示时间hr = pMFSample->GetSampleDuration(&Dur);//获取持续时间SafeRelease(&pMFSample);//释放接口return hr;
}SReader* pSReader = NULL;//SReader类对象指针
HANDLE hVthread = NULL, hAthread = NULL;//视频音频线程句柄DWORD WINAPI VideoThread(LPVOID lp)//视频线程
{HRESULT hr; pSReader->VFirst = TRUE; IMFSample* pMFSample = NULL; DWORD flags;
Agan:DWORD mRun = WaitForSingleObject(pSReader->hRun, 0);//检测“运行”信号,不等待DWORD mSeek = WaitForSingleObject(pSReader->hSeek, 0);//检测“定位”信号,不等待DWORD mExit = WaitForSingleObject(pSReader->hExit, 0);//检测“退出”信号,不等待if (mExit == WAIT_OBJECT_0)//有“退出”信号{return 0;//退出线程}if (mSeek == WAIT_OBJECT_0)//有“定位”信号{hr = pSReader->pIMFSourceReader->Flush(MF_SOURCE_READER_ALL_STREAMS);//丢弃所有排队的样本并取消所有挂起的样本请求PROPVARIANT pv;PropVariantInit(&pv);pv.vt = 20;pv.hVal.QuadPart = pSReader->SeekPos;hr = pSReader->pIMFSourceReader->SetCurrentPosition(GUID_NULL, pv);//更改源读取器位置PropVariantClear(&pv);/*试验发现,源读取器在媒体中间时间段定位后,音频视频样本不同步(时间戳相差约2秒钟),故添加下面代码以使样本时间同步*/LONGLONG VCur, VDur;hr = GetSample(pSReader->pIMFSourceReader, MF_SOURCE_READER_FIRST_VIDEO_STREAM, VCur, VDur);//获取视频样本时间LONGLONG ACur, ADur;hr = GetSample(pSReader->pIMFSourceReader, MF_SOURCE_READER_FIRST_AUDIO_STREAM, ACur, ADur);//获取音频样本时间if (ACur > VCur + VDur + 100000)//如果音频样本时间过大{while (ACur > VCur + VDur + 100000){hr = GetSample(pSReader->pIMFSourceReader, MF_SOURCE_READER_FIRST_VIDEO_STREAM, VCur, VDur);//获取下一个视频样本}}else if (VCur > ACur + ADur + 100000)//如果视频样本时间过大{while (VCur > ACur + ADur + 100000){hr = GetSample(pSReader->pIMFSourceReader, MF_SOURCE_READER_FIRST_AUDIO_STREAM, ACur, ADur);//获取下一个音频样本}}pSReader->STAR = timeGetTime();//记录运行开始时间,单位毫秒pSReader->VFirst = TRUE; pSReader->AFirst = TRUE;//设置第1个样本标记为TRUEResetEvent(pSReader->hSeek);//设置“定位”无信号}if (mRun == WAIT_OBJECT_0)//有“运行”信号{hr = pSReader->pIMFSourceReader->ReadSample(MF_SOURCE_READER_FIRST_VIDEO_STREAM, 0, NULL, &flags, NULL, &pMFSample);//读取视频样本if (hr == S_OK && pMFSample){LONGLONG Cur, Dur;hr = pMFSample->GetSampleTime(&Cur);//获取显示时间hr = pMFSample->GetSampleDuration(&Dur);//获取持续时间DWORD Lt;hr = pMFSample->GetTotalLength(&Lt);//获取有效长度DWORD count;hr = pMFSample->GetBufferCount(&count);//获取缓冲区数量IMFMediaBuffer* pMFBuffer = NULL;if (count == 1)//如果只有1个缓冲区{hr = pMFSample->GetBufferByIndex(0, &pMFBuffer);}else//如果有多个缓冲区{hr = pMFSample->ConvertToContiguousBuffer(&pMFBuffer);}BYTE* pSB = NULL;hr = pMFBuffer->Lock(&pSB, NULL, NULL);//锁定缓冲区SAMPLE_INFO info;info.B = pSReader->VFirst;info.STAR = pSReader->STAR;info.star = Cur;info.end = Cur + Dur;info.pB = pSB;info.len = Lt;info.hRun = pSReader->hRun;info.hSeek = pSReader->hSeek;pSReader->SendVideo(info);//调用视频输出函数,输出样本if (pSReader->VFirst)pSReader->VFirst = FALSE;//只有第1个样本标记为TRUEhr = pMFBuffer->Unlock();//解锁缓冲区SafeRelease(&pMFBuffer); SafeRelease(&pMFSample);//释放接口}}goto Agan;
}DWORD WINAPI AudioThread(LPVOID lp)
{HRESULT hr; pSReader->AFirst = TRUE; IMFSample* pMFSample = NULL; DWORD flags; pSReader->CUR = 0; int Count = 0;
Agan:DWORD mRun = WaitForSingleObject(pSReader->hRun, 0);//检测“运行”信号,不等待DWORD mSeek = WaitForSingleObject(pSReader->hSeek, 0);//检测“定位”信号,不等待DWORD mExit = WaitForSingleObject(pSReader->hExit, 0);//检测“退出”信号,不等待if (mExit == WAIT_OBJECT_0)//有“退出”信号{return 0;//退出线程}if (mSeek == WAIT_OBJECT_0)//有“定位”信号{if (pSReader->info.StreamCount == 2)//如果有视频流和音频流{goto Agan;//阻塞(更改位置操作在视频线程中,音频线程等待操作完成)}else//如果只有音频流,在此更改源读取器位置{hr = pSReader->pIMFSourceReader->Flush(MF_SOURCE_READER_ALL_STREAMS);//丢弃所有排队的样本并取消所有挂起的样本请求PROPVARIANT pv;PropVariantInit(&pv);pv.vt = 20;pv.hVal.QuadPart = pSReader->SeekPos;hr = pSReader->pIMFSourceReader->SetCurrentPosition(GUID_NULL, pv);//更改源读取器位置PropVariantClear(&pv);pSReader->STAR = timeGetTime();//记录运行开始时间pSReader->AFirst = TRUE;ResetEvent(pSReader->hSeek);//设置“定位”无信号}}if (mRun == WAIT_OBJECT_0)//有“运行”信号{hr = pSReader->pIMFSourceReader->ReadSample(MF_SOURCE_READER_FIRST_AUDIO_STREAM, 0, NULL, &flags, NULL, &pMFSample);//读取音频样本if (hr == S_OK && pMFSample){LONGLONG Cur, Dur;hr = pMFSample->GetSampleTime(&Cur);//获取显示时间hr = pMFSample->GetSampleDuration(&Dur);//获取持续时间Count++;if (Count > 9)//每10个样本,发送一次当前位置{pSReader->CUR = Cur;//赋值当前时间if(pSReader->SendPos)pSReader->SendPos(Cur);//发送当前时间位置值Count = 0;}DWORD Lt;hr = pMFSample->GetTotalLength(&Lt);//获取有效长度DWORD count;hr = pMFSample->GetBufferCount(&count);//获取缓冲区数量IMFMediaBuffer* pMFBuffer = NULL;if (count == 1)//如果只有1个缓冲区{hr = pMFSample->GetBufferByIndex(0, &pMFBuffer);}else//如果有多个缓冲区{hr = pMFSample->ConvertToContiguousBuffer(&pMFBuffer);}BYTE* pSB = NULL;hr = pMFBuffer->Lock(&pSB, NULL, NULL);//锁定缓冲区SAMPLE_INFO info;info.B = pSReader->AFirst;info.STAR = pSReader->STAR;info.star = Cur;info.end = Cur + Dur;info.pB = pSB;info.len = Lt;info.hRun = pSReader->hRun;info.hSeek = pSReader->hSeek;pSReader->SendAudio(info);//调用音频样本输出函数,发送音频样本if (pSReader->AFirst)pSReader->AFirst = FALSE;//只有第1个样本标记为TRUEhr = pMFBuffer->Unlock();//解锁缓冲区SafeRelease(&pMFBuffer); SafeRelease(&pMFSample);//释放接口}}goto Agan;
}//下面是导出函数定义
#pragma warning(disable:4190)#ifdef __cplusplus // If used by C++ code,
extern "C" { // we need to export the C interface
#endif__declspec(dllexport) INFO __cdecl Init(wchar_t* Path){if (pSReader)//如果SReader对象已存在{SetEvent(pSReader->hExit);//设置“退出”有信号WaitForSingleObject(hVthread, INFINITE);//等待视频线程退出WaitForSingleObject(hAthread, INFINITE);//等待音频线程退出delete pSReader;//删除SReader对象}pSReader = new SReader();//创建SReader对象IMFAttributes* pIMFAttributes = NULL;HRESULT hr = MFCreateAttributes(&pIMFAttributes, 0);if (SUCCEEDED(hr)){hr = pIMFAttributes->SetUINT32(MF_READWRITE_ENABLE_HARDWARE_TRANSFORMS, (UINT32)1);//使用基于硬件的媒体基础转换}if (SUCCEEDED(hr)){hr = pIMFAttributes->SetUINT32(MF_SOURCE_READER_ENABLE_VIDEO_PROCESSING, (UINT32)1);//允许源读取器进行有限的视频处理}if (SUCCEEDED(hr)){hr = MFCreateSourceReaderFromURL(Path, pIMFAttributes, &pSReader->pIMFSourceReader);//创建源读取器}SafeRelease(&pIMFAttributes);PROPVARIANT var;PropVariantInit(&var);if (SUCCEEDED(hr)){hr = pSReader->pIMFSourceReader->GetPresentationAttribute(MF_SOURCE_READER_MEDIASOURCE, MF_PD_DURATION, &var);//获取媒体时间长度,100纳秒单位}if (SUCCEEDED(hr)){pSReader->info.DUR = pSReader->DUR = (LONGLONG)var.uhVal.QuadPart;}PropVariantClear(&var);IMFMediaType* pAudioMTA = NULL;if (SUCCEEDED(hr)){hr = MFCreateMediaType(&pAudioMTA);//创建空的媒体类型}if (SUCCEEDED(hr)){hr = pAudioMTA->SetGUID(MF_MT_MAJOR_TYPE, MFMediaType_Audio);//设置主要类型音频}if (SUCCEEDED(hr)){hr = pAudioMTA->SetGUID(MF_MT_SUBTYPE, MFAudioFormat_PCM);//设置子类型PCM}if (SUCCEEDED(hr)){hr = pAudioMTA->SetUINT32(MF_MT_AUDIO_BITS_PER_SAMPLE, (UINT32)16);//设置样本16位}if (SUCCEEDED(hr)){hr = pAudioMTA->SetUINT32(MF_MT_AUDIO_SAMPLES_PER_SECOND, (UINT32)48000);//设置样本采样率48000}if (SUCCEEDED(hr)){pSReader->info.AudioSamplePerSec = 48000;}if (SUCCEEDED(hr)){hr = pSReader->pIMFSourceReader->SetCurrentMediaType(MF_SOURCE_READER_FIRST_AUDIO_STREAM, NULL, pAudioMTA);//设置音频输出媒体类型(此时未提供媒体类型全部信息)}SafeRelease(&pAudioMTA);IMFMediaType* pAudioMT = NULL;if (SUCCEEDED(hr)){hr = pSReader->pIMFSourceReader->GetCurrentMediaType(MF_SOURCE_READER_FIRST_AUDIO_STREAM, &pAudioMT);//获取当前音频输出媒体类型(获取的媒体类型将包含全部信息)}UINT32 SamplePerSec;if (SUCCEEDED(hr)){hr = pAudioMT->GetUINT32(MF_MT_AUDIO_SAMPLES_PER_SECOND, &SamplePerSec);//获取采样率}SafeRelease(&pAudioMT);if (SUCCEEDED(hr)){pSReader->info.AudioSamplePerSec = SamplePerSec;}pSReader->info.StreamCount = 0;if (SUCCEEDED(hr)){ResetEvent(pSReader->hExit);//设置“退出”无信号pSReader->info.StreamCount++;//流数量hAthread = CreateThread(NULL, 0, AudioThread, pSReader, 0, NULL);//创建音频线程}IMFMediaType* pVideoMTV = NULL;if (SUCCEEDED(hr)){hr = MFCreateMediaType(&pVideoMTV);//创建空的媒体类型}if (SUCCEEDED(hr)){hr = pVideoMTV->SetGUID(MF_MT_MAJOR_TYPE, MFMediaType_Video);//设置主要类型为视频}if (SUCCEEDED(hr)){hr = pVideoMTV->SetGUID(MF_MT_SUBTYPE, MFVideoFormat_RGB32);//设置子类型RGB32}if (SUCCEEDED(hr)){hr = pSReader->pIMFSourceReader->SetCurrentMediaType(MF_SOURCE_READER_FIRST_VIDEO_STREAM, NULL, pVideoMTV);//设置视频输出媒体类型}SafeRelease(&pVideoMTV);if (SUCCEEDED(hr)){IMFMediaType* pVideoMT = NULL;if (SUCCEEDED(hr)){hr = pSReader->pIMFSourceReader->GetCurrentMediaType(MF_SOURCE_READER_FIRST_VIDEO_STREAM, &pVideoMT);//获取当前视频媒体类型(此时的媒体类型将包含全部信息)}if (SUCCEEDED(hr)){hr = MFGetAttributeSize(pVideoMT, MF_MT_FRAME_SIZE, &pSReader->Width, &pSReader->Height);//获取视频图像宽高}SafeRelease(&pVideoMT);if (SUCCEEDED(hr)){pSReader->info.VideoWidth = pSReader->Width; pSReader->info.VideoHeight = pSReader->Height; pSReader->info.StreamCount++;}hVthread = CreateThread(NULL, 0, VideoThread, pSReader, 0, NULL);//创建视频线程}Sleep(1000);//等待线程初始化完成return pSReader->info;}__declspec(dllexport) int __cdecl Run(MYPROC_SendVideo SendVideo, MYPROC_SendAudio SendAudio, MYPROC_SendPos SendPos)//运行{if (pSReader){pSReader->SendVideo = SendVideo; pSReader->SendAudio = SendAudio; pSReader->SendPos = SendPos;pSReader->STAR = timeGetTime();//记录运行开始时间pSReader->VFirst = TRUE; pSReader->AFirst = TRUE;SetEvent(pSReader->hRun);//设置“运行”有信号pSReader->mState = 1;//状态标记置1return 0;}return 1;}__declspec(dllexport) int __cdecl Pause(void)//暂停{if (pSReader){ResetEvent(pSReader->hRun);//设置“运行”无信号pSReader->mState = 2;//状态标记置2return 0;}return 1;}__declspec(dllexport) int __cdecl Stop(void)//停止{if (pSReader){ResetEvent(pSReader->hRun);//设置“运行”无信号HRESULT hr = pSReader->pIMFSourceReader->Flush(MF_SOURCE_READER_ALL_STREAMS);//丢弃所有排队的样本并取消所有挂起的样本请求Sleep(200);pSReader->SeekPos = 0;PROPVARIANT pv;PropVariantInit(&pv);pv.vt = 20;pv.hVal.QuadPart = 0;hr = pSReader->pIMFSourceReader->SetCurrentPosition(GUID_NULL, pv);//设置源读取器位置0PropVariantClear(&pv);pSReader->mState = 0;//状态标记置0return 0;}return 1;}__declspec(dllexport) int __cdecl Seek(LONGLONG SeekPos)//定位{if (pSReader){pSReader->SeekPos = SeekPos;//赋值定位位置SetEvent(pSReader->hSeek);//设置“定位”有信号return 0;}return 1;}__declspec(dllexport) int __cdecl GetState()//获取状态{if (pSReader){return pSReader->mState;}return -1;}__declspec(dllexport) int __cdecl Exit(void)//退出{if (pSReader){ResetEvent(pSReader->hRun);//设置“运行”无信号SetEvent(pSReader->hExit);//设置“退出”有信号WaitForSingleObject(hVthread, INFINITE);//等待视频线程已退出WaitForSingleObject(hAthread, INFINITE);//等待音频线程已退出delete pSReader; pSReader = NULL;//删除SReader对象return 0;}return 1;}#ifdef __cplusplus
}
#endif
下载本插件