Click here to Skip to main content
15,889,200 members
Articles / Desktop Programming / MFC
Article

Simple Audio Out Oscilloscope and Spectrum Analyzer

Rate me:
Please Sign up or sign in to vote.
4.71/5 (37 votes)
26 Jul 2006CPOL2 min read 387K   30.8K   192   75
An example of how to access the Audio Mix, sample audio output, and a simple DSP processing.

Sample Image

Introduction

This is a simple application I built to understand the basics of the Windows Audio WAVE API, and to get a basic grasp on Fast Fourier Transforms (FFTs). Features of the application are: double buffering graphics, sampling an audio source, selecting an audio recording source using the mixer, calculating the FFT spectrum array, and plotting sampled data.

Background

I won't discus the details of FFT here as the math is a little beyond me. Basically, it involves complex numbers (those number like sqrt(-1)) and calculating discrete Sin/Cos transformations on sampled date. Google for many and better explanations on FFT.

Using the code

If you look through the example, you will find it quite a simple affair that has not been touched by MFC. This means that the code is easy to follow and easy to transplant into your projects. The basic flow of the program is:

  • Load a dialog window and initialize it.
  • Select the recording device (which is harder than it should be!).
  • Initialize the WAVE structures.
  • Activate the sampling.
  • Determine the status of the sampling.
  • Process the sampled data using FFT.
  • Clean up.

In the FFT process, I load the sampled data into an iterator class for processing within the FFT class. This allow for selective processing of the sampled data. In this case, I sampled the recording source in stereo, and I can selectively process the left or the right channels within the FFT class, or both channels at once. Both classes are simple to follow and to implement.

Points of interest

The audio device API within Windows is far more complex than it should be. This is pretty common when dealing with Microsoft, they never seem to make things easy! A classic example is dealing with the Audio Mixer. It took some time to find some code that can do the most basic of tasks (select a recording device), and even that is not 100% perfect. Device drivers may incorrectly label a recording device as a WAVEOUT device and the program will return an error when trying to record from it, so look out for such errors (use the VC6 debugger to determine the fault in such occurrences).

In fact, I would not recommend using the WAVE API for anything too complex as it tends to be a bit tricky to use, instead look at the Simple Directmedia Layer library, and specifically the mixer lib, which is a good example of making things simple and useful to the programmer.

The FFT implementation probably isn't the fastest around. I originally converted the source from a Java class, and then optimized the Cos/Sin processing to implement a look up table to speed things up significantly, and added a few helper methods; otherwise it has remained unchanged.

History

  • Updated the spectrum graphics to be more equalised.
  • Fixed a number of small bugs.

License

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


Written By
New Zealand New Zealand
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: Error unable to set recording WAVEOUT device Pin
AEN23-Dec-08 7:17
AEN23-Dec-08 7:17 
Generaluse many cpu percenge Pin
shallopquan13-Aug-07 21:08
shallopquan13-Aug-07 21:08 
GeneralRe: use many cpu percenge Pin
Steven De Toni15-Aug-07 18:40
Steven De Toni15-Aug-07 18:40 
QuestionSetInputDevice on Vista Pin
HouseSparrow6-Aug-07 23:06
HouseSparrow6-Aug-07 23:06 
QuestionUsing You Code Pin
eng_bite28-Jun-07 7:34
eng_bite28-Jun-07 7:34 
QuestionSetInputDevice: hMixer Undefined Value. Please help. Pin
jeffhcliu11-Jun-07 19:56
jeffhcliu11-Jun-07 19:56 
AnswerRe: SetInputDevice: hMixer Undefined Value. Please help. Pin
Steven De Toni12-Jun-07 10:47
Steven De Toni12-Jun-07 10:47 
QuestionRe: SetInputDevice: hMixer Undefined Value. Please help. [modified] Pin
jeffhcliu14-Jun-07 18:29
jeffhcliu14-Jun-07 18:29 
Hi Steven.

Thank you very much for your quick reply, really appreciate it. I am also really sorry about my late response, been swamped with other work lately.

I think I just know too little on this. I did review msdn and various sources in order to study your code and I think I understand it. But, maybe because it is based on a dialogbox, I just can't seem to incorporate it into what I already have.

"All" I need are the VALUES of the microphone level (MaxLeft and MaxRight in your code) and the spectrum (pMags[idx]). I have been tweaking and studying your code (and the relevant API calls) for weeks and I still can't get past the HMIXER bit, not to mention obtaining the values. I really don't to give this up but I really don't have that much time to spare for just this component. If it is not too much too ask, could you give me some pointers in how I can approach this?

Thanks again.

Regards,
Jeff


-- modified at 6:42 Friday 15th June, 2007

These are the parts of Steven's code I'm using:

// Audio Scope Written By Steven De Toni

#include "stdafx.h"
#include "aud.h"
#include <stdio.h>
#include <stdarg.h>
#include <windows.h>
#include <commctrl.h>
#include <mmsystem.h>

#include "FFTransform.h"
//#include "resource.h"

// --- Global Variables ---
HINSTANCE HInst = NULL;

void CALLBACK waveInProc(HWAVEIN hWaveIn, UINT uMsg, DWORD dwInstance,
DWORD dwParam1, DWORD dwParam2);

void OutputDebugMsg (const char* pszFormat, ...)
{
char buf[1024];
va_list arglist;

va_start(arglist, pszFormat);
vsprintf(buf, pszFormat, arglist);
va_end(arglist);

strcat(buf, "\n");
OutputDebugString(buf);
}

void OutputMsgBox (const char* pszFormat, ...)
{
char buf[1024];
va_list arglist;

va_start(arglist, pszFormat);
vsprintf(buf, pszFormat, arglist);
va_end(arglist);

strcat(buf, "\n");
MessageBox (NULL, buf, "", MB_ICONSTOP);
}


/*********************** PrintWaveErrorMsg() **************************
* Retrieves and displays an error message for the passed Wave In error
* number. It does this using mciGetErrorString().
*************************************************************************/

void PrintWaveErrorMsg(DWORD err, TCHAR * str)
{
char buffer[128];

OutputMsgBox ("ERROR 0x%08X: %s\r\n", err, str);

if (mciGetErrorString(err, &buffer[0], sizeof(buffer)))
{
OutputMsgBox ("%s\r\n", &buffer[0]);
}
else
{
OutputMsgBox ("0x%08X returned!\r\n", err);
}
}


/*
What a long winded way to select recording input!

MIXERLINE_COMPONENTTYPE_SRC_ANALOG
MIXERLINE_COMPONENTTYPE_SRC_AUXILIARY
MIXERLINE_COMPONENTTYPE_SRC_COMPACTDISC
MIXERLINE_COMPONENTTYPE_SRC_DIGITAL
MIXERLINE_COMPONENTTYPE_SRC_LINE
MIXERLINE_COMPONENTTYPE_SRC_MICROPHONE
MIXERLINE_COMPONENTTYPE_SRC_PCSPEAKER
MIXERLINE_COMPONENTTYPE_SRC_SYNTHESIZER
MIXERLINE_COMPONENTTYPE_SRC_TELEPHONE
MIXERLINE_COMPONENTTYPE_SRC_UNDEFINED
MIXERLINE_COMPONENTTYPE_SRC_WAVEOUT

*/
BOOL SetInputDevice (unsigned int inDev)
{
HMIXER hMixer = NULL;
int inDevIdx = -1;

if ((mixerOpen(&hMixer, 0, 0, NULL, MIXER_OBJECTF_MIXER)) != MMSYSERR_NOERROR)
{
return FALSE;
}

// get dwLineID
MIXERLINE mxl;
mxl.cbStruct = sizeof(MIXERLINE);
mxl.dwComponentType = MIXERLINE_COMPONENTTYPE_DST_WAVEIN;

if (mixerGetLineInfo((HMIXEROBJ)hMixer, &mxl, MIXER_OBJECTF_HMIXER | MIXER_GETLINEINFOF_COMPONENTTYPE) != MMSYSERR_NOERROR)
{
mixerClose (hMixer);
return FALSE;
}

// get dwControlID
MIXERCONTROL mxc;
MIXERLINECONTROLS mxlc;
DWORD dwControlType = MIXERCONTROL_CONTROLTYPE_MIXER;

mxlc.cbStruct = sizeof(MIXERLINECONTROLS);
mxlc.dwLineID = mxl.dwLineID;
mxlc.dwControlType = dwControlType;
mxlc.cControls = 0;
mxlc.cbmxctrl = sizeof(MIXERCONTROL);
mxlc.pamxctrl = &mxc;

if (mixerGetLineControls((HMIXEROBJ)hMixer, &mxlc, MIXER_OBJECTF_HMIXER | MIXER_GETLINECONTROLSF_ONEBYTYPE) != MMSYSERR_NOERROR)
{
// no mixer, try MUX
dwControlType = MIXERCONTROL_CONTROLTYPE_MUX;
mxlc.cbStruct = sizeof(MIXERLINECONTROLS);
mxlc.dwLineID = mxl.dwLineID;
mxlc.dwControlType = dwControlType;
mxlc.cControls = 0;
mxlc.cbmxctrl = sizeof(MIXERCONTROL);
mxlc.pamxctrl = &mxc;
if (mixerGetLineControls((HMIXEROBJ)hMixer, &mxlc, MIXER_OBJECTF_HMIXER | MIXER_GETLINECONTROLSF_ONEBYTYPE) != MMSYSERR_NOERROR)
{
mixerClose (hMixer);
return FALSE;
}
}

if (mxc.cMultipleItems <= 0)
{
mixerClose (hMixer);
return FALSE;
}

// get the index of the inDevice from available controls
MIXERCONTROLDETAILS_LISTTEXT* pmxcdSelectText = new MIXERCONTROLDETAILS_LISTTEXT[mxc.cMultipleItems];

if (pmxcdSelectText != NULL)
{
MIXERCONTROLDETAILS mxcd;

mxcd.cbStruct = sizeof(MIXERCONTROLDETAILS);
mxcd.dwControlID = mxc.dwControlID;
mxcd.cChannels = 1;
mxcd.cMultipleItems = mxc.cMultipleItems;
mxcd.cbDetails = sizeof(MIXERCONTROLDETAILS_LISTTEXT);
mxcd.paDetails = pmxcdSelectText;

if (mixerGetControlDetails ((HMIXEROBJ)hMixer, &mxcd, MIXER_OBJECTF_HMIXER | MIXER_GETCONTROLDETAILSF_LISTTEXT) == MMSYSERR_NOERROR)
{
// determine which controls the inputDevice source line
DWORD dwi;
for (dwi = 0; dwi < mxc.cMultipleItems; dwi++)
{
// get the line information
MIXERLINE mxl;
mxl.cbStruct = sizeof(MIXERLINE);
mxl.dwLineID = pmxcdSelectText[dwi].dwParam1;
if (mixerGetLineInfo ((HMIXEROBJ)hMixer, &mxl, MIXER_OBJECTF_HMIXER | MIXER_GETLINEINFOF_LINEID) == MMSYSERR_NOERROR && mxl.dwComponentType == inDev)
{
// found, dwi is the index.
inDevIdx = dwi;
// break;
}
}

}

delete []pmxcdSelectText;
}

if (inDevIdx < 0)
{
mixerClose (hMixer);
return FALSE;
}

// get all the values first
MIXERCONTROLDETAILS_BOOLEAN* pmxcdSelectValue = new MIXERCONTROLDETAILS_BOOLEAN[mxc.cMultipleItems];

if (pmxcdSelectValue != NULL)
{
MIXERCONTROLDETAILS mxcd;
mxcd.cbStruct = sizeof(MIXERCONTROLDETAILS);
mxcd.dwControlID = mxc.dwControlID;
mxcd.cChannels = 1;
mxcd.cMultipleItems = mxc.cMultipleItems;
mxcd.cbDetails = sizeof(MIXERCONTROLDETAILS_BOOLEAN);
mxcd.paDetails = pmxcdSelectValue;
if (mixerGetControlDetails((HMIXEROBJ)hMixer, &mxcd, MIXER_OBJECTF_HMIXER | MIXER_GETCONTROLDETAILSF_VALUE) == MMSYSERR_NOERROR)
{
// ASSERT(m_dwControlType == MIXERCONTROL_CONTROLTYPE_MIXER || m_dwControlType == MIXERCONTROL_CONTROLTYPE_MUX);

// MUX restricts the line selection to one source line at a time.
if (dwControlType == MIXERCONTROL_CONTROLTYPE_MUX)
{
ZeroMemory(pmxcdSelectValue, mxc.cMultipleItems * sizeof(MIXERCONTROLDETAILS_BOOLEAN));
}

// Turn on this input device
pmxcdSelectValue[inDevIdx].fValue = 0x1;

mxcd.cbStruct = sizeof(MIXERCONTROLDETAILS);
mxcd.dwControlID = mxc.dwControlID;
mxcd.cChannels = 1;
mxcd.cMultipleItems = mxc.cMultipleItems;
mxcd.cbDetails = sizeof(MIXERCONTROLDETAILS_BOOLEAN);
mxcd.paDetails = pmxcdSelectValue;
if (mixerSetControlDetails ((HMIXEROBJ)hMixer, &mxcd, MIXER_OBJECTF_HMIXER | MIXER_SETCONTROLDETAILSF_VALUE) != MMSYSERR_NOERROR)
{
delete []pmxcdSelectValue;
mixerClose (hMixer);
return FALSE;
}
}

delete []pmxcdSelectValue;
}

mixerClose (hMixer);
return TRUE;
}

// ***********************************************************************************

#define SPECSCOPEWIDTH 10

static BOOL inRecord = FALSE;
static HWAVEIN waveInHandle = NULL;
static WAVEHDR waveHeader[2];
static WAVEFORMATEX waveFormat;
static FFTransform* pFFTrans = NULL;
static FFTransform* pFFTransStereo = NULL;
static SampleIter* pSampleIter = NULL;
static RECT drawArea;

static HDC HdblDC = NULL;
static HBITMAP HdblOldBitmap = NULL;
static int rps = 1;

///////////////////////////////////////////////////////////////////////////////

Caud::Caud(){}

void Caud::StopRec(){
// Stop recording and tell the driver to unqueue/return all of our WAVEHDRs.
// The driver will return any partially filled buffer that was currently
// recording. Because we use waveInReset() instead of waveInStop(),
// all of the other WAVEHDRs will also be returned via MM_WIM_DONE too
waveInReset(waveInHandle);
waveInUnprepareHeader (waveInHandle, &waveHeader[0], sizeof(WAVEHDR));
waveInUnprepareHeader (waveInHandle, &waveHeader[1], sizeof(WAVEHDR));

waveInClose(waveInHandle);
VirtualFree(waveHeader[0].lpData, (waveHeader[0].dwBufferLength << 1), MEM_RELEASE);
inRecord = FALSE;

SetInputDevice (MIXERLINE_COMPONENTTYPE_SRC_MICROPHONE);

if (pFFTrans != NULL)
{
delete pFFTrans;
pFFTrans = NULL;
}

if (pFFTransStereo != NULL)
{
delete pFFTransStereo;
pFFTransStereo = NULL;
}

if (pSampleIter != NULL)
{
delete pSampleIter;
pSampleIter = NULL;
}
}

////////////////////////////////////////////////////////////////////////////////////////////////////
void Caud::StartRec() {

//InitCommonControls ();

while (1) {
// Set the recording input ...
//if (SetInputDevice (MIXERLINE_COMPONENTTYPE_SRC_WAVEOUT) == FALSE)
//{
if (SetInputDevice (MIXERLINE_COMPONENTTYPE_SRC_ANALOG) == FALSE)
{
if (SetInputDevice (MIXERLINE_COMPONENTTYPE_SRC_LAST) == FALSE)
{
OutputMsgBox ("Error unable to set recording WAVEOUT device");
break;
}
}
//}

MMRESULT err;

// Clear out both of our WAVEHDRs. At the very least, waveInPrepareHeader() expects the dwFlags field to be cleared
ZeroMemory(&waveHeader[0], sizeof(WAVEHDR) * 2);

// Initialize the WAVEFORMATEX for 16-bit, 44KHz, stereo. That's what I want to record
waveFormat.wFormatTag = WAVE_FORMAT_PCM;
waveFormat.nChannels = 2;
waveFormat.nSamplesPerSec = 44100;
waveFormat.wBitsPerSample = 16;
waveFormat.nBlockAlign = waveFormat.nChannels * (waveFormat.wBitsPerSample/8);
waveFormat.nAvgBytesPerSec = waveFormat.nSamplesPerSec * waveFormat.nBlockAlign;
waveFormat.cbSize = 0;

// Open the default WAVE In Device, specifying my callback. Note that if this device doesn't
// support 16-bit, 44KHz, stereo recording, then Windows will attempt to open another device
// that does. So don't make any assumptions about the name of the device that opens. After
// waveInOpen returns the handle, use waveInGetID to fetch its ID, and then waveInGetDevCaps
// to retrieve the actual name
if ((err = waveInOpen(&waveInHandle, WAVE_MAPPER, &waveFormat, (DWORD)waveInProc, 0, CALLBACK_FUNCTION )) != 0)
{
PrintWaveErrorMsg (err, "Can't open WAVE In Device!");
break;
}

// Allocate, prepare, and queue two buffers that the driver can use to record blocks of audio data.
// (ie, We're using double-buffering. You can use more buffers if you'd like, and you should do that
// if you suspect that you may lag the driver when you're writing a buffer to disk and are too slow
// to requeue it with waveInAddBuffer. With more buffers, you can take your time requeueing
// each).
//
// I'll allocate 2 buffers large enough to hold 2 seconds worth of waveform data at 44Khz. NOTE:
// Just to make it easy, I'll use 1 call to VirtualAlloc to allocate both buffers, but you can
// use 2 separate calls since the buffers do NOT need to be contiguous. You should do the latter if
// using many, large buffers
waveHeader[1].dwBufferLength = waveHeader[0].dwBufferLength = 512;
if (!(waveHeader[0].lpData = (char*)VirtualAlloc(0, (waveHeader[0].dwBufferLength << 1), MEM_COMMIT, PAGE_READWRITE)))
{
OutputMsgBox ("ERROR: Can't allocate memory for WAVE buffer!\n");
waveInClose(waveInHandle);
break;
}

// Fill in WAVEHDR fields for buffer starting address. We've already filled in the size fields above */
waveHeader[1].lpData = waveHeader[0].lpData + waveHeader[0].dwBufferLength;

// Leave other WAVEHDR fields at 0

// Prepare the 2 WAVEHDR's
if ((err = waveInPrepareHeader(waveInHandle, &waveHeader[0], sizeof(WAVEHDR))))
{
waveInClose(waveInHandle);
OutputMsgBox ("Error preparing WAVEHDR 1! -- %08X\n", err);
VirtualFree (waveHeader[0].lpData, (waveHeader[0].dwBufferLength << 1), MEM_RELEASE);
break;
}

if ((err = waveInPrepareHeader(waveInHandle, &waveHeader[1], sizeof(WAVEHDR))))
{
waveInClose(waveInHandle);
OutputMsgBox ("Error preparing WAVEHDR 2! -- %08X\n", err);
VirtualFree (waveHeader[0].lpData, (waveHeader[0].dwBufferLength << 1), MEM_RELEASE);
break;
}

// Queue first WAVEHDR (recording hasn't started yet)
if ((err = waveInAddBuffer(waveInHandle, &waveHeader[0], sizeof(WAVEHDR))))
{
waveInClose(waveInHandle);
OutputMsgBox ("Error queueing WAVEHDR 1! -- %08X\n", err);
VirtualFree (waveHeader[0].lpData, (waveHeader[0].dwBufferLength << 1), MEM_RELEASE);
break;
}

// Queue second WAVEHDR
if ((err = waveInAddBuffer(waveInHandle, &waveHeader[1], sizeof(WAVEHDR))))
{
waveInClose(waveInHandle);
OutputMsgBox ("Error queueing WAVEHDR 2! -- %08X\n", err);
VirtualFree (waveHeader[0].lpData, (waveHeader[0].dwBufferLength << 1), MEM_RELEASE);
break;
}

// Start recording
if ((err = waveInStart(waveInHandle)))
{
OutputMsgBox ("Error starting record! -- %08X\n", err);
waveInClose(waveInHandle);
VirtualFree (waveHeader[0].lpData, (waveHeader[0].dwBufferLength << 1), MEM_RELEASE);
break;
}

// prepare the DSP processing objects
pFFTrans = new FFTransform (waveFormat.nSamplesPerSec, waveHeader[0].dwBufferLength/(waveFormat.nChannels * (waveFormat.wBitsPerSample/8)));
pFFTransStereo = new FFTransform (waveFormat.nSamplesPerSec, waveHeader[0].dwBufferLength/(waveFormat.wBitsPerSample/8));
pSampleIter = new SampleIter();

inRecord = TRUE;
break;
}
}

/////////////////////////////////////////////////////////////////////////////////////////////////////////////////////

void CALLBACK waveInProc(HWAVEIN hWaveIn, UINT uMsg, DWORD dwInstance,
DWORD dwParam1, DWORD dwParam2) {

if (uMsg == WIM_DATA){

WAVEHDR* pHeader = (WAVEHDR*) dwParam1;
static int maxLeftSave = 0;
static int maxRightSave = 0;
static int refresh = 0;

refresh++;
if (pHeader->dwBytesRecorded > 0)
{

// OutputDebugMsg ("Recorded %d bytes\n", pHeader->dwBytesRecorded);
int idx = 0;
short int* pSamples = (short int*)pHeader->lpData;
int sampleCnt = pHeader->dwBytesRecorded/(sizeof(short int));

// ------------------------------------------------------------------

// ******* Calculate Power Meter *******
int lm = 0, rm = 0, maxLeft = 0, maxRight = 0;

for (idx = 0; idx < sampleCnt; idx += 2)
{
if ((lm = abs(pSamples[idx]) >> 6) > maxLeft)
maxLeft = lm;

if ((rm = abs(pSamples[idx+1]) >> 6) > maxRight)
maxRight = rm;
}

if (maxLeft < maxLeftSave)
{
if ((maxLeft = maxLeftSave - 4) < 0)
maxLeft = 0;

}

if (maxRight < maxRightSave)
{
if ((maxRight = maxRightSave - 4) < 0)
maxRight = 0;
}


maxLeftSave = maxLeft;
maxRightSave = maxRight;


}

if (inRecord) // Are we still recording?
{
// Yes. Now we need to requeue this buffer so the driver can use it for another block of audio
// data. NOTE: We shouldn't need to waveInPrepareHeader() a WAVEHDR that has already been prepared once
waveInAddBuffer(waveInHandle, pHeader, sizeof(WAVEHDR));
}
}
}


////////////////////////////////////////////////////////////////


I am only aiming to get the input value (MaxLeft and MaxRight) now. First I call StartRec() at the beginning and StopRec() at the end of my own program. I am probably something stupidly wrong here. Anyway, any help would be MUCH MUCH appreciated. Thank you lots.

Jeff
QuestionCould not download source file Pin
david_taipei7-Jun-07 21:56
david_taipei7-Jun-07 21:56 
QuestionHelp in Converting "Callback Scope". Thanks. Pin
jeffhcliu7-Jun-07 8:40
jeffhcliu7-Jun-07 8:40 
AnswerRe: Help in Converting "Callback Scope". Thanks. Pin
Steven De Toni7-Jun-07 13:37
Steven De Toni7-Jun-07 13:37 
GeneralRe: Help in Converting "Callback Scope". Thanks. Pin
jeffhcliu7-Jun-07 14:25
jeffhcliu7-Jun-07 14:25 
Generalsound card with 3 mixers Pin
kenadams11129-Mar-07 3:15
kenadams11129-Mar-07 3:15 
GeneralMore Advanced Version of Audio Scope Pin
Steven De Toni26-Mar-07 18:21
Steven De Toni26-Mar-07 18:21 
QuestionAny flowchart or psuedo code available for this project Pin
Vidyadhar_India26-Mar-07 16:58
Vidyadhar_India26-Mar-07 16:58 
AnswerRe: Any flowchart or psuedo code available for this project Pin
Steven De Toni26-Mar-07 18:18
Steven De Toni26-Mar-07 18:18 
Questionhow to get amplitude and frequency Pin
lowkt2-Mar-07 3:34
lowkt2-Mar-07 3:34 
QuestionStrange behaviour and Equalizer Pin
GianniGP29-Jan-07 0:09
GianniGP29-Jan-07 0:09 
AnswerRe: Strange behaviour and Equalizer Pin
GianniGP31-Jan-07 3:45
GianniGP31-Jan-07 3:45 
QuestionRe: Strange behaviour and Equalizer Pin
jeffhcliu9-Jun-07 10:17
jeffhcliu9-Jun-07 10:17 
GeneralWant to convert this in C# Pin
Indian Ocean7-Jan-07 3:13
Indian Ocean7-Jan-07 3:13 
GeneralRe: Want to convert this in C# Pin
Steven De Toni10-Jan-07 16:01
Steven De Toni10-Jan-07 16:01 
GeneralRe: Want to convert this in C# Pin
Tauhid Shaikh23-Sep-07 19:58
Tauhid Shaikh23-Sep-07 19:58 
GeneralRe: Want to convert this in C# Pin
osirisgothra7-Nov-07 2:37
osirisgothra7-Nov-07 2:37 
GeneralSmall bug for spectrum calculation Pin
mcanti2-Dec-06 1:19
mcanti2-Dec-06 1:19 

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.