Click here to Skip to main content
15,886,833 members
Articles / Desktop Programming / MFC
Article

Converting Wav file to MP3 or other format using DirectShow

Rate me:
Please Sign up or sign in to vote.
4.84/5 (49 votes)
30 Jul 2002CPOL3 min read 700.9K   22.2K   188   128
Simple class to convert stereo 44 kHz, 16 bit wav file to another format, including MP3. The class shows how to use DirectShow API for audio conversion.

Sample Image - DShowEncoder.gif

Introduction

This article explains how to use the DirectShow API for simple audio conversion, particularly Wav to MP3 conversion. Audio codecs in the DirectShow API are of three type : native codecs, ACM codecs, and DMO (DirectX Media Object) codecs.

There are only few audio native codecs for audio compression. For MP3 encoding, the only one that I've found is the LAME DirectShow wrapper from Elecard. Most MP3 encoders are in the ACM (Audio Compression Manager) format, wich was introduced with the Windows Multimedia API. CDSEncoder class and it's relative classes CDSCodec and CDSCodecFormat enumerate ACM codecs and their respective compression parameters, construct a graph and do the encoding.

The GraphBuilder and other filters

Image 2

The graph consist of five filters:

  • File source (async) for reading the input wav file,
  • WAV Parser for wav parsing,
  • ACM codec for audio compression (in this case : MP3 ACM, wrapped by the ACM Wrapper Filter),
  • WAV Dest, for wav output multiplexing,
  • File Writer for writing the output file.

Important note

The WAV Dest filter is not included in standard filters, but need to be compiled from the DirectX SDK (SDK_root\Samples\Multimedia\DirectShow\Filters\WavDest). For convenience, the compiled WAV Dest filter is included in the demo zip, but you have to register it by RegSrv32 wavdest.ax.

ACM codecs and the ACM Wrapper filter

All of the ACM codecs are listed in DirectShow in the Audio Compressors Filter Category (CLSID_AudioCompressorCategory) and cannot be instantiated directly. We have to use the Device Enumerator to use them.

Note : Depending of your configuration, several ACM codecs for a same format can be installed on your computer. This can be the case for MP3 codecs. You set priority or deactivate some of them by the use of control panel, as show in the following figure.

Codec configuration

The DeviceEnumerator or how to browse ACM codecs

The Device Enumerator must be used to retrieve an instance of an ACM codec. It returns the codecs list by the IEnumMoniker interface, so we can get the filter interface (IBaseFilter) by a call to IMoniker::BindToObject() and the filter name by a call to IMoniker::BindToStorage().

Configuring the ACM codec with IAMStreamConfig interface

Once the desired codec is instantiated, we can obtain an IBaseFilter interface for filter configuration. Since each IBaseFilter have one or more Pin, we have to search the output Pin by the use of the IEnumPins interface and IPin::QueryDirection() calls.
With the output Pin, we can query the IAMStreamConfig interface to configure the following property :

  • Numbers of channels,
  • Samples per second,
  • Average byte per second,
  • Bits per sample.

Note : For some codecs (including MP3), the call to IAMStreamConfig::SetFormat() must be after the graph rendering.

The classes

CDSEncoder

CDSEncoder assumes the following task :

  • Enumerate the Audio codecs (CLSID_AudioCompressorCategory),
  • Build, render, and run the graph.

class CDSEncoder : public CArray<CDSCodec*, CDSCodec*>
{
public:
  void BuildGraph(CString szSrcFileName, CString szDestFileName, 
    int nCodec, int nFormat);
  CDSEncoder();
  virtual ~CDSEncoder();

protected:
  void BuildCodecArray();
  HRESULT AddFilterByClsid(IGraphBuilder *pGraph, LPCWSTR wszName, 
    const GUID& clsid, IBaseFilter **ppF);
  BOOL SetFilterFormat(AM_MEDIA_TYPE* pStreamFormat, 
    IBaseFilter* pBaseFilter);

  IGraphBuilder *m_pGraphBuilder;
};

As CDSEncoder inherits from CArray, the collection of codecs is exposed by CArray methods with each codecs returned as CDSCodec object.

CDSCodec

CDSCodec assumes the following task :

  • Enumerate the codec supported parameters,
  • expose the codec name.

class CDSCodec : public CArray<CDSCodecFormat*, CDSCodecFormat*>
{
public:
  CDSCodec();
  virtual ~CDSCodec();

  CString m_szCodecName;
  IMoniker  *m_pMoniker;
  void BuildCodecFormatArray();
};

As CDSCodec inherits from CArray, the collection of codecs supported parameters is exposed by CArray methods with each parameters returned as CDSCodecFormat object.

CDSCodecFormat

CDSCodecFormat exposes the properties of one-codec parameters :

  • Number of channels,
  • Samples per second,
  • Bytes per second,
  • Bits per samples.

class CDSCodecFormat  
{
public:
  WORD BitsPerSample();
  DWORD BytesPerSec();
  DWORD SamplesPerSecond();
  WORD NumberOfChannels();
  CDSCodecFormat();
  virtual ~CDSCodecFormat();

public:
  AM_MEDIA_TYPE* m_pMediaType;
};

Known issues

Errors Checking

The article goal is to demonstrate the use of DirectShow for simple audio conversion. These classes are not as safe as they have to be. Please keep this in mind if you plan to use it in a production environment.

Source Wav format

There are no sampling conversion, so you can only generate 44 kHz output files if you use 44 kHz Wav.

Windows Media

Windows Media format can be used only with a certificate that can be obtained by the Windows Media SDK from Microsoft.

License

This article, along with any associated source code and files, is licensed under The Code Project Open License (CPOL)


Written By
Software Developer (Senior) G. LABOURE
France France
This member has not yet provided a Biography. Assume it's interesting and varied, and probably something to do with programming.

Comments and Discussions

 
GeneralRe: Anyone tried this under vista? Pin
furbo8-Jul-07 11:23
furbo8-Jul-07 11:23 
QuestionHow to convert .flv files to mp3 file ? Pin
amany13/11/200619-Nov-06 2:18
amany13/11/200619-Nov-06 2:18 
Generalconverting a recorded .wav file into 16 bit PCM format .wav file Pin
sl_krns27-Jul-06 19:48
sl_krns27-Jul-06 19:48 
GeneralSmail Pin
smailmustafa18-May-06 4:52
smailmustafa18-May-06 4:52 
Generala question about input and out put Pin
hobin092014-Mar-06 21:50
hobin092014-Mar-06 21:50 
GeneralLincence for using this code Pin
Avikram2-Jan-06 19:04
Avikram2-Jan-06 19:04 
GeneralCDA To MP3 Pin
CaneMascherato29-May-05 20:37
CaneMascherato29-May-05 20:37 
GeneralRe: CDA To MP3 [modified] Pin
Yuriy Zanichkovskyy5-Jan-09 21:12
Yuriy Zanichkovskyy5-Jan-09 21:12 
GeneralError connecting the MP3 Filter with the WaveDest Filter Pin
xguido26-May-05 3:08
xguido26-May-05 3:08 
QuestionHow to convert Wav data from buffer instead of file Pin
serup18-May-05 23:24
serup18-May-05 23:24 
QuestionHow to convert mp3 to wav ??? Pin
chanwoopark6-May-05 18:42
chanwoopark6-May-05 18:42 
AnswerRe: How to convert mp3 to wav ??? Pin
Yulianto.6-May-05 19:13
Yulianto.6-May-05 19:13 
QuestionRe: How to convert mp3 to wav ??? Pin
koala912-Apr-06 21:32
koala912-Apr-06 21:32 
General.NET version Pin
zuken2122-Mar-05 20:24
zuken2122-Mar-05 20:24 
QuestionHow can I get progress indication ? Pin
furbo10-Mar-05 1:57
furbo10-Mar-05 1:57 
AnswerRe: How can I get progress indication ? Pin
furbo31-Mar-05 22:54
furbo31-Mar-05 22:54 
GeneralLexastudio Pin
Anonymous26-Oct-04 14:43
Anonymous26-Oct-04 14:43 
GeneralSOLVED DSHOW.H AND LINKING ERROR Pin
a_soul_reaver20-Oct-04 9:52
a_soul_reaver20-Oct-04 9:52 
GeneralRe: SOLVED DSHOW.H AND LINKING ERROR Pin
Guitool20-Oct-04 12:37
Guitool20-Oct-04 12:37 
QuestionDoes using an MP3 encoder/decoder built into Windows/Directshow require a licence from Thomson ? Pin
dogby18-Oct-04 15:02
dogby18-Oct-04 15:02 
AnswerRe: Does using an MP3 encoder/decoder built into Windows/Directshow require a licence from Thomson ? Pin
Guitool20-Oct-04 12:01
Guitool20-Oct-04 12:01 
Generalmp3 no format to choose Pin
driveway23-Sep-04 8:22
driveway23-Sep-04 8:22 
GeneralRe: mp3 no format to choose Pin
Guitool20-Oct-04 13:03
Guitool20-Oct-04 13:03 
Generalfor program coding Pin
Sunil Goswami11-Sep-04 0:37
Sunil Goswami11-Sep-04 0:37 
GeneralRe: for program coding Pin
Guitool20-Oct-04 13:00
Guitool20-Oct-04 13:00 

General General    News News    Suggestion Suggestion    Question Question    Bug Bug    Answer Answer    Joke Joke    Praise Praise    Rant Rant    Admin Admin   

Use Ctrl+Left/Right to switch messages, Ctrl+Up/Down to switch threads, Ctrl+Shift+Left/Right to switch pages.