2018-11-13 07:41:21 +01:00
|
|
|
// SPDX-License-Identifier: GPL-2.0
|
|
|
|
//
|
|
|
|
// mapper.c - an interface of muxer/demuxer between buffer with data frames and
|
|
|
|
// formatted files.
|
|
|
|
//
|
|
|
|
// Copyright (c) 2018 Takashi Sakamoto <o-takashi@sakamocchi.jp>
|
|
|
|
//
|
|
|
|
// Licensed under the terms of the GNU General Public License, version 2.
|
|
|
|
|
|
|
|
#include "mapper.h"
|
|
|
|
#include "misc.h"
|
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
|
|
|
|
static const char *const mapper_type_labels[] = {
|
|
|
|
[MAPPER_TYPE_MUXER] = "muxer",
|
|
|
|
[MAPPER_TYPE_DEMUXER] = "demuxer",
|
|
|
|
};
|
|
|
|
|
|
|
|
static const char *const mapper_target_labels[] = {
|
2018-11-13 07:41:22 +01:00
|
|
|
[MAPPER_TARGET_SINGLE] = "single",
|
2018-11-13 07:41:23 +01:00
|
|
|
[MAPPER_TARGET_MULTIPLE] = "multiple",
|
2018-11-13 07:41:21 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
int mapper_context_init(struct mapper_context *mapper,
|
|
|
|
enum mapper_type type, unsigned int cntr_count,
|
|
|
|
unsigned int verbose)
|
|
|
|
{
|
|
|
|
const struct mapper_data *data = NULL;
|
|
|
|
|
|
|
|
assert(mapper);
|
|
|
|
assert(cntr_count > 0);
|
|
|
|
|
|
|
|
// Detect forgotten to destruct.
|
|
|
|
assert(mapper->private_data == NULL);
|
|
|
|
|
|
|
|
memset(mapper, 0, sizeof(*mapper));
|
|
|
|
|
2018-11-13 07:41:22 +01:00
|
|
|
if (type == MAPPER_TYPE_MUXER) {
|
|
|
|
if (cntr_count == 1) {
|
|
|
|
data = &mapper_muxer_single;
|
|
|
|
mapper->target = MAPPER_TARGET_SINGLE;
|
2018-11-13 07:41:23 +01:00
|
|
|
} else {
|
|
|
|
data = &mapper_muxer_multiple;
|
|
|
|
mapper->target = MAPPER_TARGET_MULTIPLE;
|
2018-11-13 07:41:22 +01:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (cntr_count == 1) {
|
|
|
|
data = &mapper_demuxer_single;
|
|
|
|
mapper->target = MAPPER_TARGET_SINGLE;
|
2018-11-13 07:41:23 +01:00
|
|
|
} else {
|
|
|
|
data = &mapper_demuxer_multiple;
|
|
|
|
mapper->target = MAPPER_TARGET_MULTIPLE;
|
2018-11-13 07:41:22 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-11-13 07:41:21 +01:00
|
|
|
mapper->ops = &data->ops;
|
|
|
|
mapper->type = type;
|
|
|
|
|
|
|
|
mapper->private_data = malloc(data->private_size);
|
|
|
|
if (mapper->private_data == NULL)
|
|
|
|
return -ENOMEM;
|
|
|
|
memset(mapper->private_data, 0, data->private_size);
|
|
|
|
|
|
|
|
mapper->cntr_count = cntr_count;
|
|
|
|
mapper->verbose = verbose;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int mapper_context_pre_process(struct mapper_context *mapper,
|
|
|
|
snd_pcm_access_t access,
|
|
|
|
unsigned int bytes_per_sample,
|
|
|
|
unsigned int samples_per_frame,
|
|
|
|
unsigned int frames_per_buffer,
|
|
|
|
struct container_context *cntrs)
|
|
|
|
{
|
|
|
|
int err;
|
|
|
|
|
|
|
|
assert(mapper);
|
|
|
|
assert(access >= SND_PCM_ACCESS_MMAP_INTERLEAVED);
|
|
|
|
assert(access <= SND_PCM_ACCESS_RW_NONINTERLEAVED);
|
|
|
|
assert(bytes_per_sample > 0);
|
|
|
|
assert(samples_per_frame > 0);
|
|
|
|
assert(cntrs);
|
|
|
|
|
2018-11-13 07:41:23 +01:00
|
|
|
// The purpose of multiple target is to mux/demux each channels to/from
|
|
|
|
// containers.
|
|
|
|
if (mapper->target == MAPPER_TARGET_MULTIPLE &&
|
|
|
|
samples_per_frame != mapper->cntr_count)
|
|
|
|
return -EINVAL;
|
|
|
|
|
2018-11-13 07:41:21 +01:00
|
|
|
mapper->access = access;
|
|
|
|
mapper->bytes_per_sample = bytes_per_sample;
|
|
|
|
mapper->samples_per_frame = samples_per_frame;
|
|
|
|
mapper->frames_per_buffer = frames_per_buffer;
|
|
|
|
|
|
|
|
err = mapper->ops->pre_process(mapper, cntrs, mapper->cntr_count);
|
|
|
|
if (err < 0)
|
|
|
|
return err;
|
|
|
|
|
|
|
|
if (mapper->verbose > 0) {
|
|
|
|
fprintf(stderr, "Mapper: %s\n",
|
|
|
|
mapper_type_labels[mapper->type]);
|
|
|
|
fprintf(stderr, " target: %s\n",
|
|
|
|
mapper_target_labels[mapper->target]);
|
|
|
|
fprintf(stderr, " access: %s\n",
|
|
|
|
snd_pcm_access_name(mapper->access));
|
|
|
|
fprintf(stderr, " bytes/sample: %u\n",
|
|
|
|
mapper->bytes_per_sample);
|
|
|
|
fprintf(stderr, " samples/frame: %u\n",
|
|
|
|
mapper->samples_per_frame);
|
|
|
|
fprintf(stderr, " frames/buffer: %lu\n",
|
|
|
|
mapper->frames_per_buffer);
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int mapper_context_process_frames(struct mapper_context *mapper,
|
|
|
|
void *frame_buffer,
|
|
|
|
unsigned int *frame_count,
|
|
|
|
struct container_context *cntrs)
|
|
|
|
{
|
|
|
|
assert(mapper);
|
|
|
|
assert(frame_buffer);
|
|
|
|
assert(frame_count);
|
|
|
|
assert(*frame_count <= mapper->frames_per_buffer);
|
|
|
|
assert(cntrs);
|
|
|
|
|
|
|
|
return mapper->ops->process_frames(mapper, frame_buffer, frame_count,
|
|
|
|
cntrs, mapper->cntr_count);
|
|
|
|
}
|
|
|
|
|
|
|
|
void mapper_context_post_process(struct mapper_context *mapper)
|
|
|
|
{
|
|
|
|
assert(mapper);
|
|
|
|
|
|
|
|
if (mapper->ops && mapper->ops->post_process)
|
|
|
|
mapper->ops->post_process(mapper);
|
|
|
|
}
|
|
|
|
|
|
|
|
void mapper_context_destroy(struct mapper_context *mapper)
|
|
|
|
{
|
|
|
|
assert(mapper);
|
|
|
|
|
|
|
|
if (mapper->private_data)
|
|
|
|
free(mapper->private_data);
|
|
|
|
mapper->private_data = NULL;
|
|
|
|
}
|