Older/MediaServer/libmov/source/mov-opus.c

78 lines
3.1 KiB
C
Raw Normal View History

2024-10-01 00:12:57 +08:00
#include "mov-internal.h"
#include <assert.h>
#include <stdlib.h>
#include <string.h>
#include <errno.h>
// http://www.opus-codec.org/docs/opus_in_isobmff.html
// 4.3.2 Opus Specific Box
/*
class ChannelMappingTable (unsigned int(8) OutputChannelCount){
unsigned int(8) StreamCount;
unsigned int(8) CoupledCount;
unsigned int(8 * OutputChannelCount) ChannelMapping;
}
aligned(8) class OpusSpecificBox extends Box('dOps'){
unsigned int(8) Version;
unsigned int(8) OutputChannelCount;
unsigned int(16) PreSkip;
unsigned int(32) InputSampleRate;
signed int(16) OutputGain;
unsigned int(8) ChannelMappingFamily;
if (ChannelMappingFamily != 0) {
ChannelMappingTable(OutputChannelCount);
}
}
*/
int mov_read_dops(struct mov_t* mov, const struct mov_box_t* box)
{
struct mov_track_t* track = mov->track;
struct mov_sample_entry_t* entry = track->stsd.current;
if(box->size >= 10)
{
if (entry->extra_data_size < box->size + 8)
{
void* p = realloc(entry->extra_data, (size_t)box->size + 8);
if (NULL == p) return -ENOMEM;
entry->extra_data = p;
}
memcpy(entry->extra_data, "OpusHead", 8);
entry->extra_data[8] = 1; // OpusHead version
mov_buffer_r8(&mov->io); // version 0
entry->extra_data[9] = mov_buffer_r8(&mov->io); // channel
entry->extra_data[11] = mov_buffer_r8(&mov->io); // PreSkip (MSB -> LSB)
entry->extra_data[10] = mov_buffer_r8(&mov->io);
entry->extra_data[15] = mov_buffer_r8(&mov->io); // InputSampleRate (LSB -> MSB)
entry->extra_data[14] = mov_buffer_r8(&mov->io);
entry->extra_data[13] = mov_buffer_r8(&mov->io);
entry->extra_data[12] = mov_buffer_r8(&mov->io);
entry->extra_data[17] = mov_buffer_r8(&mov->io); // OutputGain (LSB -> MSB)
entry->extra_data[16] = mov_buffer_r8(&mov->io);
mov_buffer_read(&mov->io, entry->extra_data + 18, (size_t)box->size - 10);
entry->extra_data_size = (int)box->size + 8;
}
return mov_buffer_error(&mov->io);
}
size_t mov_write_dops(const struct mov_t* mov)
{
const struct mov_track_t* track = mov->track;
const struct mov_sample_entry_t* entry = track->stsd.current;
if (entry->extra_data_size < 18)
return 0;
assert(0 == memcmp(entry->extra_data, "OpusHead", 8));
mov_buffer_w32(&mov->io, entry->extra_data_size); /* size */
mov_buffer_write(&mov->io, "dOps", 4);
mov_buffer_w8(&mov->io, 0); // The Version field shall be set to 0.
mov_buffer_w8(&mov->io, entry->extra_data[9]); // channel count
mov_buffer_w16(&mov->io, (entry->extra_data[11]<<8) | entry->extra_data[10]); // PreSkip (LSB -> MSB)
mov_buffer_w32(&mov->io, (entry->extra_data[15]<<8) | (entry->extra_data[14]<<8) | (entry->extra_data[13]<<8) | entry->extra_data[12]); // InputSampleRate (LSB -> MSB)
mov_buffer_w16(&mov->io, (entry->extra_data[17]<<8) | entry->extra_data[16]); // OutputGain (LSB -> MSB)
mov_buffer_write(&mov->io, entry->extra_data + 18, entry->extra_data_size - 18);
return entry->extra_data_size;
}