0

ここに投稿された私の質問に続いて、 をさらに調査しLAMEgdbところ、常に 576​​ 個のサンプルしか入力pcmファイルから読み取られないことが明らかになりました。ただし、これらのサンプルは、2304 の 1D または 2*1152 の 2D 配列を生成するために使用されます。私が理解できないのはこの変換です。関連するコードをここに投稿し、説明を求めます。コードの一部を説明し、質問をコードに挿入しました???

  samples_read = read_samples_pcm(global.musicin, insamp, num_channels * samples_to_read);
  /* global.musicin = FILE pointer to input pcm file.
   * insamp is a 1D int array of [2*1152] size.
   * num_channels = 1. samples_to_read = 576. */
    if (samples_read < 0) {
        return samples_read;
    }
    p = insamp + samples_read;
    samples_read /= num_channels;  /*samples_read = 576 after this step. prior to this it is optimized out.*/
    if (buffer != NULL) { /* output to int buffer */
        if (num_channels == 2) {
            for (i = samples_read; --i >= 0;) {
                buffer[1][i] = *--p;
                buffer[0][i] = *--p;
            }
        }
        else if (num_channels == 1) {
            memset(buffer[1], 0, samples_read * sizeof(int));
            for (i = samples_read; --i >= 0;) {
                buffer[0][i] = *--p;
                /*??? Is this line a fancy way of copying first 576 samples of insamp into buffer[0]?*/
            }
        }
        else
            assert(0);
    }

関数read_samples_pcmは次のように定義されます。

static int
read_samples_pcm(FILE * const musicin, int sample_buffer[2304], int samples_to_read)
{
    int     samples_read;

    samples_read = sf_read_int((SNDFILE *) musicin, sample_buffer, samples_to_read);

#if 0
    switch (global.pcmbitwidth) {
    case 8:
        for (i = 0; i < samples_read; i++)
            sample_buffer[i] <<= (8 * sizeof(int) - 8);
        break;
    case 16:  /*input pcm is 16 bit say.*/
        for (i = 0; i < samples_read; i++)
            sample_buffer[i] <<= (8 * sizeof(int) - 16); /*??? what is going on? */
        break;
    case 24:
        for (i = 0; i < samples_read; i++)
            sample_buffer[i] <<= (8 * sizeof(int) - 24);
        break;
    case 32:
        break;
    default:
        if (silent < 10) {
            error_printf("Only 8, 16, 24 and 32 bit input files supported \n");
        }
        exit(1);
    }
#endif

    return samples_read;
}
4

1 に答える 1

1

/ ??? この行は、insamp の最初の 576 サンプルを buffer[0] にコピーするための凝った方法ですか? /

基本的に、はい。通常の開始 - >終了の順序ではなく、データの終了 - >開始をコピーしているだけです。よくわかりませんが、メモリキャッシュのパフォーマンス上の理由からこれを好む人もいると思います。

/*??? 何が起こっている?*/

16 ビットの入力を 32 ビットの数値にスケーリングしています。それを見てみましょう:

sample_buffer[i] <<= (8 * sizeof(int) - 16);

sizeof(int) が 4 バイトの場合、8 * 4 - 16 = 16 です。したがって、16 ビットの数値を 16 だけ左にシフトして、32 ビットの数値にスケーリングします。sizeof(int) が 8 バイトの場合、8 * 8 - 16 = 48 です。int が 64 ビットの場合、48 ビットの左シフトが必要です。

ここであなたのすべての質問に答えたかどうかわかりません。お気軽にフォローしてください。

于 2013-01-31T02:56:57.997 に答える