/******************************************************************************* Copyright (c) 1983-2016 Advantech Co., Ltd. ******************************************************************************** THIS IS AN UNPUBLISHED WORK CONTAINING CONFIDENTIAL AND PROPRIETARY INFORMATION WHICH IS THE PROPERTY OF ADVANTECH CORP., ANY DISCLOSURE, USE, OR REPRODUCTION, WITHOUT WRITTEN AUTHORIZATION FROM ADVANTECH CORP., IS STRICTLY PROHIBITED. ================================================================================ REVISION HISTORY -------------------------------------------------------------------------------- $Log: $ -------------------------------------------------------------------------------- $NoKeywords: $ */ /****************************************************************************** * * Windows Example: * AsynchronousOneWaveformAO.cpp * * Example Category * AO * Description: * This example demonstrates how to use Asynchronous One Waveform AO function. * * Instructions for Running: * 1 Set the 'deviceDescription' for opening the device. * 2 Set the 'profilePath' to save the profile path of being initialized device. * 3 Set the 'channelStart' as the first channel for analog data output. * 4 Set the 'channelCount' to decide how many sequential channels to output analog data. * * I/O Connections Overview: * Please refer to your hardware reference manual. * ******************************************************************************/ #include #include #include #include "../inc/compatibility.h" #include "../../../inc/bdaqctrl.h" using namespace Automation::BDaq; //----------------------------------------------------------------------------------- // Configure the following parameters before running the demo //----------------------------------------------------------------------------------- #define ONE_WAVE_POINT_COUNT 2048 //define how many data to makeup a waveform period. int32 samples = ONE_WAVE_POINT_COUNT; #define deviceDescription L"DemoDevice,BID#0" const wchar_t* profilePath = L"../../profile/DemoDevice.xml"; int32 startChannel = 0; int32 channelCount = 10; enum WaveStyle{ Sine, Sawtooth, Square }; //function GenerateWaveform: generate one waveform for each selected analog data output channel ErrorCode GenerateWaveform( BufferedAoCtrl * ,int32 ,int32 , double * , int32 ,WaveStyle ); inline void waitAnyKey() { do{SLEEP(1);} while(!kbhit()); } // This function is used to deal with 'Stopped' Event. void BDAQCALL OnStoppedEvent(void * sender, BfdAoEventArgs * args, void * userParam) { printf("\nBufferedAO stopped: offset = %d, count = %d\n", args->Offset, args->Count); } int main(int argc, char* argv[]) { ErrorCode ret = Success; // Step 1: Create a 'BufferedAoCtrl' for buffered AO function. BufferedAoCtrl * bfdAoCtrl = BufferedAoCtrl::Create(); // Step 2: Set the notification event Handler by which we can known the state of operation effectively. bfdAoCtrl->addStoppedHandler(OnStoppedEvent, NULL); do { // Step 3: Select a device by device number or device description and specify the access mode. // in this example we use ModeWrit mode so that we can fully control the device, including configuring, sampling, etc. DeviceInformation devInfo(deviceDescription); ret = bfdAoCtrl->setSelectedDevice(devInfo); CHK_RESULT(ret); ret = bfdAoCtrl->setStreaming(false);// specify the running mode: one-buffered. CHK_RESULT(ret); ret = bfdAoCtrl->LoadProfile(profilePath);//Loads a profile to initialize the device. CHK_RESULT(ret); // Step 4: Set necessary parameters. ScanChannel * scanChannel = bfdAoCtrl->getScanChannel(); ret = scanChannel->setChannelStart(startChannel); CHK_RESULT(ret); ret = scanChannel->setChannelCount(channelCount); CHK_RESULT(ret); ret = scanChannel->setSamples(samples); CHK_RESULT(ret); // Step 5: Prepare the buffered AO. ret = bfdAoCtrl->Prepare(); CHK_RESULT(ret); // Generate waveform data int scaledWaveFormLen = channelCount*ONE_WAVE_POINT_COUNT*sizeof(double); double* scaledWaveForm = (double*)malloc(scaledWaveFormLen); if( NULL == scaledWaveForm ) { printf( "Insufficient memory available\n" ); break; } ret = GenerateWaveform( bfdAoCtrl,startChannel,channelCount, scaledWaveForm,(int32)(channelCount*ONE_WAVE_POINT_COUNT),Sine); CHK_RESULT(ret); ret = bfdAoCtrl->SetData(channelCount*ONE_WAVE_POINT_COUNT,scaledWaveForm); CHK_RESULT(ret); // Step 6: Start Asynchronous One Waveform AO, 'Asynchronous' means the method returns immediately // after the Buffered AO has been started. The StoppedHandler's 'BfdAoEvent' method will be called // after the Buffered AO is completed. printf("Asynchronous Buffered AO is in progress.\n"); printf("Please wait... any key to quit !\n"); ret = bfdAoCtrl->Start(); CHK_RESULT(ret); free(scaledWaveForm); // Step 7: Do anything you are interesting while the device is outputting data. do { // do something yourself ! SLEEP(1); }while(!kbhit()); // step 8: Stop the operation if it is running. ret = bfdAoCtrl->Stop(1); CHK_RESULT(ret); }while(false); // Step 9: Close device, release any allocated resource. bfdAoCtrl->Dispose(); // If something wrong in this execution, print the error code on screen for tracking. if(BioFailed(ret)) { wchar_t enumString[256]; AdxEnumToString(L"ErrorCode", (int32)ret, 256, enumString); printf("Some error occurred. And the last error code is 0x%X. [%ls]\n", ret, enumString); waitAnyKey();// Wait any key to quit ! } return 0; } ErrorCode GenerateWaveform(BufferedAoCtrl * bfdAoCtrl, int32 channelStart,int32 channelCount,double * waveBuffer,int32 SamplesCount,WaveStyle style) { ErrorCode ret = Success; int32 channel = 0; int32 channelCountMax = 0; int32 oneWaveSamplesCount = SamplesCount/channelCount; int32 i = 0; MathInterval ranges[64] ; ValueRange valRange; channelCountMax = bfdAoCtrl->getFeatures()->getChannelCountMax(); for(i = 0;i < channelCountMax ;i++ ) { valRange = bfdAoCtrl->getChannels()->getItem(i).getValueRange(); if ( V_ExternalRefBipolar == valRange || valRange == V_ExternalRefUnipolar ) { if (bfdAoCtrl->getFeatures()->getExternalRefAntiPolar()) { double referenceValue; if (valRange == V_ExternalRefBipolar) { referenceValue = bfdAoCtrl->getChannels()->getItem(i).getExtRefBipolar(); if (referenceValue >= 0) { ranges[i].Max = referenceValue; ranges[i].Min = 0 - referenceValue; } else { ranges[i].Max = 0 - referenceValue; ranges[i].Min = referenceValue; } } else { referenceValue = bfdAoCtrl->getChannels()->getItem(i).getExtRefUnipolar(); if (referenceValue >= 0) { ranges[i].Max = 0; ranges[i].Min = 0 - referenceValue; } else { ranges[i].Max = 0 - referenceValue; ranges[i].Min = 0; } } } else { double referenceValue; if (valRange == V_ExternalRefBipolar) { referenceValue = bfdAoCtrl->getChannels()->getItem(i).getExtRefBipolar(); if (referenceValue >= 0) { ranges[i].Max = referenceValue; ranges[i].Min = 0 - referenceValue; } else { ranges[i].Max = 0 - referenceValue; ranges[i].Min = referenceValue; } } else { referenceValue = bfdAoCtrl->getChannels()->getItem(i).getExtRefUnipolar(); if (referenceValue >= 0) { ranges[i].Max = referenceValue; ranges[i].Min = 0; } else { ranges[i].Max = 0; ranges[i].Min = referenceValue; } } } } else { ret = AdxGetValueRangeInformation( valRange,0,NULL,&ranges[i],NULL); if(BioFailed(ret)) { return ret; } } } //generate waveform data and put them into the buffer which the parameter 'waveBuffer' give in, the Amplitude these waveform for(i = 0; i < oneWaveSamplesCount; i++ ) { for( int32 j = channelStart; j < channelStart+channelCount; j++ ) { //pay attention to channel rollback(when startChannel+channelCount>chNumberMax+1) channel = j%channelCountMax; double amplitude = (ranges[channel].Max - ranges[channel].Min) / 2; double offset = (ranges[channel].Max + ranges[channel].Min) / 2; switch ( style) { case Sine: *waveBuffer++ = amplitude*(sin((double)i*2.0*( 3.14159 )/oneWaveSamplesCount )) + offset; break; case Sawtooth: if ((i >= 0) && (i < (oneWaveSamplesCount / 4.0))) { *waveBuffer++ = amplitude*( i/(oneWaveSamplesCount/4.0)) + offset; } else { if ((i >= (oneWaveSamplesCount / 4.0)) && (i < 3 * (oneWaveSamplesCount/4.0))) { *waveBuffer++ = amplitude* ((2.0*(oneWaveSamplesCount/4.0)-i)/(oneWaveSamplesCount/4.0)) + offset; } else { *waveBuffer++ = amplitude* ((i-oneWaveSamplesCount)/(oneWaveSamplesCount/4.0)) + offset; } } break; case Square: if ((i >= 0) && (i < (oneWaveSamplesCount / 2))) { *waveBuffer++ = amplitude * 1 + offset; } else { *waveBuffer++ = amplitude * (-1) + offset; } break; default: printf("invalid wave style,generate waveform error !"); ret = ErrorUndefined; } } } return ret; };