| /* Copyright (c) 2017 Google Inc. |
| Written by Andrew Allen */ |
| /* |
| Redistribution and use in source and binary forms, with or without |
| modification, are permitted provided that the following conditions |
| are met: |
| |
| - Redistributions of source code must retain the above copyright |
| notice, this list of conditions and the following disclaimer. |
| |
| - Redistributions in binary form must reproduce the above copyright |
| notice, this list of conditions and the following disclaimer in the |
| documentation and/or other materials provided with the distribution. |
| |
| THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS |
| ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT |
| LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR |
| A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER |
| OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, |
| EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, |
| PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR |
| PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF |
| LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING |
| NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS |
| SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. |
| */ |
| |
| #ifdef HAVE_CONFIG_H |
| #include "config.h" |
| #endif |
| |
| #include "mathops.h" |
| #include "os_support.h" |
| #include "opus_private.h" |
| #include "opus_defines.h" |
| #include "opus_projection.h" |
| #include "opus_multistream.h" |
| #include "mapping_matrix.h" |
| #include "stack_alloc.h" |
| |
| #ifdef ENABLE_EXPERIMENTAL_AMBISONICS |
| |
| struct OpusProjectionDecoder |
| { |
| opus_int32 demixing_matrix_size_in_bytes; |
| /* Encoder states go here */ |
| }; |
| |
| #if !defined(DISABLE_FLOAT_API) |
| static void opus_projection_copy_channel_out_float( |
| void *dst, |
| int dst_stride, |
| int dst_channel, |
| const opus_val16 *src, |
| int src_stride, |
| int frame_size, |
| void *user_data) |
| { |
| float *float_dst; |
| const MappingMatrix *matrix; |
| float_dst = (float *)dst; |
| matrix = (const MappingMatrix *)user_data; |
| |
| if (dst_channel == 0) |
| OPUS_CLEAR(float_dst, frame_size * dst_stride); |
| |
| if (src != NULL) |
| mapping_matrix_multiply_channel_out_float(matrix, src, dst_channel, |
| src_stride, float_dst, dst_stride, frame_size); |
| } |
| #endif |
| |
| static void opus_projection_copy_channel_out_short( |
| void *dst, |
| int dst_stride, |
| int dst_channel, |
| const opus_val16 *src, |
| int src_stride, |
| int frame_size, |
| void *user_data) |
| { |
| opus_int16 *short_dst; |
| const MappingMatrix *matrix; |
| short_dst = (opus_int16 *)dst; |
| matrix = (const MappingMatrix *)user_data; |
| if (dst_channel == 0) |
| OPUS_CLEAR(short_dst, frame_size * dst_stride); |
| |
| if (src != NULL) |
| mapping_matrix_multiply_channel_out_short(matrix, src, dst_channel, |
| src_stride, short_dst, dst_stride, frame_size); |
| } |
| |
| static MappingMatrix *get_demixing_matrix(OpusProjectionDecoder *st) |
| { |
| return (MappingMatrix*)((char*)st + align(sizeof(OpusProjectionDecoder))); |
| } |
| |
| static OpusMSDecoder *get_multistream_decoder(OpusProjectionDecoder *st) |
| { |
| return (OpusMSDecoder*)((char*)st + align(sizeof(OpusProjectionDecoder) + |
| st->demixing_matrix_size_in_bytes)); |
| } |
| |
| opus_int32 opus_projection_decoder_get_size(int channels, int streams, |
| int coupled_streams) |
| { |
| opus_int32 matrix_size; |
| opus_int32 decoder_size; |
| |
| matrix_size = |
| mapping_matrix_get_size(streams + coupled_streams, channels); |
| if (!matrix_size) |
| return 0; |
| |
| decoder_size = opus_multistream_decoder_get_size(streams, coupled_streams); |
| if (!decoder_size) |
| return 0; |
| |
| return align(sizeof(OpusProjectionDecoder)) + matrix_size + decoder_size; |
| } |
| |
| int opus_projection_decoder_init(OpusProjectionDecoder *st, opus_int32 Fs, |
| int channels, int streams, int coupled_streams, |
| unsigned char *demixing_matrix, opus_int32 demixing_matrix_size) |
| { |
| int nb_input_streams; |
| opus_int32 expected_matrix_size; |
| int i, ret; |
| unsigned char mapping[255]; |
| VARDECL(opus_int16, buf); |
| ALLOC_STACK; |
| |
| /* Verify supplied matrix size. */ |
| nb_input_streams = streams + coupled_streams; |
| expected_matrix_size = nb_input_streams * channels * sizeof(opus_int16); |
| if (expected_matrix_size != demixing_matrix_size) |
| { |
| RESTORE_STACK; |
| return OPUS_BAD_ARG; |
| } |
| |
| /* Convert demixing matrix input into internal format. */ |
| ALLOC(buf, nb_input_streams * channels, opus_int16); |
| for (i = 0; i < nb_input_streams * channels; i++) |
| { |
| int s = demixing_matrix[2*i + 1] << 8 | demixing_matrix[2*i]; |
| s = ((s & 0xFFFF) ^ 0x8000) - 0x8000; |
| buf[i] = (opus_int16)s; |
| } |
| |
| /* Assign demixing matrix. */ |
| st->demixing_matrix_size_in_bytes = |
| mapping_matrix_get_size(channels, nb_input_streams); |
| if (!st->demixing_matrix_size_in_bytes) |
| { |
| RESTORE_STACK; |
| return OPUS_BAD_ARG; |
| } |
| |
| mapping_matrix_init(get_demixing_matrix(st), channels, nb_input_streams, 0, |
| buf, demixing_matrix_size); |
| |
| /* Set trivial mapping so each input channel pairs with a matrix column. */ |
| for (i = 0; i < channels; i++) |
| mapping[i] = i; |
| |
| ret = opus_multistream_decoder_init( |
| get_multistream_decoder(st), Fs, channels, streams, coupled_streams, mapping); |
| RESTORE_STACK; |
| return ret; |
| } |
| |
| OpusProjectionDecoder *opus_projection_decoder_create( |
| opus_int32 Fs, int channels, int streams, int coupled_streams, |
| unsigned char *demixing_matrix, opus_int32 demixing_matrix_size, int *error) |
| { |
| int size; |
| int ret; |
| OpusProjectionDecoder *st; |
| |
| /* Allocate space for the projection decoder. */ |
| size = opus_projection_decoder_get_size(channels, streams, coupled_streams); |
| if (!size) { |
| if (error) |
| *error = OPUS_ALLOC_FAIL; |
| return NULL; |
| } |
| st = (OpusProjectionDecoder *)opus_alloc(size); |
| if (!st) |
| { |
| if (error) |
| *error = OPUS_ALLOC_FAIL; |
| return NULL; |
| } |
| |
| /* Initialize projection decoder with provided settings. */ |
| ret = opus_projection_decoder_init(st, Fs, channels, streams, coupled_streams, |
| demixing_matrix, demixing_matrix_size); |
| if (ret != OPUS_OK) |
| { |
| opus_free(st); |
| st = NULL; |
| } |
| if (error) |
| *error = ret; |
| return st; |
| } |
| |
| #ifdef FIXED_POINT |
| int opus_projection_decode(OpusProjectionDecoder *st, const unsigned char *data, |
| opus_int32 len, opus_int16 *pcm, int frame_size, |
| int decode_fec) |
| { |
| return opus_multistream_decode_native(get_multistream_decoder(st), data, len, |
| pcm, opus_projection_copy_channel_out_short, frame_size, decode_fec, 0, |
| get_demixing_matrix(st)); |
| } |
| #else |
| int opus_projection_decode(OpusProjectionDecoder *st, const unsigned char *data, |
| opus_int32 len, opus_int16 *pcm, int frame_size, |
| int decode_fec) |
| { |
| return opus_multistream_decode_native(get_multistream_decoder(st), data, len, |
| pcm, opus_projection_copy_channel_out_short, frame_size, decode_fec, 1, |
| get_demixing_matrix(st)); |
| } |
| #endif |
| |
| #ifndef DISABLE_FLOAT_API |
| int opus_projection_decode_float(OpusProjectionDecoder *st, const unsigned char *data, |
| opus_int32 len, float *pcm, int frame_size, int decode_fec) |
| { |
| return opus_multistream_decode_native(get_multistream_decoder(st), data, len, |
| pcm, opus_projection_copy_channel_out_float, frame_size, decode_fec, 0, |
| get_demixing_matrix(st)); |
| } |
| #endif |
| |
| int opus_projection_decoder_ctl(OpusProjectionDecoder *st, int request, ...) |
| { |
| va_list ap; |
| int ret = OPUS_OK; |
| |
| va_start(ap, request); |
| ret = opus_multistream_decoder_ctl_va_list(get_multistream_decoder(st), |
| request, ap); |
| va_end(ap); |
| return ret; |
| } |
| |
| void opus_projection_decoder_destroy(OpusProjectionDecoder *st) |
| { |
| opus_free(st); |
| } |
| |
| #else /* ENABLE_EXPERIMENTAL_AMBISONICS */ |
| |
| opus_int32 opus_projection_decoder_get_size( |
| int channels, |
| int streams, |
| int coupled_streams) |
| { |
| (void)channels; |
| (void)streams; |
| (void)coupled_streams; |
| return OPUS_UNIMPLEMENTED; |
| } |
| |
| OpusProjectionDecoder *opus_projection_decoder_create( |
| opus_int32 Fs, |
| int channels, |
| int streams, |
| int coupled_streams, |
| unsigned char *demixing_matrix, |
| opus_int32 demixing_matrix_size, |
| int *error) |
| { |
| (void)Fs; |
| (void)channels; |
| (void)streams; |
| (void)coupled_streams; |
| (void)demixing_matrix; |
| (void)demixing_matrix_size; |
| if (error) *error = OPUS_UNIMPLEMENTED; |
| return NULL; |
| } |
| |
| int opus_projection_decoder_init( |
| OpusProjectionDecoder *st, |
| opus_int32 Fs, |
| int channels, |
| int streams, |
| int coupled_streams, |
| unsigned char *demixing_matrix, |
| opus_int32 demixing_matrix_size) |
| { |
| (void)st; |
| (void)Fs; |
| (void)channels; |
| (void)streams; |
| (void)coupled_streams; |
| (void)demixing_matrix; |
| (void)demixing_matrix_size; |
| return OPUS_UNIMPLEMENTED; |
| } |
| |
| int opus_projection_decode( |
| OpusProjectionDecoder *st, |
| const unsigned char *data, |
| opus_int32 len, |
| opus_int16 *pcm, |
| int frame_size, |
| int decode_fec) |
| { |
| (void)st; |
| (void)data; |
| (void)len; |
| (void)pcm; |
| (void)frame_size; |
| (void)decode_fec; |
| return OPUS_UNIMPLEMENTED; |
| } |
| |
| int opus_projection_decode_float( |
| OpusProjectionDecoder *st, |
| const unsigned char *data, |
| opus_int32 len, |
| float *pcm, |
| int frame_size, |
| int decode_fec) |
| { |
| (void)st; |
| (void)data; |
| (void)len; |
| (void)pcm; |
| (void)frame_size; |
| (void)decode_fec; |
| return OPUS_UNIMPLEMENTED; |
| } |
| |
| int opus_projection_decoder_ctl( |
| OpusProjectionDecoder *st, |
| int request, |
| ...) |
| { |
| (void)st; |
| (void)request; |
| return OPUS_UNIMPLEMENTED; |
| } |
| |
| void opus_projection_decoder_destroy( |
| OpusProjectionDecoder *st) |
| { |
| (void)st; |
| } |
| |
| #endif /* ENABLE_EXPERIMENTAL_AMBISONICS */ |