尚未崩坏 发表于 2025-3-10 15:41:24

Windows软件插件-音视频文件读取器

下载本插件
本插件读取音频和视频文件,输出音频样本和视频样本,音频样本为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个样本标记为TRUE
                ResetEvent(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个样本标记为TRUE
                        hr = 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个样本标记为TRUE
                        hr = 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;//状态标记置1
                        return 0;
                }
                return 1;
        }

        __declspec(dllexport) int __cdecl Pause(void)//暂停
        {
                if (pSReader)
                {
                        ResetEvent(pSReader->hRun);//设置“运行”无信号
                        pSReader->mState = 2;//状态标记置2
                        return 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);//设置源读取器位置0
                        PropVariantClear(&pv);
                        pSReader->mState = 0;//状态标记置0
                        return 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

下载本插件

免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!更多信息从访问主页:qidao123.com:ToB企服之家,中国第一个企服评测及商务社交产业平台。
页: [1]
查看完整版本: Windows软件插件-音视频文件读取器