M7350/qcom-opensource/mm-audio/audio-native/qdsp5/hdmi_dts.c

524 lines
14 KiB
C
Raw Normal View History

2024-09-09 08:52:07 +00:00
/* hdmi_dts.c - native PCM test application
*
* * Copyright (C) 2008 The Android Open Source Project
* Copyright (c) 2011-2012, The Linux Foundation. All rights reserved.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
#include <stdio.h>
#include <stdlib.h>
#include <fcntl.h>
#include <stdint.h>
#include <sys/mman.h>
#include <unistd.h>
#include <string.h>
#include <sys/ioctl.h>
#include <linux/msm_audio.h>
#include <pthread.h>
#include <errno.h>
#include "audiotest_def.h"
#include "control.h"
#include "iec_60958_61937.h"
#include "audio_parsers.h"
const char *dev_file_name;
static int quit, repeat;
static int pause_flag = 0;
static int play_state = 0;
static struct config_60958_61937 config_60958_61937;
static struct codec_61937_config codec_61937_config;
static struct audio_parser_codec_info audio_codec_info;
static unsigned char *dts_61937_burst;
static unsigned char *hdmi_non_l_rep_per;
static unsigned char *dts_frame;
#define DEV_FILE_NAME "/dev/msm_lpa_if_out"
#define MAX_FILE_SIZE 128
static int hdmi_dts_play(struct audtest_config *config)
{
struct msm_audio_config audio_config;
unsigned int dts_file_sz;
unsigned int actual_read_size;
unsigned char *dts_file_data;
FILE *fp;
int afd;
int sz;
unsigned int i;
int alsa_control;
unsigned char *ch;
int dev_id;
int max_dts_frame_sz, cur_dts_frame_sz;
char *device_name = "hdmi_pass_through";
int rc = 0;
unsigned int repetition_period = 0;
int exit_on_fail = 0;
unsigned int dma_buf_sz = 0;
unsigned int tmp = 0;
unsigned int frame_count = 0;
unsigned int actual_write_size = 0;
unsigned char silent_frame[11];
unsigned int silent_frame_count = 0;
struct msm_audio_config aud_config;
play_state = 1;
fprintf(stderr, "%s():\n", __func__);
/******************** read_file ***************************************/
while (repeat) {
fprintf(stderr, "Repeat %d\n", repeat);
fp = fopen(config->file_name,"rb");
if (fp == NULL) {
fprintf(stderr, "hdmi_dts : cannot open '%s'\n",
config->file_name);
return -1;
}
/* Get file size and reset file position */
fseek(fp, 0, SEEK_END);
dts_file_sz = ftell(fp);
fseek(fp, 0, SEEK_SET);
fprintf(stderr, "size of dts file %s = %u\n",
config->file_name, dts_file_sz);
dts_file_data = (unsigned char *)
(malloc(sizeof(unsigned char) * dts_file_sz));
if (dts_file_data == NULL) {
fprintf(stderr, "could not allocate memeory for dts file data."
" file length %u\n" , dts_file_sz);
fclose(fp);
return -1;
}
actual_read_size = fread((void*) dts_file_data, 1, dts_file_sz, fp);
if (actual_read_size != dts_file_sz) {
fprintf(stderr, "could not read DTS file %s\n",
config->file_name);
free(dts_file_data);
fclose(fp);
return -1;
}
max_dts_frame_sz = init_audio_parser(dts_file_data, dts_file_sz,
AUDIO_PARSER_CODEC_DTS);
dts_frame = (unsigned char *)
(malloc(sizeof(unsigned char) * max_dts_frame_sz));
if (dts_frame == NULL) {
fprintf(stderr, "could not allocate memeory for dts frame."
" frame size %u\n" ,max_dts_frame_sz);
free(dts_file_data);
fclose(fp);
return -1;
}
init_60958_61937_framer();
/********** end of read_file ******************************************/
dev_id = msm_get_device(device_name);
if (msm_en_device(dev_id,1)) {
fprintf(stderr, "could not enable device %s\n",
device_name);
exit_on_fail = 1;
goto error_en_alsa_dev;
}
afd = open(DEV_FILE_NAME, O_WRONLY);
if (afd < 0) {
fprintf(stderr, "cannot open audio device\n");
rc = -1;
exit_on_fail = 1;
goto error_open_dev;
}
audio_codec_info.codec_type = AUDIO_PARSER_CODEC_DTS;
rc = get_first_frame_info(&audio_codec_info);
if (rc < 0 ) {
fprintf(stderr, "Failed to get first frame indfo\n");
exit_on_fail = 1;
return -1;
}
config_60958_61937.codec_type = audio_codec_info.codec_config.dts_fr_info.dts_type;
fprintf(stderr, "dts type %d\n", config_60958_61937.codec_type);
rc = get_60958_61937_config(&config_60958_61937);
if(rc == -1) {
fprintf(stderr, "get_60958_61937_config failed\n");
rc = -1;
exit_on_fail = 1;
goto error_get_60958_61937_config;
}
dma_buf_sz = config_60958_61937.dma_buf_sz;
fprintf(stderr, "Dma buf_sz %d\n", dma_buf_sz);
dts_61937_burst = (unsigned char *)
(malloc(config_60958_61937.sz_61937_burst));
if (dts_61937_burst == NULL) {
fprintf(stderr, "could not allocate memory for dts 61937 burst."
" burst size %u\n" , config_60958_61937.sz_61937_burst);
rc = -1;
exit_on_fail = 1;
goto error_no_mem_dts_61937_burst;
}
hdmi_non_l_rep_per = (unsigned char *)
(malloc(dma_buf_sz));
if (hdmi_non_l_rep_per == NULL) {
fprintf(stderr, "could not allocate memeory for dts 60958 "
"rep per frames. dts 60958rep per frame size %u\n" ,
config_60958_61937.rep_per_60958);
rc = -1;
exit_on_fail = 1;
goto error_no_mem_hdmi_non_l_rep_per;
}
fprintf(stderr, "burst length %d , rep_per_60958 %d \n",
config_60958_61937.sz_61937_burst,
config_60958_61937.rep_per_60958);
aud_config.buffer_size = dma_buf_sz;
aud_config.sample_rate = audio_codec_info.codec_config.dts_fr_info.sample_rate;
fprintf(stderr, "Sample_rate = %d dma buf size = %d",
aud_config.sample_rate, dma_buf_sz);
if (ioctl(afd, AUDIO_SET_CONFIG, &aud_config)) {
fprintf(stderr, "could not set AUDIO_SET_IEC_CODEC_CONFIG\n");
rc = -1;
exit_on_fail = 1;
goto error_ioctl_audio_get_config;
}
if (ioctl(afd, AUDIO_GET_CONFIG, &audio_config)) {
fprintf(stderr, "could not get audio_config\n");
rc = -1;
exit_on_fail = 1;
goto error_ioctl_audio_get_config;
}
fprintf(stderr, "initiate_play: buffer_size=%d, buffer_count=%d\n",
audio_config.buffer_size, audio_config.buffer_count);
fprintf(stderr, "prefill: send Pause Fram\n");
sz = dma_buf_sz;
/******************** write_file ***************************************/
/* Driver requires two buffers to be prefilled before AUDIO_START */
for (i = 0; i < 2; i++) {
get_60958_61937_pause_burst(hdmi_non_l_rep_per,
config_60958_61937.rep_per_60958, &config_60958_61937);
if (write(afd, hdmi_non_l_rep_per, sz) != sz) {
fprintf(stderr, "could not write pause frame %d\n", i);
rc = -1;
exit_on_fail = 1;
goto error_dev_write;
}
}
fprintf(stderr, "start playback\n");
audio_codec_info.codec_type = AUDIO_PARSER_CODEC_DTS;
memset(&codec_61937_config, sizeof(struct codec_61937_config), 0);
rc = ioctl(afd, AUDIO_START, 0);
if (rc < 0 ) {
fprintf(stderr, "%s: Unable to start driver\n", __func__);
rc = 1;
exit_on_fail = 1;
goto error_ioctl_audio_start;
}
/* Sending 192 block of 60958 frames to AVR before sending the actual
* data so that AVR is synchronized to DTS bitstream configuration
*/
rc = get_audio_frame(dts_frame, max_dts_frame_sz,
&audio_codec_info);
if (rc < 0 ) {
fprintf(stderr, "%s: no more aduio frames\n",
__func__);
rc = 0;
break;
}
codec_61937_config.codec_type =
audio_codec_info.codec_config.dts_fr_info.dts_type;
codec_61937_config.codec_config.dts_fr_config.dts_fr_sz_8bit =
audio_codec_info.codec_config.dts_fr_info.dts_fr_sz_8bit;
codec_61937_config.codec_config.dts_fr_config.sample_rate =
audio_codec_info.codec_config.dts_fr_info.sample_rate;
codec_61937_config.codec_config.dts_fr_config.dts_type =
audio_codec_info.codec_config.dts_fr_info.dts_type;
codec_61937_config.codec_config.dts_fr_config.reverse_bytes =
audio_codec_info.codec_config.dts_fr_info.reverse_bytes;
cur_dts_frame_sz =
codec_61937_config.codec_config.dts_fr_config.dts_fr_sz_8bit;
get_61937_burst(dts_61937_burst,
config_60958_61937.sz_61937_burst,
dts_frame, cur_dts_frame_sz,
&codec_61937_config);
get_60958_frame(hdmi_non_l_rep_per,
config_60958_61937.rep_per_60958,
dts_61937_burst,
config_60958_61937.sz_61937_burst,
&codec_61937_config);
/* Writing 192 block i.e. 192 * 8 bytes */
if (write(afd, hdmi_non_l_rep_per,1536 ) != 1536) {
fprintf(stderr, "could not write pause frame %d\n", i);
rc = -1;
exit_on_fail = 1;
goto error_dev_write;
}
/* Sending silent frames for initial duration allowing AVR to
* synchronize and avoid data loss during synchronization
*/
get_silent_frame (silent_frame);
codec_61937_config.codec_type =
audio_codec_info.codec_config.dts_fr_info.dts_type;
codec_61937_config.codec_config.dts_fr_config.dts_fr_sz_8bit = 11;
codec_61937_config.codec_config.dts_fr_config.sample_rate =
audio_codec_info.codec_config.dts_fr_info.sample_rate;
codec_61937_config.codec_config.dts_fr_config.dts_type =
audio_codec_info.codec_config.dts_fr_info.dts_type;
codec_61937_config.codec_config.dts_fr_config.reverse_bytes =
audio_codec_info.codec_config.dts_fr_info.reverse_bytes;
cur_dts_frame_sz = 11;
get_61937_burst(dts_61937_burst,
config_60958_61937.sz_61937_burst,
silent_frame, cur_dts_frame_sz,
&codec_61937_config);
get_60958_frame(hdmi_non_l_rep_per,
config_60958_61937.rep_per_60958,
dts_61937_burst,
config_60958_61937.sz_61937_burst,
&codec_61937_config);
switch(audio_codec_info.codec_config.dts_fr_info.dts_type) {
case DTS_TYPE_1:
silent_frame_count = 100;
break;
case DTS_TYPE_2:
silent_frame_count = 50;
break;
case DTS_TYPE_3:
silent_frame_count = 25;
break;
default:
silent_frame_count = 0;
}
for (i = 0; i < silent_frame_count; i++) {
if (write(afd, hdmi_non_l_rep_per, sz) != sz) {
fprintf(stderr, "could not write pause frame %d\n", i);
rc = -1;
exit_on_fail = 1;
goto error_dev_write;
}
}
/* Sending actual data to AVR */
max_dts_frame_sz = init_audio_parser(dts_file_data, dts_file_sz,
AUDIO_PARSER_CODEC_DTS);
audio_codec_info.codec_type = AUDIO_PARSER_CODEC_DTS;
memset(&codec_61937_config, sizeof(struct codec_61937_config), 0);
for (;;) {
if(!pause_flag) {
rc = get_audio_frame(dts_frame, max_dts_frame_sz,
&audio_codec_info);
if (rc < 0 ) {
fprintf(stderr, "%s: no more aduio frames\n",
__func__);
rc = 0;
break;
}
codec_61937_config.codec_type =
audio_codec_info.codec_config.dts_fr_info.dts_type;
codec_61937_config.codec_config.dts_fr_config.dts_fr_sz_8bit =
audio_codec_info.codec_config.dts_fr_info.dts_fr_sz_8bit;
codec_61937_config.codec_config.dts_fr_config.sample_rate =
audio_codec_info.codec_config.dts_fr_info.sample_rate;
codec_61937_config.codec_config.dts_fr_config.dts_type =
audio_codec_info.codec_config.dts_fr_info.dts_type;
codec_61937_config.codec_config.dts_fr_config.reverse_bytes =
audio_codec_info.codec_config.dts_fr_info.reverse_bytes;
cur_dts_frame_sz =
codec_61937_config.codec_config.dts_fr_config.dts_fr_sz_8bit;
get_61937_burst(dts_61937_burst,
config_60958_61937.sz_61937_burst,
dts_frame, cur_dts_frame_sz,
&codec_61937_config);
get_60958_frame(hdmi_non_l_rep_per,
config_60958_61937.rep_per_60958,
dts_61937_burst,
config_60958_61937.sz_61937_burst,
&codec_61937_config);
} else {
get_60958_61937_pause_burst(hdmi_non_l_rep_per,
config_60958_61937.rep_per_60958,
&config_60958_61937);
}
if (write(afd, hdmi_non_l_rep_per, sz) != sz) {
fprintf(stderr, "could not write %s\n", DEV_FILE_NAME);
exit_on_fail = 1;
break;
}
}
error_ioctl_audio_start:
error_dev_write:
error_ioctl_audio_get_config:
free(hdmi_non_l_rep_per);
error_no_mem_hdmi_non_l_rep_per:
free(dts_61937_burst);
error_no_mem_dts_61937_burst:
error_get_60958_61937_config:
close(afd);
error_open_dev:
if (msm_en_device(dev_id, 0) < 0)
fprintf(stderr, "ERROR: could not disable %s\n",
device_name);
error_en_alsa_dev:
free(dts_frame);
fclose(fp);
deinit_60958_61937_framer();
free(dts_file_data);
repeat--;
if (repeat > 0) {
if(exit_on_fail == 1)
goto exit;
sleep(5);
}
}
fprintf(stderr, "End of playback\n");
exit:
play_state = 0;
return rc;
}
static void* hdmi_dts_thread(void* arg) {
struct audiotest_thread_context *context =
(struct audiotest_thread_context*) arg;
int ret_val;
ret_val = hdmi_dts_play(&context->config);
free(context->config.file_name);
free_context(context);
pthread_exit((void*) ret_val);
return NULL;
}
int hdmi_dts_read_params(void) {
struct audiotest_thread_context *context;
char *token;
int ret_val = 0;
if ((context = get_free_context()) == NULL) {
ret_val = -1;
} else {
if (!play_state) {
context->config.file_name = "/data/data.dts";
dev_file_name = "/dev/msm_lpa_if_out";
repeat = 1;
quit = 0;
pause_flag = 0;
}
token = strtok(NULL, " ");
while (token != NULL) {
if (!memcmp(token,"-id=", (sizeof("-id=")-1))) {
context->cxt_id = atoi(&token[sizeof("-id=") - 1]);
} else if (!memcmp(token, "-dev=",
(sizeof("-dev=") - 1))) {
dev_file_name = token + (sizeof("-dev=")-1);
} else if (!memcmp(token, "-pause=",
(sizeof("-pause=")-1))) {
pause_flag = atoi(&token[sizeof("-pause=") - 1]);
free_context(context);
return ret_val;
} else if (!memcmp(token, "-repeat=",
(sizeof("-repeat=") - 1))) {
repeat = atoi(&token[sizeof("-repeat=") - 1]);
} else {
context->config.file_name = (char*)malloc(MAX_FILE_SIZE);
if(!context->config.file_name)
return -1;
strlcpy(context->config.file_name, token, MAX_FILE_SIZE);
}
token = strtok(NULL, " ");
}
context->type = AUDIOTEST_TEST_MOD_PCM_DEC;
pthread_create( &context->thread, NULL,
hdmi_dts_thread, (void*) context);
}
return ret_val;
}
const char *hdmi_dts_help_txt =
"To Play dts file on hdmi: type \n"
"echo \"hdmi_dts path_of_file -id=xxx -dev=/dev/msm_lpa_if_out \" > tmp/audio_test \n"
"To Pause dts file on hdmi: type \n"
"echo \"hdmi_dts -pause=0/1 \n";
void hdmi_dts_help_menu(void) {
printf("%s\n", hdmi_dts_help_txt);
}