PDA

View Full Version : [C]Programmino audio:dente di sega


MinaVagante
16-01-2009, 16:31
Ciao a tutti,
ho un esercizio in cui devo generare una forma d'onda a dente di sega inserendo su riga di comando frequenza canle dx, ampiezza dx, frequenza canale sx, ampiezza sx...
Premetto che ho la soluzione, ma non ho idea da dove iniziare. Non ho un buon rapporto con i file audio :doh:
Allora le ampiezze sono esrpesse come percentuale rispetto all'ampiezza massima. Già ho difficoltà a capire cosa voglia dire ampiezza massima :confused:
Vi posto cosa non mi è chiaro nella soluzione (intanto una parte :D ) :
Lcyclesamples=1.0/atoi(argv[1])*44100;
Rcyclesamples=1.0/atoi(argv[2])*44100;

Lmaxampl=((1<<15)-1)/100.0*atoi(argv[3]);
Rmaxampl... .....;

Lstep=2*Lmaxampl/Lcyclesamples;
Rstep=... ....;

Lampl=-Lmaxampl;
Rampl=... ;

intanto questa parte introduttiva: argv[1] frequenza canale sinistro, argv[2]frequenza canale destro, argv[3] ampiezza percenvutale sx....
Allora, la prima istruzione presumo mi dia il numero di campioni per il canale sinistro la seconda analogamente per il destro, giusto???
Terza istruzione e quarta teoricamente dovrei trovare l'ampiezza dei due canali, e ((1<<15)-1) dovrebbe essere l'ampiezza massima ammissibile (che non capisco da dove esca, e soprattutto quel numero coincide con 1*10^15-1???
La quinta e la sesta istruzione proprio non le capisco :confused:
Aiuto, grazie a tutti

gugoXX
16-01-2009, 16:50
Se gia' non l'hai fatto, per prima cosa ti consiglierei di studiare cos'e' una forma d'onda periodica e quali sono le sue caratteristiche, indipdentemente dal fatto che sia un'onda sonora o meno.

Allora le ampiezze sono esrpesse come percentuale rispetto all'ampiezza massima. Già ho difficoltà a capire cosa voglia dire ampiezza massima :confused:

Fintantoche la membrana della cassa acustica resta ferma, non produce alcun suono.
Ad ogni istante temporale occorre muovere la membrana della cassa acustica dalla sua posizione di riposo. Questo movimento puo' essere positivo (la membrana esce di fuori) oppure negativo (la membrana rientra dentro).
Per pilotare il movimento occorre sollecitare la cassa con un segnale ad un determinato numero di volt.
Ogni posizione della membrana e' associata ad un ben preciso valore di volt, appunto + o -.
aumentando il volume si incrementa il range di questi volt. In pratica la membrana verra' spostata di piu', e il volume del suono emesso appunto sara' maggiore.

Fissato un determinato volume, vengono automaticamente fissati i volt massimi positivi (e specularmente negativi). possiamo suddividere lo spazio di movimento della membrana in tanti piccoli step.
Nelle applicazioni correnti lo spazio di lavoro e' modellato con un valore a 16 bit (i suoni generati sono tipicamente appunto a 16 bit).
Pertanto 2^16 diverse posizioni della membrana della cassa.
Ovvero, 2^15 positive e 2^15 negative.
Pertanto, qual e' il valore massimo positivo ammissibile? 2^15 -1, ovvero in binario
0111 1111 1111 1111b

Come si calcola 2^15 - 1 in C? (1<<15) -1
Che e' proprio quel valore la'.


Vi posto cosa non mi è chiaro nella soluzione (intanto una parte :D ) :
Lcyclesamples=1.0/atoi(argv[1])*44100;
Rcyclesamples=1.0/atoi(argv[2])*44100;

Lmaxampl=((1<<15)-1)/100.0*atoi(argv[3]);
Rmaxampl... .....;

Lstep=2*Lmaxampl/Lcyclesamples;
Rstep=... ....;

Lampl=-Lmaxampl;
Rampl=... ;

intanto questa parte introduttiva: argv[1] frequenza canale sinistro, argv[2]frequenza canale destro, argv[3] ampiezza percenvutale sx....
Allora, la prima istruzione presumo mi dia il numero di campioni per il canale sinistro la seconda analogamente per il destro, giusto???
Terza istruzione e quarta teoricamente dovrei trovare l'ampiezza dei due canali, e ((1<<15)-1) dovrebbe essere l'ampiezza massima ammissibile (che non capisco da dove esca, e soprattutto quel numero coincide con 1*10^15-1???
La quinta e la sesta istruzione proprio non le capisco :confused:
Aiuto, grazie a tutti

lo step, per questo dente di sega, e' di quanto occorre spostare la membrana (incrementare l'ampiezza del suono) dalla posizione precedente in ciascuna unita' temporale, per poter raggiungere il massimo desiderato proprio dopo Lcyclesamples, e quindi generare un suono dente di sega alla frequenza voluta.

MinaVagante
16-01-2009, 17:17
mh un momento :D
C'è una cosa che non ho capito però: da quello che hai scritto, mi sembra di aver capito che Lcyclesamples rappresenti il numero di campioni all'interno del periodo, o no??Ma Lcyclesamples non è il numero totale di campioni?? In quanto ho 1/Hz*(campioni/sec)=(campioni)??

gugoXX
16-01-2009, 18:09
mh un momento :D
C'è una cosa che non ho capito però: da quello che hai scritto, mi sembra di aver capito che Lcyclesamples rappresenti il numero di campioni all'interno del periodo, o no??Ma Lcyclesamples non è il numero totale di campioni?? In quanto ho 1/Hz*(campioni/sec)=(campioni)??

LCyclesSamples dovrebbe essere il numero di campioni necessario per passare da 0 all'ampiezza massima.
44.100/X.

Quando X vale 1, significa che ci mettero' 44.100 campioni (1 secondo)
quando X vale 2, singnifica che ci mettero' 22.050 campioni (1/2 secondo)
Quando X vale 4, ci mettero' 11.025 campioni. (1/4 secondo)

quindi X, parametro di ingresso e' la frequenza, e tanto piu' sara' altra tanto piu' acuto sara' il segnale e quindi tanto piu' ripido sara' il segnale a dente di sega.
E LCyclesSamples e' la distanza tra 2 picchi analoghi del segnale, ovvero la lunghezza d'onda.

MinaVagante
16-01-2009, 18:34
Ah ok, quindi dovrebbero essere i campioni a semiperiodo.
Non ho capito il perchè però: 44.100 non è la frequenza di campionamento del segnale audio?? Nel senso: se io dovessi copiare un file audio in un nuovo file, allora devo campionarlo con una frequenza pari a 44.100 per non perdere informazioni. Ma qui cosa centra??Non ho mica un segnale da campionare :mbe:

gugoXX
16-01-2009, 21:33
44.100Hz e' una delle frequenze di campionamento, forse la piu' comune.
Quando devi riprodurre un suono campionato a 44.100Hz dovrai farlo di nuovo a 44.100Hz.
In fase di riproduzione e' in pratica quanti comandi al secondo darai alla membrana.
In fase di campionamento e' quanti segnali al secondo leggerai dal microfono.

Comunque e' errato dire che a 44.100Hz non si perdono informazioni.
Si perdono informazioni che sono non udibili dall'orecchio umano, oppure non si degrada il segnale audio piu' di quanto lo fosse prima, quando ascoltato da un orecchio umano.

MinaVagante
17-01-2009, 08:10
si si in quel senso si perdono informazioni grazie :D
Quello che non capisco è perchè devo utilizzare 44.100 se non ho campionato nessun suono, nel senso, avrei potuto usare un'altra frequenza??? Il campionamento non lo si usa quando è già presente un suono e ad esempio voglio riprodurlo,ma qui devo crearlo io del tutto, non so se sono riusctio a esporre il mio dubbio. Posto la seconda parte del programma e approfitto della tua disponibilità e conoscenza :D Intanto grazie
seconda parte
Lcounter=-Rcounter=1;
for(i=0;i<wave.numofstereosamples;i++)
{ //Canale sinistro
if(i>Lcounter*Lcyclesamples)
{ Lampl=-Lmaxampl;
SAMPLE(wave,left,i)=Lampl;
Lcounter++;
}
else
{ Lampl+=Lstep;
SAMPLE(wave,left,i)=Lampl;
}
....
}

Analogamente il canale di destra. Numstereosamples sono i campioni del file che devo creare. In questo for non riesco a capire molto :confused:

gugoXX
17-01-2009, 09:19
44.100Hz e' la massima frequenza di campionamento/riproduzione su molte delle nostre schede sonore, ed e' stata scelta da questo esempio perche' praticamente tutte le schede sonore la posseggono.
Essendo la massima sara' quella che dara' la resa sonora migliore.

Quello loop serve proprio per decidere, ad ogni istante, qual e' la posizione della membrana.
numofstereosamples e' la lunghezza dell'intero campione sonoro, quindi parametro relazionato con il tempo totale di esecuzione
Cosa c'e' dentro l'if serve per fare l'operazione di modulo per azzerare di nuovo la posizione della membrana, ovvero ritornare a 0 con il valore del campione da spedire alla scheda sonora.
Non e' il massimo della chiarezza, e poteva essere scritto un po' meglio.


int modarr=0;
for(i=0;i<wave.numofstereosamples;i++,modarr++)
{
//Canale sinistro
if(modarr == Lcyclesamples)
{
modarr=0;
Lampl=0; // Qui riporto la membrana in posizione di riposo
}
else
{
Lampl+=Lstep; // Qui sposto la membrana di un pochino
}
SAMPLE(wave,left,i)=Lampl;

....
}

MinaVagante
17-01-2009, 09:34
dal tuo chiarissimo, ma nell'esempio che ti ho riportato non capisco la condizione dell'if, che da quello che ho capito,è la condizione che fa ritornare a zero il segnale doipo aver raggiunto il massimo no?

gugoXX
17-01-2009, 09:45
dal tuo chiarissimo, ma nell'esempio che ti ho riportato non capisco la condizione dell'if, che da quello che ho capito,è la condizione che fa ritornare a zero il segnale doipo aver raggiunto il massimo no?

In realta' fa ritornare a 0 il campione ogni volta che il contatore totale dei samples riprodotti finora e' pari alla prossimo multiplo di Lcyclesamples, ovvero la lunghezza d'onda
(per questo puo' essere riscritto piu' chiaramente, dato che a noi non interessa sapere quante volte e' stata riprodotto il dente di sega, ma e' sufficiente tenere conto di dove siamo arrivati nell'istanza corrente)

MinaVagante
17-01-2009, 11:26
Ma nel tuo non viene fatta la parte negativa no???

gugoXX
17-01-2009, 19:07
Ma nel tuo non viene fatta la parte negativa no???

No, e penso neppure nella sua.