moteur audio

moteur audio - C++ - Programmation

Marsh Posté le 14-03-2007 à 15:07:19    

bonjour  
 
j'ai ecrit un moteur audio temps reel pour faire du traitement de signal en direct. J'ai ajouté un SetThreadPriority pour chaque procédure In et Out mais ca ne change rien, quand la fenêtre est réduite ou restaurée, il y a une courte interruption du processus. Quelqu'un a une piste ?
 

Code :
  1. #include <windows.h>
  2. #include "stdafx.h"
  3. #include "resource.h"
  4. #include <mmsystem.h>
  5. #include <malloc.h>
  6. #include <stdio.h>
  7. #include <commctrl.h>
  8. #define BUFFER_SIZE 16384
  9. #define SAMPLERATE 44100
  10. #define CANAUX 2
  11. #define OCTETS 2
  12. HWND DlgMain;
  13. PBYTE Buffer;
  14. HWAVEIN hWaveIn;
  15. HWAVEOUT hWaveOut;
  16. PBYTE pBufferIn1, pBufferIn2, pBufferIn3, pBufferOut1, pBufferOut2, pBufferOut3;
  17. PWAVEHDR pWaveHdrIn1, pWaveHdrIn2, pWaveHdrIn3, pWaveHdrOut1, pWaveHdrOut2, pWaveHdrOut3;
  18. WAVEFORMATEX waveform;
  19. BOOL flag_in_open = FALSE;
  20. BOOL flag_out_open = FALSE;
  21. HANDLE WorkerThreadHandleIn;
  22. DWORD WorkerThreadIdIn;
  23. HANDLE WorkerThreadHandleOut;
  24. DWORD WorkerThreadIdOut;
  25. //****************************************************************************
  26. DWORD WINAPI WorkerThreadProcIn (void *Arg)
  27. {
  28.   SetThreadPriority (GetCurrentThread (), THREAD_PRIORITY_TIME_CRITICAL);
  29.   MSG Msg;
  30.   while (GetMessage (&Msg, NULL, 0, 0) == TRUE)
  31.    {
  32.     switch (Msg.message)
  33.      {
  34.       case MM_WIM_OPEN:
  35.        {
  36.         MessageBox(DlgMain,"In open","RTA",MB_OK);
  37.         flag_in_open = TRUE;
  38.         waveInAddBuffer(hWaveIn,pWaveHdrIn1,sizeof(WAVEHDR));
  39.         waveInAddBuffer(hWaveIn,pWaveHdrIn2,sizeof(WAVEHDR));
  40.         waveInAddBuffer(hWaveIn,pWaveHdrIn3,sizeof(WAVEHDR));
  41.         waveInStart(hWaveIn);   
  42.         break;
  43.        }
  44.       case MM_WIM_DATA:
  45.        {
  46.         WAVEHDR *Hdr = (WAVEHDR *)Msg.lParam;
  47.         CopyMemory(Buffer,Hdr->lpData,BUFFER_SIZE);
  48.         waveInAddBuffer (hWaveIn, Hdr, sizeof (*Hdr));
  49.         break;
  50.        }
  51.      }
  52.    }
  53.   return 0;  
  54. }
  55. //****************************************************************************
  56. DWORD WINAPI WorkerThreadProcOut (void *Arg)
  57. {
  58.   SetThreadPriority (GetCurrentThread (), THREAD_PRIORITY_TIME_CRITICAL);
  59.   MSG Msg;
  60.   while (GetMessage (&Msg, NULL, 0, 0) == TRUE)
  61.    {
  62.     switch (Msg.message)
  63.      {
  64.       case MM_WOM_OPEN:
  65.        {
  66.         MessageBox(DlgMain,"Out open","RTA",MB_OK);
  67.         flag_out_open = TRUE;
  68.         waveOutPrepareHeader(hWaveOut,pWaveHdrOut1,sizeof(WAVEHDR));
  69.         waveOutWrite(hWaveOut,pWaveHdrOut1,sizeof(WAVEHDR));
  70.         waveOutPrepareHeader(hWaveOut,pWaveHdrOut2,sizeof(WAVEHDR));
  71.         waveOutWrite(hWaveOut,pWaveHdrOut2,sizeof(WAVEHDR));
  72.         waveOutPrepareHeader(hWaveOut,pWaveHdrOut3,sizeof(WAVEHDR));
  73.         waveOutWrite(hWaveOut,pWaveHdrOut3,sizeof(WAVEHDR));
  74.         break;
  75.        }
  76.       case MM_WOM_DONE:
  77.        {
  78.         WAVEHDR *Hdr = (WAVEHDR *)Msg.lParam;
  79.         waveOutPrepareHeader (hWaveOut, Hdr, sizeof (*Hdr));
  80.         waveOutWrite (hWaveOut, Hdr, sizeof (*Hdr));
  81.         CopyMemory(Hdr->lpData,Buffer,BUFFER_SIZE);
  82.         break;
  83.        }
  84.      }
  85.    }
  86.   return 0;  
  87. }
  88. //****************************************************************************
  89. BOOL CALLBACK DlgProc(HWND hWnd, UINT msg, WPARAM wParam, LPARAM lParam)
  90. {   
  91.   static int volume_general;
  92.   if (msg == WM_INITDIALOG)
  93.    {
  94.     DlgMain = hWnd;
  95.     return 0;
  96.    }
  97.   if (msg == WM_COMMAND)
  98.    {
  99.     if (wParam == ID_START)
  100.      {
  101.       if(flag_in_open == TRUE)
  102.        {
  103.         MessageBox(hWnd,"périphérique déjà ouvert","Erreur",0);
  104.         return 0;
  105.        }
  106.        
  107.       WorkerThreadHandleIn = CreateThread (NULL, 0, WorkerThreadProcIn, NULL, 0, &WorkerThreadIdIn );
  108.       WorkerThreadHandleOut = CreateThread (NULL, 0, WorkerThreadProcOut, NULL, 0, &WorkerThreadIdOut );
  109.       pWaveHdrIn1 = (PWAVEHDR) malloc(sizeof(WAVEHDR));
  110.       pWaveHdrIn2 = (PWAVEHDR) malloc(sizeof(WAVEHDR));
  111.       pWaveHdrIn3 = (PWAVEHDR) malloc(sizeof(WAVEHDR));
  112.       pWaveHdrOut1 = (PWAVEHDR) malloc(sizeof(WAVEHDR));
  113.       pWaveHdrOut2 = (PWAVEHDR) malloc(sizeof(WAVEHDR));
  114.       pWaveHdrOut3 = (PWAVEHDR) malloc(sizeof(WAVEHDR));
  115.        
  116.       Buffer   = (PBYTE)malloc(BUFFER_SIZE);
  117.       pBufferIn1 = (PBYTE)malloc(BUFFER_SIZE);
  118.       pBufferIn2 = (PBYTE)malloc(BUFFER_SIZE);
  119.       pBufferIn3 = (PBYTE)malloc(BUFFER_SIZE);
  120.       pBufferOut1 = (PBYTE)malloc(BUFFER_SIZE);
  121.       pBufferOut2 = (PBYTE)malloc(BUFFER_SIZE);
  122.       pBufferOut3 = (PBYTE)malloc(BUFFER_SIZE);
  123.      
  124.       if(!pBufferIn1 || !pBufferIn2 || !pBufferIn3 || !pBufferOut1 || !pBufferOut2 || !pBufferOut3)
  125.        {
  126.         if(pBufferIn1) free (pBufferIn1);
  127.         if(pBufferIn2) free (pBufferIn2);
  128.         if(pBufferIn3) free (pBufferIn3);
  129.         if(pBufferOut1) free (pBufferOut1);
  130.         if(pBufferOut2) free (pBufferOut2);
  131.         if(pBufferOut3) free (pBufferOut3);
  132.         MessageBox(hWnd,"Erreur d'allocation de mémoire","Erreur",0);
  133.         return TRUE;
  134.        }     
  135.      
  136.       waveform.nChannels = CANAUX; // 1 pour mono 2 pour stereo
  137.       waveform.wBitsPerSample = 8 * OCTETS; // 8 ou 16 bit
  138.       waveform.nAvgBytesPerSec = SAMPLERATE * waveform.nChannels * waveform.wBitsPerSample/8; // nombre d'octets par seconde
  139.       waveform.wFormatTag = 1; // 1 pour PCM
  140.       waveform.nSamplesPerSec = SAMPLERATE; // frequence d'echantillonnage
  141.       waveform.nBlockAlign = 1;
  142.       waveform.cbSize = 0;
  143.       if(waveInOpen(&hWaveIn,WAVE_MAPPER,&waveform,(DWORD)WorkerThreadIdIn,0,CALLBACK_THREAD))
  144.        {
  145.         free(pBufferIn1);
  146.         free(pBufferIn2);
  147.         free(pBufferIn3);
  148.         MessageBox(hWnd,"ouverture du périphérique d'entrée impossible","WARNING",MB_OK);
  149.         return 0;
  150.        }
  151.       pWaveHdrIn1->lpData          = (LPSTR)pBufferIn1;
  152.       pWaveHdrIn1->dwBufferLength  = BUFFER_SIZE;
  153.       pWaveHdrIn1->dwBytesRecorded = 0;
  154.       pWaveHdrIn1->dwUser          = 0;
  155.       pWaveHdrIn1->dwFlags         = 0;
  156.       pWaveHdrIn1->dwLoops         = 0;
  157.       pWaveHdrIn1->lpNext          = NULL;
  158.       pWaveHdrIn1->reserved        = 0;     
  159.       waveInPrepareHeader(hWaveIn,pWaveHdrIn1,sizeof(WAVEHDR));
  160.      
  161.       pWaveHdrIn2->lpData          = (LPSTR)pBufferIn2;
  162.       pWaveHdrIn2->dwBufferLength  = BUFFER_SIZE;
  163.       pWaveHdrIn2->dwBytesRecorded = 0;
  164.       pWaveHdrIn2->dwUser          = 0;
  165.       pWaveHdrIn2->dwFlags         = 0;
  166.       pWaveHdrIn2->dwLoops         = 0;
  167.       pWaveHdrIn2->lpNext          = NULL;
  168.       pWaveHdrIn2->reserved        = 0;     
  169.       waveInPrepareHeader(hWaveIn,pWaveHdrIn2,sizeof(WAVEHDR));
  170.       pWaveHdrIn3->lpData          = (LPSTR)pBufferIn3;
  171.       pWaveHdrIn3->dwBufferLength  = BUFFER_SIZE;
  172.       pWaveHdrIn3->dwBytesRecorded = 0;
  173.       pWaveHdrIn3->dwUser          = 0;
  174.       pWaveHdrIn3->dwFlags         = 0;
  175.       pWaveHdrIn3->dwLoops         = 0;
  176.       pWaveHdrIn3->lpNext          = NULL;
  177.       pWaveHdrIn3->reserved        = 0;     
  178.       waveInPrepareHeader(hWaveIn,pWaveHdrIn3,sizeof(WAVEHDR));       
  179.       if(waveOutOpen(&hWaveOut,WAVE_MAPPER,&waveform,(DWORD)WorkerThreadIdOut,0,CALLBACK_THREAD))
  180.        {
  181.         free(pBufferOut1);
  182.         free(pBufferOut2);
  183.         free(pBufferOut3);
  184.         MessageBox(hWnd,"ouverture du périphérique de sortie impossible","WARNING",MB_OK);
  185.         return 0;
  186.        }     
  187.       pWaveHdrOut1->lpData = (LPSTR)pBufferOut1;
  188.       pWaveHdrOut1->dwBufferLength = BUFFER_SIZE;
  189.       pWaveHdrOut1->dwBytesRecorded = 0;
  190.       pWaveHdrOut1->dwUser = 0;
  191.       pWaveHdrOut1->dwFlags = 0;
  192.       pWaveHdrOut1->dwLoops = 0;
  193.       pWaveHdrOut1->lpNext = NULL;
  194.       pWaveHdrOut1->reserved = 0;     
  195.       waveOutPrepareHeader(hWaveOut,pWaveHdrOut1,sizeof(WAVEHDR));
  196.      
  197.       pWaveHdrOut2->lpData = (LPSTR)pBufferOut2;
  198.       pWaveHdrOut2->dwBufferLength = BUFFER_SIZE;
  199.       pWaveHdrOut2->dwBytesRecorded = 0;
  200.       pWaveHdrOut2->dwUser = 0;
  201.       pWaveHdrOut2->dwFlags = 0;
  202.       pWaveHdrOut2->dwLoops = 0;
  203.       pWaveHdrOut2->lpNext = NULL;
  204.       pWaveHdrOut2->reserved = 0;     
  205.       waveOutPrepareHeader(hWaveOut,pWaveHdrOut2,sizeof(WAVEHDR));
  206.      
  207.       pWaveHdrOut3->lpData =(LPSTR)pBufferOut3;
  208.       pWaveHdrOut3->dwBufferLength =BUFFER_SIZE;
  209.       pWaveHdrOut3->dwBytesRecorded =0;
  210.       pWaveHdrOut3->dwUser =0;
  211.       pWaveHdrOut3->dwFlags =0;
  212.       pWaveHdrOut3->dwLoops =0;
  213.       pWaveHdrOut3->lpNext =NULL;
  214.       pWaveHdrOut3->reserved =0;     
  215.       waveOutPrepareHeader(hWaveOut,pWaveHdrOut3,sizeof(WAVEHDR));       
  216.       return 0;
  217.      }
  218.     if (wParam == IDCANCEL)
  219.      {
  220.       if(MessageBox(hWnd, "Voulez-vous vraiment arrêter l'application en cours ?", "Real Time Audio",MB_YESNO|MB_ICONEXCLAMATION|MB_DEFBUTTON2) == IDNO)
  221.        {
  222.         return TRUE;
  223.        }
  224.       EndDialog(hWnd, 0);
  225.       return 0;
  226.      }
  227.    }
  228.   return 0;
  229. }
  230. //****************************************************************************
  231.  
  232. int APIENTRY WinMain(HINSTANCE hInstance, HINSTANCE hPrevInstance, LPSTR lpCmdLine, int nCmdShow)
  233. {
  234.   InitCommonControls();
  235.   DialogBox(hInstance, MAKEINTRESOURCE(IDD_DIALOG1),HWND_DESKTOP,DlgProc);
  236.   return 0;
  237. }
  238. //****************************************************************************

Reply

Marsh Posté le 14-03-2007 à 15:07:19   

Reply

Marsh Posté le 14-03-2007 à 18:58:48    

Hésite pas a augmenter ton buffer_size, je le fixera à 65536.
De plus, je ferais un sleep entre chaque filling de buffer tant que celui ci n'a pas son flag à WHDR_DONE.

Reply

Marsh Posté le 14-03-2007 à 20:53:39    

merci je vais essayer
65536 c'est super long comme retatd c'est plus vraiment du temps reel  :) Y' a pas mieux comme solution ?
 

karlkox a écrit :

Hésite pas a augmenter ton buffer_size, je le fixera à 65536.
De plus, je ferais un sleep entre chaque filling de buffer tant que celui ci n'a pas son flag à WHDR_DONE.


Reply

Marsh Posté le 14-03-2007 à 21:57:01    

Déjà, ce que tu peux faire, c'est un tableau de WAVEHDR, dans ton thread, tu incrémenteras une variable (à locker) qui indiquera le buffer en cours de traitement, tu pourras ainsi pointer sur le bon WAVEHDR, tu l'écrit sur le device puis tu attends que son flag soit sur WHDR_DONE.
C'est ce que fait la SDL (je me suis basé la dessus) et ça a réglé le même problème que le tient que j'ai eu. (mais j'ai du augmenter le buffer size, driver creative inside ...)

Reply

Marsh Posté le 14-03-2007 à 22:10:27    

C'est quoi le probleme avec les drivers creative rapport a la latence?

Reply

Marsh Posté le 14-03-2007 à 22:31:40    

Disons que Creative fait du bon matos mais que niveau ingénierie logiciel, ils peuvent mieux faire, ce n'est pas à la hauteur du matos qu'ils font en tout cas.

Reply

Marsh Posté le 14-03-2007 à 22:33:45    

bon bah moi en attendant je reécris comme tu as indiqué et je mettrais le nouveau code ici.

Reply

Marsh Posté le 14-03-2007 à 22:41:39    

J'aurais aime une reponse plus precise, car je fais de l'acquisition audio sur une carte creative avec une precision temporelle de l'ordre de la milliseconde avec un buffer de 96 echantillons, et ca marche plutot pas mal. Quelles sont les tares cachees du driver ?

Reply

Marsh Posté le 14-03-2007 à 22:56:31    

J'ai une Creative moi aussi et je tape dans du 2ms sans problème, ce n'est pas forcément un problème de latence mais un problème plus global : stabilité, grésillement possible, impossibilité de reprogrammer le spu sans passer par un sdk (qui n'est fournis que si on fait partis d'une grosse boite de dev).

Reply

Marsh Posté le 14-03-2007 à 23:00:01    

oui, maintenant que tu le dis, une fois sur dix environ, j'ai une acquisition qui foire, et a place du joli bip de reference que je suis cense enregistrer, j'ai des gresillements. Ca viendrait donc du driver ?

Reply

Marsh Posté le 14-03-2007 à 23:00:01   

Reply

Marsh Posté le 14-03-2007 à 23:11:15    

Le problème de grésillement est assez récurant chez Creative malheureusement SAUF sur la X-Fi Elite Pro (que j'ai).

Reply

Marsh Posté le 16-03-2007 à 21:41:57    

Bonjour  
 
Moi je ne comprends pas tres bien comment introduire CreateEvent,  WaitForSingleObject, SetEvent, WHDR_DONE etc... dans mon code.
 
Quelqu'un peut il m'aider svp ?
 
merci

Reply

Marsh Posté le 16-03-2007 à 23:24:40    

Tu as ce qu'il te faut ici, ça utilise le WaveIn mais ça s'utilise exactement de la même manière pour le WaveOut.

Reply

Marsh Posté le 16-03-2007 à 23:31:55    

est ce que j'ai le droit de traiter le MM_WIM_DATA et le MM_WOM_DONE par le meme thread ?

Reply

Marsh Posté le 16-03-2007 à 23:37:24    

le directsound serait pas moins pire que le WaveOut ?
 
parceque quand j'entends parler de grésillements, et que je vois que c'est une approche par message, j'ai peur :D

Reply

Marsh Posté le 17-03-2007 à 07:16:21    

bjone a écrit :

le directsound serait pas moins pire que le WaveOut ?
 
parceque quand j'entends parler de grésillements, et que je vois que c'est une approche par message, j'ai peur :D


Y'a aussi possibilite d'utiliser des vrais callbacks dans WaveOut  

Reply

Marsh Posté le 17-03-2007 à 07:18:11    

Ah bon ? ca peut etre interessant que tu modifies mon code pour montrer comment
 
merci

Reply

Marsh Posté le 17-03-2007 à 07:25:40    

La fonction waveOutOpen accepte comme dernier parametre  
CALLBACK_FUNCTION au lieu de CALLBACK_THREAD. cf la msdn

Reply

Marsh Posté le 17-03-2007 à 07:28:05    

oui je connais CALLBACK_FUNCTION mais le probleme est d'introduire et gerer le partage des ressources avec CreateEvent, WaitForSingleObject, SetEvent, WHDR_DONE etc.

Reply

Marsh Posté le 17-03-2007 à 11:26:45    

Je sais pas si c'est adéquat, mais il y a moyen de trouver des drivers ASIO pour les creatives sur le net...
http://forum.hardware.fr/hfr/Video [...] 8445_1.htm


Message édité par breizhbugs le 17-03-2007 à 11:28:57
Reply

Marsh Posté le 17-03-2007 à 13:54:13    

gabuzomeuh2 a écrit :

est ce que j'ai le droit de traiter le MM_WIM_DATA et le MM_WOM_DONE par le meme thread ?


 
Oui bien sûr, aucun problème.
 
 
bjone>ce n'est pas tant un problème de driver de sortie son mais un problème d'implémentation de celui ci, il suffit de faire abstraction du driver avant de s'attaquer vraiment à la partie API sonore (dsound, winmm, oal ...).
Perso, j'utilise les threads et aucun problème, ça m'a aussi permis de ne plus avoir ce problème de grésillement qui est dû à une mauvaise synchronisation entre la notification windows et le buffer à remplir.
 
 
breizhbugs>les pilotes ASIO font partis intégrante des pilotes. (CTASIO)


Message édité par karlkox le 17-03-2007 à 13:54:39
Reply

Marsh Posté le 25-03-2007 à 05:35:05    

Voila j'ai tout réecrit. On dirait que ca marche.
Tu peux me dire ce que tu en penses KarLKox ?
 

Code :
  1. #include <windows.h>
  2. #include "stdafx.h"
  3. #include "resource.h"
  4. #include <mmsystem.h>
  5. #include <malloc.h>
  6. #define TAILLE_BLOC   4096
  7. #define NB_BUFFER     3
  8. #define CANAUX        2
  9. #define OCTETS        2
  10. #define BUFFER_SIZE   CANAUX * OCTETS * TAILLE_BLOC
  11. #define SAMPLERATE    44100
  12. BOOL OpenedIn = FALSE;
  13. HANDLE recordDone;
  14. HWAVEIN hWaveIn;
  15. DWORD threadidIn;
  16. HANDLE RecordHandle;
  17. PWAVEHDR pWaveHeaderIn[NB_BUFFER];
  18. PBYTE pBufferIn[NB_BUFFER];
  19. BOOL OpenedOut = FALSE;
  20. HANDLE playDone;
  21. HWAVEOUT hWaveOut;
  22. DWORD threadidOut;
  23. HANDLE PlayHandle;
  24. PWAVEHDR pWaveHeaderOut[NB_BUFFER];
  25. PBYTE pBufferOut[NB_BUFFER];
  26. char Buffer[BUFFER_SIZE];
  27. HWND DlgMain;
  28. CRITICAL_SECTION CriticalSection;
  29. //****************************************************************************
  30. void copier_buffer(char dest[], char src[], long taille)
  31. {
  32.   __asm
  33.    {
  34.        mov  ecx,   taille
  35.        cmp  ecx,   0
  36.        je   fin
  37.        mov  edi,   dest
  38.        mov  esi,   src
  39. debut: mov  al,    [esi]
  40.        mov  [edi], al
  41.        inc  esi
  42.        inc  edi
  43.        loop debut
  44. fin:       
  45.    }
  46. }
  47. //****************************************************************************
  48. void CALLBACK waveInProc(HWAVEIN hwi, UINT uMsg, DWORD dwInstance, DWORD dwParam1, DWORD dwParam2)
  49. {
  50.   if (uMsg == WIM_OPEN)
  51.    {
  52.     MessageBox(DlgMain,"WIM_OPEN","RTA",MB_OK);
  53.     OpenedIn = TRUE;
  54.    }
  55.   if (uMsg == WIM_CLOSE)
  56.    {
  57.     MessageBox(DlgMain,"WIM_CLOSE","RTA",MB_OK);
  58.    }
  59.   if (uMsg == WIM_DATA)  
  60.    {
  61.     SetEvent(recordDone);     
  62.    }
  63. }
  64. //****************************************************************************
  65. // 01 WHDR_DONE      indique que c'est fini avec ce buffer.
  66. // 02 WHDR_PREPARED  indique que le buffer a ete prepare (waveInPrepareHeader ou waveOutPrepareHeader)
  67. // 04 WHDR_BEGINLOOP indique que ce buffer est le premier dans une boucle, drapeau utilisé seulement en sortie.
  68. // 08 WHDR_ENDLOOP   indique que ce buffer est le dernier dans une boucle. Ce drapeau est utilisé seulement en sortie.
  69. // 10 WHDR_INQUEUE   indique que le buffer est mis en queue pour la sortie.
  70. //****************************************************************************
  71. DWORD WINAPI InThreadProc(LPVOID lpParam)
  72. {   
  73.   int i=0;
  74.   for(;;)
  75.    {
  76.     WaitForSingleObject(recordDone, INFINITE);
  77.     if (!(pWaveHeaderIn[i]->dwFlags & WHDR_DONE)) continue;
  78.      
  79.     EnterCriticalSection(&CriticalSection );
  80.      
  81.     copier_buffer(Buffer, pWaveHeaderIn[i]->lpData, BUFFER_SIZE); // sauvegarder bloc recu
  82.      
  83.     LeaveCriticalSection(&CriticalSection );
  84.     waveInUnprepareHeader(hWaveIn, pWaveHeaderIn[i], sizeof(WAVEHDR));
  85.     pWaveHeaderIn[i]->dwFlags = 0;
  86.     waveInPrepareHeader(hWaveIn, pWaveHeaderIn[i], sizeof(WAVEHDR));
  87.     waveInAddBuffer(hWaveIn, pWaveHeaderIn[i], sizeof(WAVEHDR));
  88.     __asm
  89.      {
  90.       mov eax, i
  91.       mov ebx, NB_BUFFER
  92.       inc eax            // i++
  93.       cmp eax, ebx       // compare i et NB_BUFFER
  94.       jb  fin1           // si i < NB_BUFFER goto fin1
  95.       xor eax, eax       // si i = NB_BUFFER i=0
  96. fin1: mov i,   eax
  97.      }
  98.    }
  99.   waveInReset(hWaveIn);
  100.   waveInClose(hWaveIn);
  101.   MessageBox(DlgMain,"FIN","RTA",MB_OK);
  102.   OpenedIn = FALSE;
  103.   return 0;
  104. }
  105. //****************************************************************************
  106. void CALLBACK waveOutProc(HWAVEOUT hwo, UINT uMsg, DWORD dwInstance, DWORD dwParam1, DWORD dwParam2)
  107. {
  108.   if (uMsg == WOM_OPEN)
  109.    {
  110.     MessageBox(DlgMain,"WOM_OPEN","RTA",MB_OK);
  111.     OpenedOut = TRUE;
  112.    }
  113.   if (uMsg == WOM_CLOSE)
  114.    {
  115.     MessageBox(DlgMain,"WOM_CLOSE","RTA",MB_OK);
  116.    }
  117.   if (uMsg == WOM_DONE)  
  118.    {
  119.     SetEvent(playDone);     
  120.    }
  121. }
  122. //****************************************************************************
  123. DWORD WINAPI OutThreadProc(LPVOID lpParam)
  124. {   
  125.   int i=0;
  126.   for(;;)
  127.    {
  128.     WaitForSingleObject(playDone, INFINITE);     
  129.     if (!(pWaveHeaderOut[i]->dwFlags & WHDR_DONE)) continue;
  130.      
  131.     copier_buffer(pWaveHeaderOut[i]->lpData, Buffer, BUFFER_SIZE); // reprendre bloc recu
  132.     waveOutUnprepareHeader(hWaveOut, pWaveHeaderOut[i], sizeof(WAVEHDR));
  133.     pWaveHeaderOut[i]->dwFlags = 0;
  134.     waveOutPrepareHeader(hWaveOut, pWaveHeaderOut[i], sizeof(WAVEHDR));
  135.     waveOutWrite(hWaveOut, pWaveHeaderOut[i], sizeof(WAVEHDR));
  136.     __asm
  137.      {
  138.       mov eax, i
  139.       mov ebx, NB_BUFFER
  140.       inc eax            // i++
  141.       cmp eax, ebx       // compare i et NB_BUFFER
  142.       jb  fin1           // si i < NB_BUFFER goto fin1
  143.       xor eax, eax       // si i = NB_BUFFER i=0
  144. fin1: mov i,   eax
  145.      }
  146.    }
  147.   return 0;
  148. }
  149. //****************************************************************************
  150. int openWaveDev()
  151. {
  152.   int i;
  153.   WAVEFORMATEX WaveFormat;
  154.    
  155.   WaveFormat.nChannels = CANAUX; // 1 pour mono 2 pour stereo
  156.   WaveFormat.wBitsPerSample = 8 * OCTETS; // 8 ou 16 bit
  157.   WaveFormat.nAvgBytesPerSec = SAMPLERATE * WaveFormat.nChannels * WaveFormat.wBitsPerSample/8; // nombre d'octets par seconde
  158.   WaveFormat.wFormatTag = 1; // 1 pour PCM
  159.   WaveFormat.nSamplesPerSec = SAMPLERATE; // frequence d'echantillonnage
  160.   WaveFormat.nBlockAlign = 1;
  161.   WaveFormat.cbSize = 0;
  162.   for(i=0; i<NB_BUFFER; i++)
  163.    {
  164.     pWaveHeaderIn[i] = (PWAVEHDR) malloc(sizeof(WAVEHDR));
  165.     pBufferIn[i] = (PBYTE)malloc(BUFFER_SIZE);
  166.    }
  167.    
  168.   InitializeCriticalSection(&CriticalSection);
  169.   if (waveInOpen(&hWaveIn, WAVE_MAPPER, &WaveFormat, (DWORD)waveInProc, (DWORD)&CriticalSection, CALLBACK_FUNCTION))
  170.    {
  171.     MessageBox(DlgMain,"Ouverture périphérique d'entrée impossible","RTA",MB_OK);
  172.     return 0;
  173.    }
  174.   for(i=0; i<NB_BUFFER; i++)
  175.    {
  176.     pWaveHeaderIn[i]->lpData          = (LPSTR)pBufferIn[i];
  177.     pWaveHeaderIn[i]->dwBufferLength  = BUFFER_SIZE;
  178.     pWaveHeaderIn[i]->dwBytesRecorded = 0;
  179.     pWaveHeaderIn[i]->dwUser          = 0;
  180.     pWaveHeaderIn[i]->dwFlags         = 0;
  181.     pWaveHeaderIn[i]->dwLoops         = 0;
  182.     pWaveHeaderIn[i]->lpNext          = NULL;
  183.     pWaveHeaderIn[i]->reserved        = 0;
  184.    }
  185.   for(i=0; i<NB_BUFFER; i++)
  186.    {
  187.     if (waveInPrepareHeader(hWaveIn, pWaveHeaderIn[i], sizeof(WAVEHDR)))
  188.      {
  189.       MessageBox(DlgMain,"Erreur PrepareHeader","RTA",MB_OK);
  190.       return 0;
  191.      }
  192.    }
  193.   for(i=0; i<NB_BUFFER; i++)
  194.    {
  195.     if (waveInAddBuffer(hWaveIn, pWaveHeaderIn[i], sizeof(WAVEHDR)))
  196.      {
  197.       MessageBox(DlgMain,"Erreur AddBuffer","RTA",MB_OK);
  198.       return 0;
  199.      }
  200.    }
  201.   ResetEvent(recordDone);
  202.   if (waveInStart(hWaveIn))
  203.    {
  204.     waveInClose(hWaveIn);
  205.    }
  206. //****
  207.   for(i=0; i<NB_BUFFER; i++)
  208.    {
  209.     pWaveHeaderOut[i] = (PWAVEHDR) malloc(sizeof(WAVEHDR));
  210.     pBufferOut[i] = (PBYTE)malloc(BUFFER_SIZE);
  211.    }
  212.   if (waveOutOpen(&hWaveOut, WAVE_MAPPER, &WaveFormat, (DWORD)waveOutProc, 0L, CALLBACK_FUNCTION))
  213.    {
  214.     MessageBox(DlgMain,"Ouverture périphérique de sortie impossible","RTA",MB_OK);
  215.     return 0;
  216.    }
  217.   for(i=0; i<NB_BUFFER; i++)
  218.    {
  219.     pWaveHeaderOut[i]->lpData          = (LPSTR)pBufferOut[i];
  220.     pWaveHeaderOut[i]->dwBufferLength  = BUFFER_SIZE;
  221.     pWaveHeaderOut[i]->dwBytesRecorded = 0;
  222.     pWaveHeaderOut[i]->dwUser          = 0;
  223.     pWaveHeaderOut[i]->dwFlags         = 0;
  224.     pWaveHeaderOut[i]->dwLoops         = 0;
  225.     pWaveHeaderOut[i]->lpNext          = NULL;
  226.     pWaveHeaderOut[i]->reserved        = 0;
  227.    }
  228.   for(i=0; i<NB_BUFFER; i++)
  229.    {
  230.     if (waveOutPrepareHeader(hWaveOut, pWaveHeaderOut[i], sizeof(WAVEHDR)))
  231.      {
  232.       MessageBox(DlgMain,"Erreur PrepareHeader","RTA",MB_OK);
  233.       return 0;
  234.      }
  235.    }
  236.   for(i=0; i<NB_BUFFER; i++)
  237.    {
  238.     if (waveOutWrite(hWaveOut, pWaveHeaderOut[i], sizeof(WAVEHDR)))
  239.      {
  240.       MessageBox(DlgMain,"Erreur OutWrite","RTA",MB_OK);
  241.       return 0;
  242.      }
  243.    }
  244.   ResetEvent(playDone);
  245. //*******
  246.   return 0;
  247. }
  248. //****************************************************************************
  249. BOOL CALLBACK DlgProc(HWND hWnd, UINT msg, WPARAM wParam, LPARAM lParam)
  250. {
  251.   if (msg == WM_INITDIALOG)
  252.    {
  253.     DlgMain = hWnd;
  254.     return 0;
  255.    }
  256.    
  257.    
  258.    
  259.    
  260.   if (msg == WM_COMMAND)
  261.    {
  262.     if (wParam == IDC_START)
  263.      {
  264.       if(OpenedIn == TRUE)
  265.        {       
  266.         MessageBox(DlgMain,"Déjà ouvert","RTA",MB_OK);
  267.         return 0;
  268.        }       
  269.       SetDlgItemText(DlgMain, IDC_STATIC1, "" );
  270.       recordDone = CreateEvent(0, FALSE, FALSE, 0);
  271.       ResetEvent(recordDone);
  272.       RecordHandle = CreateThread(NULL, 0, (LPTHREAD_START_ROUTINE) InThreadProc, NULL, 0, &threadidIn);
  273.       playDone = CreateEvent(0, FALSE, FALSE, 0);
  274.       ResetEvent(playDone);
  275.       PlayHandle = CreateThread(NULL, 0, (LPTHREAD_START_ROUTINE) OutThreadProc, NULL, 0, &threadidOut);
  276.       openWaveDev();       
  277.       return 0;
  278.      }
  279.    
  280.   if (wParam == IDCANCEL)
  281.      {
  282.       EndDialog(hWnd, 0);
  283.       return 0;
  284.      }
  285.    }
  286.   return 0;
  287. }
  288. //****************************************************************************
  289.  
  290. int APIENTRY WinMain(HINSTANCE hInstance, HINSTANCE hPrevInstance, LPSTR lpCmdLine, int nCmdShow)
  291. {
  292.   HANDLE hMutex;
  293.   hMutex = CreateMutex (NULL,FALSE, "RTA" );
  294.   if (GetLastError() == ERROR_ALREADY_EXISTS)
  295.    {
  296.     MessageBox(NULL,"Ce programme est déjà en service !","WARNING",MB_OK|MB_ICONEXCLAMATION);
  297.     return 0;
  298.    }
  299.   DialogBox(hInstance, MAKEINTRESOURCE(IDD_DIALOG1),HWND_DESKTOP,DlgProc);
  300.   return 0;
  301. }
  302. //****************************************************************************

Reply

Marsh Posté le 25-03-2007 à 17:59:14    

Pourquoi diable avoir mis de l'assembleur la dedans? Ensuite cherche du cote de l'instruction "movsb" et meme "rep movsb", ce que tu implementes via une boucle, le processeur sait deja le faire :-)
Et pour finir, un simple memcpy aurait fait l'affaire non?

Reply

Marsh Posté le 25-03-2007 à 21:05:02    

Ca me parait déjà plus clean et efficient, bon pour la boucle asm, en effet, un memcpy est aussi rapide donc pas besoin de passer par la, à la limite en simd (sse/sse2), histoire de copier plus de bloc par cycle.

Reply

Marsh Posté le 25-03-2007 à 21:59:59    

En effet ca a l'air plus efficient et il n'y a plus d'interruption de processus quand la Fenetre est reduite ou restaurée.  
Malgré tout, on dirait que le InThreadPRoc et OutThreadProc finissent par se marcher dessus au bout d'un cerain nombre de tours. Qqn a une solution ?

Reply

Marsh Posté le 25-03-2007 à 22:01:35    

J'aimerai mettre le traitement de WIM_DATA et WOM_DONE dans le mm thread.

Reply

Marsh Posté le    

Reply

Sujets relatifs:

Leave a Replay

Make sure you enter the(*)required information where indicate.HTML code is not allowed